diff --git a/.dockerignore b/.dockerignore index f369e1e7..10ec6b5a 100644 --- a/.dockerignore +++ b/.dockerignore @@ -1,33 +1,8 @@ -# Ignore secure files (permission issues) -backend/secure_files -backend/data -backend/uploads -backend/logs - -# Git +node_modules +npm-debug.log +release .git .gitignore - -# Node modules -**/node_modules - -# Build artifacts -**/dist -**/*.log - -# IDE -.vscode -.idea - -# Env files -.env -.env.local - -# Docker -docker-compose*.yml -!docker-compose.yml - -# Docs -docs/ *.md -!README.md +.env +.env.* diff --git a/.env.default b/.env.default deleted file mode 100644 index 98b34e1c..00000000 --- a/.env.default +++ /dev/null @@ -1,150 +0,0 @@ -# ============================================================================ -# ClaraVerse Environment Configuration - DEFAULT TEMPLATE -# ============================================================================ -# This file contains sensible defaults for quick setup. -# The quickstart.sh/quickstart.bat scripts will: -# 1. Copy this file to .env -# 2. Auto-generate secure encryption keys -# 3. You're ready to go! -# -# For manual setup: -# 1. Copy this file: cp .env.default .env -# 2. Generate keys: openssl rand -hex 32 (for ENCRYPTION_MASTER_KEY) -# openssl rand -hex 64 (for JWT_SECRET) -# 3. Replace the placeholder values below -# 4. Run: docker compose up -d -# -# For advanced configuration, see .env.example for detailed documentation -# ============================================================================ - -# ================================ -# AUTO-GENERATED KEYS -# ================================ -# These will be automatically generated by quickstart scripts -# DO NOT set these manually unless you know what you're doing - -# Encryption key for user data (conversations, credentials) -# WARNING: Losing this key = losing access to encrypted data! -ENCRYPTION_MASTER_KEY=auto-generated-on-first-run - -# JWT secret for authentication tokens -# Used to sign and verify JWT tokens -JWT_SECRET=auto-generated-on-first-run - -# ================================ -# ENVIRONMENT MODE -# ================================ -# Options: development, production -ENVIRONMENT=development - -# ================================ -# FRONTEND URLs (for building) -# ================================ -# Development defaults - work out-of-box for local Docker setup -VITE_API_BASE_URL=http://localhost:3001 -VITE_WS_URL=ws://localhost:3001 -VITE_APP_NAME=ClaraVerse -VITE_APP_VERSION=2.0.0 -VITE_ENABLE_ANALYTICS=false - -# ================================ -# BACKEND CONFIGURATION -# ================================ -# CORS: Comma-separated allowed origins -# These defaults work for local development -ALLOWED_ORIGINS=http://localhost,http://localhost:80,http://localhost:5173,http://localhost:5174,http://localhost:3000,http://localhost:8080 - -# Frontend URL (for payment redirects) -FRONTEND_URL=http://localhost:80 - -# Backend public URL (for generating download URLs) -BACKEND_URL=http://localhost:3001 - -# ================================ -# DATABASE CONFIGURATION -# ================================ -# MySQL - for providers, models, and capabilities -MYSQL_ROOT_PASSWORD=claraverse_root_2024 -MYSQL_PASSWORD=claraverse_pass_2024 - -# MongoDB - for conversations, workflows, and data persistence -MONGODB_URI=mongodb://mongodb:27017/claraverse - -# Redis - for job scheduling and WebSocket pub/sub -REDIS_URL=redis://redis:6379 - -# ================================ -# SEARCH ENGINE -# ================================ -# SearXNG (local search engine, no API key needed) -SEARXNG_URLS=http://searxng:8080 - -# ================================ -# CODE EXECUTION (E2B) -# ================================ -# E2B Local Mode - No API key needed! -# This runs code execution locally using Docker -E2B_MODE=local -E2B_LOCAL_USE_DOCKER=true -E2B_SANDBOX_POOL_SIZE=3 -E2B_EXECUTION_TIMEOUT=30000 -E2B_RATE_LIMIT_PER_MIN=20 - -# ================================ -# OPTIONAL SERVICES -# ================================ -# These are optional - leave empty to skip -# You can add these later in Settings UI - -# Supabase Authentication (optional - for production auth) -# Get from: https://app.supabase.com/project/_/settings/api -VITE_SUPABASE_URL= -VITE_SUPABASE_ANON_KEY= -SUPABASE_URL= -SUPABASE_KEY= - -# E2B Cloud Mode (optional - if you have API key) -# Get from: https://e2b.dev/dashboard -# Comment out E2B_MODE=local above and uncomment these: -# E2B_MODE=production -# E2B_API_KEY= - -# Composio Integrations (optional) -# Get from: https://app.composio.dev -COMPOSIO_API_KEY= -COMPOSIO_GOOGLESHEETS_AUTH_CONFIG_ID= -COMPOSIO_GMAIL_AUTH_CONFIG_ID= - -# Cloudflare Turnstile (optional - for bot protection) -# Get from: https://dash.cloudflare.com/turnstile -VITE_TURNSTILE_SITE_KEY= - -# ================================ -# ADMIN CONFIGURATION -# ================================ -# Comma-separated list of user IDs with superadmin access -# Leave empty - first registered user becomes admin automatically -SUPERADMIN_USER_IDS= - -# Development API key for testing (only works when ENVIRONMENT != production) -DEV_API_KEY=claraverse-dev-key-2024 - -# JWT Token Expiry -JWT_ACCESS_TOKEN_EXPIRY=15m -JWT_REFRESH_TOKEN_EXPIRY=168h - -# ================================ -# RATE LIMITING (requests per minute) -# ================================ -# Protect your instance from abuse with these sensible defaults -RATE_LIMIT_GLOBAL_API=200 -RATE_LIMIT_PUBLIC_READ=120 -RATE_LIMIT_AUTHENTICATED=60 -RATE_LIMIT_WEBSOCKET=20 -RATE_LIMIT_IMAGE_PROXY=60 - -# ================================ -# DOCKER BUILD CONFIGURATION -# ================================ -# Skip tests during Docker build for faster builds -SKIP_TESTS=false diff --git a/.env.example b/.env.example index 45de836e..a198cc96 100644 --- a/.env.example +++ b/.env.example @@ -1,127 +1,6 @@ -# ============================================================================ -# ClaraVerse Environment Configuration -# ============================================================================ -# This is the SINGLE SOURCE OF TRUTH for Docker Compose deployments. -# Docker Compose automatically reads this file. -# -# Setup Instructions: -# 1. Copy this file: cp .env.example .env -# 2. Fill in required values (marked with TODO) -# 3. Run: ./dev-docker.sh up -# -# For production: -# 1. Copy this file to your server -# 2. Update values for production -# 3. Run: docker-compose -f docker-compose.yml -f docker-compose.production.yml up -d --build -# ============================================================================ +# Supabase Configuration +VITE_SUPABASE_URL=your_supabase_url_here +VITE_SUPABASE_ANON_KEY=your_anon_key_here -# ================================ -# ENVIRONMENT MODE -# ================================ -# Options: development, production -ENVIRONMENT=development - -# ================================ -# FRONTEND URLs (for building) -# ================================ -# Development: http://localhost:3001 / ws://localhost:3001 -# Production: https://api.yourdomain.com / wss://api.yourdomain.com -VITE_API_BASE_URL=http://localhost:3001 -VITE_WS_URL=ws://localhost:3001 - -# ================================ -# SUPABASE AUTHENTICATION -# ================================ -# TODO: Get these from https://app.supabase.com/project/_/settings/api -# Required for user authentication (optional for development) -VITE_SUPABASE_URL= -VITE_SUPABASE_ANON_KEY= - -# Backend (same values as above) -SUPABASE_URL= -SUPABASE_KEY= - -# ================================ -# BACKEND CONFIGURATION -# ================================ -# CORS: Comma-separated allowed origins -# Development: http://localhost,http://localhost:5173,http://localhost:5174 -# Production: https://yourdomain.com -ALLOWED_ORIGINS=http://localhost,http://localhost:5173,http://localhost:5174,http://localhost:3000,http://localhost:8080 - -# Encryption key for user data (MongoDB conversations, credentials) -# TODO: Generate with: openssl rand -hex 32 -# WARNING: Losing this key = losing access to encrypted data! -ENCRYPTION_MASTER_KEY= - -# Frontend URL (for payment redirects) -FRONTEND_URL=http://localhost:5173 - -# Backend public URL (for generating download URLs) -BACKEND_URL=http://localhost:3001 - -# Cloudflare Turnstile (optional - for bot protection) -VITE_TURNSTILE_SITE_KEY= - -# ================================ -# MYSQL DATABASE -# ================================ -# MySQL root password (default: claraverse_root_2024) -MYSQL_ROOT_PASSWORD=claraverse_root_2024 - -# MySQL user password (default: claraverse_pass_2024) -MYSQL_PASSWORD=claraverse_pass_2024 - -# ================================ -# MONGODB & REDIS -# ================================ -# MongoDB - for conversations, workflows, and data persistence -# Docker: mongodb://mongodb:27017/claraverse -# Atlas: mongodb+srv://user:pass@cluster.mongodb.net/claraverse -# Self-host: mongodb://user:pass@host:27017/claraverse -MONGODB_URI=mongodb://mongodb:27017/claraverse - -# Redis - for job scheduling and WebSocket pub/sub -REDIS_URL=redis://redis:6379 - -# ================================ -# EXTERNAL SERVICES -# ================================ -# E2B Code Interpreter (optional - for code execution) -# Get API key from: https://e2b.dev/dashboard -E2B_API_KEY= - -# SearXNG (internal Docker URL, or comma-separated for load balancing) -SEARXNG_URLS=http://searxng:8080 - -# Composio (optional - for integrations) -COMPOSIO_API_KEY= -COMPOSIO_GOOGLESHEETS_AUTH_CONFIG_ID= -COMPOSIO_GMAIL_AUTH_CONFIG_ID= - - - -# ================================ -# ADMIN (Optional) -# ================================ -# Comma-separated list of Supabase user IDs with superadmin access -# Example: user-id-1,user-id-2,user-id-3 -SUPERADMIN_USER_IDS= - -# ================================ -# RATE LIMITING (requests per minute) -# ================================ -# Global API rate limit (all /api/* routes) -RATE_LIMIT_GLOBAL_API=200 - -# Public read-only endpoints -RATE_LIMIT_PUBLIC_READ=120 - -# Authenticated user requests -RATE_LIMIT_AUTHENTICATED=60 - -# WebSocket connections -RATE_LIMIT_WEBSOCKET=20 - -# Image proxy (to prevent bandwidth abuse) -RATE_LIMIT_IMAGE_PROXY=60 +# Only for server-side usage, do not use in client-side code +VITE_SUPABASE_SERVICE_KEY=your_service_key_here_server_only diff --git a/.gitattributes b/.gitattributes deleted file mode 100644 index 6a0223c6..00000000 --- a/.gitattributes +++ /dev/null @@ -1,44 +0,0 @@ -# Auto-detect text files and perform LF normalization -* text=auto - -# Shell scripts should always use LF (Unix-style) -*.sh text eol=lf - -# Windows batch files should always use CRLF (Windows-style) -*.bat text eol=crlf - -# Docker files should always use LF -Dockerfile* text eol=lf -docker-compose*.yml text eol=lf -.dockerignore text eol=lf - -# Configuration files -*.yml text eol=lf -*.yaml text eol=lf -*.json text eol=lf -*.toml text eol=lf - -# Source code -*.ts text eol=lf -*.tsx text eol=lf -*.js text eol=lf -*.jsx text eol=lf -*.go text eol=lf -*.py text eol=lf - -# Documentation -*.md text eol=lf - -# Environment files -.env* text eol=lf - -# Binary files -*.png binary -*.jpg binary -*.jpeg binary -*.gif binary -*.ico binary -*.webp binary -*.pdf binary -*.woff binary -*.woff2 binary diff --git a/.github/FUNDING.yml b/.github/FUNDING.yml new file mode 100644 index 00000000..ac9aae77 --- /dev/null +++ b/.github/FUNDING.yml @@ -0,0 +1,15 @@ +# These are supported funding model platforms + +github: +patreon: # Replace with a single Patreon username +open_collective: # Replace with a single Open Collective username +ko_fi: # Replace with a single Ko-fi username +tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel +community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry +liberapay: # Replace with a single Liberapay username +issuehunt: # Replace with a single IssueHunt username +lfx_crowdfunding: # Replace with a single LFX Crowdfunding project-name e.g., cloud-foundry +polar: # Replace with a single Polar username +buy_me_a_coffee: claraverse +thanks_dev: # Replace with a single thanks.dev username +custom: # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2'] diff --git a/.github/workflows/docker-publish.yml b/.github/workflows/docker-publish.yml new file mode 100644 index 00000000..d9ad1aac --- /dev/null +++ b/.github/workflows/docker-publish.yml @@ -0,0 +1,60 @@ +name: Build and Publish Docker Image + +on: + push: + tags: + - 'v*' + workflow_dispatch: + +env: + REGISTRY: docker.io + IMAGE_NAME: clara17verse/clara-backend + DOCKER_USERNAME: clara17verse + # The token should be stored as a GitHub secret, not directly in the workflow + +jobs: + build-and-push: + runs-on: ubuntu-latest + environment: production + permissions: + contents: read + packages: write + + steps: + - name: Checkout repository + uses: actions/checkout@v4 + + - name: Set up QEMU + uses: docker/setup-qemu-action@v3 + + - name: Set up Docker Buildx + uses: docker/setup-buildx-action@v3 + + - name: Log in to Docker Hub + uses: docker/login-action@v3 + with: + username: ${{ env.DOCKER_USERNAME }} + password: ${{ secrets.DOCKER_PAT }} + + - name: Extract metadata (tags, labels) + id: meta + uses: docker/metadata-action@v5 + with: + images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }} + tags: | + type=semver,pattern={{version}} + type=semver,pattern={{major}}.{{minor}} + type=sha,prefix= + type=raw,value=latest,enable={{is_default_branch}} + + - name: Build and push Docker image + uses: docker/build-push-action@v5 + with: + context: ./py_backend + file: ./py_backend/Dockerfile + platforms: linux/amd64,linux/arm64 + push: true + tags: ${{ steps.meta.outputs.tags }} + labels: ${{ steps.meta.outputs.labels }} + cache-from: type=gha + cache-to: type=gha,mode=max \ No newline at end of file diff --git a/.gitignore b/.gitignore index 60df40a6..73d46109 100644 --- a/.gitignore +++ b/.gitignore @@ -1,38 +1,160 @@ -frontend/.env +# Logs +logs +*.log +npm-debug.log* +yarn-debug.log* +yarn-error.log* +pnpm-debug.log* +lerna-debug.log* + +node_modules +dist +dist-ssr +*.local + +# Editor directories and files +.vscode/* +!.vscode/extensions.json +.idea +.DS_Store +*.suo +*.ntvs* +*.njsproj +*.sln +*.sw? .env -*.csv -*.xlsx -*.xls -*.pdf -*.docx -*.doc -*.pptx -*.ppt -*.odt -*.odp - -# Ignore txt files except requirements.txt -*.txt -!requirements.txt - -# Ignore markdown except README -*.md -!README.md - -# Ignore JSON files except essential ones -*.json -!package.json -!package-lock.json -!tsconfig.json -!tsconfig.*.json -!eslint.config.json -!**/fixtures/**/*.json -!providers.example.json -.claude/settings.local.json -__pycache__/ -frontend/.claude/settings.local.json -comfyui -_bmad -_bmad* -.agentvibes -.claude \ No newline at end of file +.env.* +!.env.example +/tools/.env + +# Security files +secret_key.txt +*_key.txt +*.pem +*.key + +# Electron specific +release +.electron-builder +electron-builder.yml +*.dmg +*.exe +*.deb +*.AppImage +*.snap +*.blockmap +*.zip +latest-mac.yml +latest-linux.yml +latest-win.yml +.bolt +__pycache__ + + +# Code signing files +package-mac-signed.sh +certificate.p12 + +./Support +python-env +.clara + +./py_backend/__pycache__ +./py_backend/cache +./py_backend/python-env +./py_backend/python-env/cache +./py_backend/python-env/pip-cache +./py_backend/python-env/python-env + +.cursor + +# any files in tools +tools/* + +/clara_interpreter_dockerstuff + +electron/llamacpp-binaries/models/* + +testing_sdk/* + +sdk/node_modules/* + +clara-sdk-docs/* + +electron/llamacpp-binaries/config.yaml + +sdk_examples/* +electron/llamacpp-binaries/config.yaml +battery-report.html +electron/llamacpp-binaries/config.yaml +landing/* +issues.md +# Local Netlify folder +.netlify + +# LLAMACPP BINARIES - Exclude only large binary files from platform directories +# Keep directory structure and config files, exclude only binaries + +# Exclude binary files from specific platform directories +electron/llamacpp-binaries/darwin-arm64/*.dylib +# electron/llamacpp-binaries/darwin-arm64/llama-* +electron/llamacpp-binaries/darwin-arm64/lib* +electron/llamacpp-binaries/darwin-arm64/*-server +electron/llamacpp-binaries/darwin-arm64/ggml-* + +electron/llamacpp-binaries/linux-x64/*.so +# electron/llamacpp-binaries/linux-x64/llama-* +electron/llamacpp-binaries/linux-x64/lib* +electron/llamacpp-binaries/linux-x64/*-server +electron/llamacpp-binaries/linux-x64/ggml-* +electron/llamacpp-binaries/linux-x64/rpc-server +electron/llamacpp-binaries/linux-x64/vulkan-shaders-gen + +# electron/llamacpp-binaries/win32-x64/*.dll +# # electron/llamacpp-binaries/win32-x64/llama-* +# electron/llamacpp-binaries/win32-x64/lib* +# electron/llamacpp-binaries/win32-x64/*-server +# electron/llamacpp-binaries/win32-x64/ggml-* + +# Keep config files, documentation, and scripts +!electron/llamacpp-binaries/*.md +!electron/llamacpp-binaries/*.json +!electron/llamacpp-binaries/*.yaml +!electron/llamacpp-binaries/*.sh +!electron/llamacpp-binaries/LICENSE* +py_backend/cache/* +py_backend/.venv/* +py_backend/light_rag_examples/* +electron/llamacpp-binaries/darwin-arm64-backup* +notarization.env +# anything with .env in the name +.env +scripts/notarize-production.sh +notarize-production.sh +setup-notarization-env.sh +mcp_workspace/* +electron/llamacpp-binaries/* + +.bmad-core +.claude +.github +.vscode +web-bundles +docs-internal-agents +searXNG-Clara +searxng-config +.bmad-infrastructure-devops +clara-mcp/mcp_workspace/* +llama-swap/* +dev_docs/* +config.yaml +model_folders.json +settings.json +/binaries/* +downloads/* +config.yaml.backup.* +electron/claracore/progress_state.* +package-lock.json + +# Agent deployment container testing +test_agent_container/ diff --git a/CHANGELOG.md b/CHANGELOG.md new file mode 100644 index 00000000..636d7b1c --- /dev/null +++ b/CHANGELOG.md @@ -0,0 +1,290 @@ +# Changelog + +All notable changes to Clara will be documented in this file. + +The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), +and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). + +## [0.1.4] - 2025-08-28 + +### 🚀 Major Features Added + +#### 🔹 Agents +- **MCP Autonomy**: Agents can now use MCP to operate fully autonomously — automate almost any PC task (except Paint, Excel, Word for now) +- **Speech Capabilities**: Agents can talk once tasks are done for more natural interactions +- **Scheduled Automation**: Automate with time. Example: Agents that log in/out of apps daily — works ~90% of the time on first try, retries if needed + +#### 🔹 RAG (Retrieval-Augmented Generation) +- **Improved Performance**: Enhanced RAG engine with better retrieval accuracy +- **Redesigned UI**: Clean, intuitive interface for better user experience +- **Task Histories**: Easier tracking of previous operations and results +- **Document Retry**: Graceful error handling for failed document processing + +#### 🔹 Chat +- **Redesigned Chat Input**: Simpler interface with no need for advanced MCP settings +- **Voice UX Improvements**: Smoother voice interactions and better speech recognition +- **Clara Memories Enhancements**: More efficient and user-friendly memory management + +#### 🔹 Tasks +- **Automate & Monitor**: Run tasks in the background, schedule them, monitor progress, and collect results +- **Background Execution**: Set-and-forget task automation +- **Progress Tracking**: Real-time monitoring of task execution +- **Results Collection**: Automated gathering and organization of task outputs + +### ✨ What You Can Try Today +- Set up an agent to auto-login to apps every morning +- Use voice-enabled chat to summarize your daily reports +- Schedule a workflow to clean up folders or back up files nightly +- Build a RAG-powered assistant to search across your company docs +- Automate email checks or notifications without relying on cloud services + +### 🛠️ Technical Improvements +- Enhanced automation reliability and retry mechanisms +- Improved voice processing and speech synthesis +- Better memory management for Clara Memories +- Optimized task scheduling and background processing + +### 🎯 Release Focus +This release is about making automation practical, reliable, and personal — right from your desktop. No cloud dependencies, full privacy control, and enterprise-grade task automation capabilities. + +--- + +## [Unreleased] - 0.1.5 + +### 🚧 In Development +- Multi-user support with authentication +- Cloud deployment templates for AWS/GCP/Azure +- Plugin system for custom AI tools +- Enhanced N8N workflow integration + +--- + +## [0.1.3] - 2024-12-22 + +### 🎉 Major Milestone: Complete Docker Transformation +ClaraVerse has undergone a **revolutionary transformation** from an Electron desktop application to a **Docker-powered web application** similar to OpenWebUI, while maintaining complete privacy and local execution. + +### 🚀 Major Features Added +- **🐳 Docker-First Architecture**: Complete Docker Compose setup with 7 integrated services +- **🔧 LumaUI: Complete Web Development Environment**: WebContainer integration with Monaco Editor +- **🧠 Enhanced AI Capabilities**: Dynamic token allocation (16k-128k) with autonomous execution +- **🎨 Advanced Preview System**: Dual preview modes with console integration + +### ✨ New Features +- **Docker Services Stack**: Clara Web UI, Backend API, LlamaSwap, N8N, ComfyUI, Redis, PostgreSQL +- **LumaUI Development**: Project templates, AI code generation, terminal integration, file management +- **Smart AI Integration**: Precision editing modes, tool call limits, error recovery + +### 🛠️ Technical Improvements +- **Performance & Reliability**: Fixed race conditions, optimized token usage, enhanced error handling +- **Developer Experience**: TypeScript integration, hot reload, debugging tools +- **Architecture Enhancements**: Service isolation, health monitoring, scalability + +### 🐛 Critical Bug Fixes +- **Docker Conversion**: Removed Electron dependencies, fixed console errors +- **LumaUI Stability**: Fixed WebContainer remounting, auto mode loops, file sync issues +- **AI Integration**: Fixed tool schema validation, token limits, conversation history + +### 🔧 Breaking Changes +- **Migration from Electron to Docker**: Desktop app discontinued, now web application +- **New Installation**: Use Docker Compose instead of app installers +- **LumaUI Interface**: WebContainer-based projects, enhanced AI chat, auto-save behavior + +--- + +## [0.1.2] - 2024-05-30 + +### 🚀 Major Features Added +- **Custom Model Path Management**: Added support for custom download paths for model downloads +- **Enhanced Local Storage Management**: Improved storage handling and configuration +- **SDK for Users**: Added comprehensive SDK for developers to build on Clara +- **Granular Configuration System**: Enhanced settings with more detailed configuration options +- **Multi-Platform Optimizations**: Tested and optimized for Linux, improved Windows compatibility +- **Server Management Integration**: Moved servers to settings for better organization + +### ✨ New Features +- **Custom Download Path Support**: Users can now specify custom paths for model downloads +- **Enhanced MCP Diagnosis**: Added support for nvm node versions in PATH for MCP diagnosis +- **Linux 64-bit CPU Binaries**: Added dedicated binaries for Linux 64-bit systems +- **Windows CUDA Binaries**: Added CUDA support for Windows users +- **Call, TTS, and STT Integration**: Added text-to-speech and speech-to-text capabilities +- **Enhanced Python Backend**: Improved stability and performance of the Python backend +- **Provider Management**: Added comprehensive provider management functionality in settings + +### 🛠️ Improvements +- **Security Enhancements**: Fixed security issues with exposed API keys and vulnerable dependencies +- **UI/UX Improvements**: Multiple quality-of-life improvements across the interface +- **Performance Optimizations**: Enhanced performance across multiple components +- **Documentation Updates**: Updated README and documentation for better clarity +- **Build System Improvements**: Enhanced build processes and dependency management + +### 🐛 Bug Fixes +- **Dependency Vulnerabilities**: Fixed multiple security vulnerabilities in dependencies +- **API Key Exposure**: Resolved issues with exposed API keys +- **Model Management**: Fixed various bugs in model downloading and management +- **UI Responsiveness**: Fixed various UI responsiveness issues +- **Cross-Platform Compatibility**: Resolved platform-specific issues + +### 🔧 Technical Improvements +- **Code Quality**: Refactored multiple components for better maintainability +- **Build Process**: Enhanced build and deployment processes +- **Testing**: Improved testing coverage and reliability +- **Documentation**: Enhanced code documentation and user guides + +--- + +## [0.1.1] - 2024-05-20 + +### 🚀 Major Features Added +- **Electron Integration**: Full desktop application support with native features +- **Image Generation Support**: Comprehensive image generation capabilities +- **Node-Based Workflow System**: Visual workflow builder with drag-and-drop functionality +- **App Creator Enhancement**: Complete refactoring of the node registration mechanism + +### ✨ New Features +- **Clipboard Node**: Added clipboard functionality for workflows +- **Concatenation Tool**: New tool for string concatenation in workflows +- **Visual App Runner**: Enhanced app runner with chat UI and horizontal image+text inputs +- **Image Handling**: Improved image handling in nodes with runtime image replacement +- **Auto-Save Functionality**: Added automatic saving for user work +- **Template System**: Added templates for image generation + +### 🛠️ Improvements +- **UI/UX Enhancements**: Multiple quality-of-life improvements +- **Code Highlighting**: Removed syntax highlighting and border styling from code blocks for cleaner appearance +- **App Deletion Process**: Moved app deletion to AppCreator with improved deletion process +- **Workflow Integration**: Enhanced workflow system with better node management + +### 🐛 Bug Fixes +- **Image Node Issues**: Fixed image handling bugs in workflow nodes +- **UI Responsiveness**: Resolved various UI layout issues +- **Workflow Execution**: Fixed bugs in workflow execution engine + +### 🔧 Technical Improvements +- **Code Refactoring**: Major refactoring of the complete node register mechanism +- **Component Architecture**: Improved component structure for better maintainability +- **Build System**: Enhanced build processes for Electron integration + +--- + +## [0.1.0] - 2024-05-01 + +### 🎉 Initial Release +- **Core Chat Interface**: Basic AI chat functionality with local LLM support +- **Privacy-First Architecture**: Complete local processing with no cloud dependencies +- **Multi-Provider Support**: Support for various AI model providers +- **Basic UI Framework**: Initial user interface with essential features +- **Local Storage**: Client-side data storage system +- **Open Source Foundation**: MIT licensed with full source code availability + +### ✨ Initial Features +- **Local AI Chat**: Chat with AI models running locally +- **Model Management**: Basic model loading and management +- **Responsive Design**: Mobile and desktop responsive interface +- **Settings System**: Basic configuration and settings management +- **File Handling**: Initial file upload and processing capabilities + +### 🔧 Technical Foundation +- **React Frontend**: Built with modern React and TypeScript +- **Electron Support**: Desktop application framework +- **Vite Build System**: Fast development and build processes +- **Local Storage API**: IndexedDB integration for local data persistence +- **Modular Architecture**: Component-based architecture for extensibility + +--- + +## Installation & Upgrade Guide + +### Fresh Installation +```bash +# Clone the repository +git clone https://github.com/badboysm890/ClaraVerse.git +cd ClaraVerse + +# Install dependencies +npm install + +# Run development server +npm run dev + +# Or run desktop application +npm run electron:dev +``` + +### Upgrading from Previous Versions +```bash +# Pull latest changes +git pull origin main + +# Update dependencies +npm install + +# Rebuild application +npm run build +``` + +### Docker Installation +```bash +# Run with Docker +docker run -p 8069:8069 clara-ollama:latest +``` + +--- + +## Breaking Changes + +### From 0.1.0 to 0.1.1 +- **Node System**: Complete refactoring of the node registration system +- **Image Handling**: Changes to image processing pipeline +- **App Creator**: Significant changes to app creation workflow + +### From 0.1.1 to 0.1.2 +- **Settings Structure**: Server settings moved to new location +- **Model Paths**: Custom model path configuration added +- **Storage Management**: Enhanced local storage structure + +--- + +## Migration Guide + +### Migrating to 0.1.2 +1. **Settings Update**: Check your server settings as they have been reorganized +2. **Model Paths**: Configure custom model download paths if needed +3. **Dependencies**: Update all dependencies using `npm install` +4. **Storage**: Clear local storage if experiencing issues (use clear_storage.js) + +--- + +## Known Issues + +### Current Known Issues (0.1.2) +- Some legacy workflow configurations may need manual updating +- Windows users may need to run as administrator for certain model downloads +- macOS users need to manually approve unsigned applications + +### Workarounds +- **macOS App Damage Warning**: Right-click app and select "Open", then approve in System Preferences +- **Windows Admin Rights**: Run as administrator if model downloads fail +- **Linux Permissions**: Ensure proper permissions for model storage directories + +--- + +## Support & Feedback + +- **Email**: [praveensm890@gmail.com](mailto:praveensm890@gmail.com) +- **GitHub Issues**: [Report bugs and request features](https://github.com/badboysm890/ClaraVerse/issues) +- **Discussions**: [Join community discussions](https://github.com/badboysm890/ClaraVerse/discussions) + +--- + +## Contributing + +We welcome contributions! Please see our [Contributing Guide](CONTRIBUTING.md) for details on how to: +- Report bugs +- Suggest features +- Submit pull requests +- Improve documentation + +--- + +*For more information, visit the [Clara Documentation](https://github.com/badboysm890/ClaraVerse) or join our community discussions.* \ No newline at end of file diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 00000000..33b34542 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,13 @@ +FROM node:20-alpine AS builder + +WORKDIR /app +COPY package*.json ./ +RUN npm install +COPY . . +RUN npm run build + +FROM nginx:alpine +COPY --from=builder /app/dist /usr/share/nginx/html +COPY nginx.conf /etc/nginx/conf.d/default.conf +EXPOSE 8069 +CMD ["nginx", "-g", "daemon off;"] diff --git a/LICENSE b/LICENSE index 981aec6b..79443301 100644 --- a/LICENSE +++ b/LICENSE @@ -1,702 +1,21 @@ -ClaraVerse - Privacy-First AI Workspace -Copyright (C) 2025 claraverse-space - -This program is free software: you can redistribute it and/or modify -it under the terms of the GNU Affero General Public License as published -by the Free Software Foundation, either version 3 of the License, or -(at your option) any later version. - -This program is distributed in the hope that it will be useful, -but WITHOUT ANY WARRANTY; without even the implied warranty of -MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -GNU Affero General Public License for more details. - -You should have received a copy of the GNU Affero General Public License -along with this program. If not, see . - -Additional Terms (AGPL Section 7): -- Network Use: If you run a modified version of ClaraVerse as a network service, - you must make the complete source code available to users of that service. -- Attribution: You must retain all copyright notices and give appropriate credit - to the ClaraVerse developers. - -DUAL LICENSING OPTIONS: - -ClaraVerse is available under dual licensing: - -1. AGPL-3.0 (This License - Free & Open Source) - - Free for everyone - - Must share all modifications - - Network copyleft applies - -2. Commercial License (Alternative Licensing) - - Small companies (<1000 employees): FREE upon request - - Large enterprises (1000+ employees): Paid license with enterprise support - - No requirement to share modifications - - Can integrate into proprietary software - -Contact: support@claraverse.space for commercial licensing inquiries - -================================================================================ - - GNU AFFERO GENERAL PUBLIC LICENSE - Version 3, 19 November 2007 - - Copyright (C) 2007 Free Software Foundation, Inc. - Everyone is permitted to copy and distribute verbatim copies - of this license document, but changing it is not allowed. - - Preamble - - The GNU Affero General Public License is a free, copyleft license for -software and other kinds of works, specifically designed to ensure -cooperation with the community in the case of network server software. - - The licenses for most software and other practical works are designed -to take away your freedom to share and change the works. By contrast, -our General Public Licenses are intended to guarantee your freedom to -share and change all versions of a program--to make sure it remains free -software for all its users. - - When we speak of free software, we are referring to freedom, not -price. Our General Public Licenses are designed to make sure that you -have the freedom to distribute copies of free software (and charge for -them if you wish), that you receive source code or can get it if you -want it, that you can change the software or use pieces of it in new -free programs, and that you know you can do these things. - - Developers that use our General Public Licenses protect your rights -with two steps: (1) assert copyright on the software, and (2) offer -you this License which gives you legal permission to copy, distribute -and/or modify the software. - - A secondary benefit of defending all users' freedom is that -improvements made in alternate versions of the program, if they -receive widespread use, become available for other developers to -incorporate. Many developers of free software are heartened and -encouraged by the resulting cooperation. However, in the case of -software used on network servers, this result may fail to come about. -The GNU General Public License permits making a modified version and -letting the public access it on a server without ever releasing its -source code to the public. - - The GNU Affero General Public License is designed specifically to -ensure that, in such cases, the modified source code becomes available -to the community. It requires the operator of a network server to -provide the source code of the modified version running there to the -users of that server. Therefore, public use of a modified version, on -a publicly accessible server, gives the public access to the source -code of the modified version. - - An older license, called the Affero General Public License and -published by Affero, was designed to accomplish similar goals. This is -a different license, not a version of the Affero GPL, but Affero has -released a new version of the Affero GPL which permits relicensing under -this license. - - The precise terms and conditions for copying, distribution and -modification follow. - - TERMS AND CONDITIONS - - 0. Definitions. - - "This License" refers to version 3 of the GNU Affero General Public License. - - "Copyright" also means copyright-like laws that apply to other kinds of -works, such as semiconductor masks. - - "The Program" refers to any copyrightable work licensed under this -License. Each licensee is addressed as "you". "Licensees" and -"recipients" may be individuals or organizations. - - To "modify" a work means to copy from or adapt all or part of the work -in a fashion requiring copyright permission, other than the making of an -exact copy. The resulting work is called a "modified version" of the -earlier work or a work "based on" the earlier work. - - A "covered work" means either the unmodified Program or a work based -on the Program. - - To "propagate" a work means to do anything with it that, without -permission, would make you directly or secondarily liable for -infringement under applicable copyright law, except executing it on a -computer or modifying a private copy. Propagation includes copying, -distribution (with or without modification), making available to the -public, and in some countries other activities as well. - - To "convey" a work means any kind of propagation that enables other -parties to make or receive copies. Mere interaction with a user through -a computer network, with no transfer of a copy, is not conveying. - - An interactive user interface displays "Appropriate Legal Notices" -to the extent that it includes a convenient and prominently visible -feature that (1) displays an appropriate copyright notice, and (2) -tells the user that there is no warranty for the work (except to the -extent that warranties are provided), that licensees may convey the -work under this License, and how to view a copy of this License. If -the interface presents a list of user commands or options, such as a -menu, a prominent item in the list meets this criterion. - - 1. Source Code. - - The "source code" for a work means the preferred form of the work -for making modifications to it. "Object code" means any non-source -form of a work. - - A "Standard Interface" means an interface that either is an official -standard defined by a recognized standards body, or, in the case of -interfaces specified for a particular programming language, one that -is widely used among developers working in that language. - - The "System Libraries" of an executable work include anything, other -than the work as a whole, that (a) is included in the normal form of -packaging a Major Component, but which is not part of that Major -Component, and (b) serves only to enable use of the work with that -Major Component, or to implement a Standard Interface for which an -implementation is available to the public in source code form. A -"Major Component", in this context, means a major essential component -(kernel, window system, and so on) of the specific operating system -(if any) on which the executable work runs, or a compiler used to -produce the work, or an object code interpreter used to run it. - - The "Corresponding Source" for a work in object code form means all -the source code needed to generate, install, and (for an executable -work) run the object code and to modify the work, including scripts to -control those activities. However, it does not include the work's -System Libraries, or general-purpose tools or generally available free -programs which are used unmodified in performing those activities but -which are not part of the work. For example, Corresponding Source -includes interface definition files associated with source files for -the work, and the source code for shared libraries and dynamically -linked subprograms that the work is specifically designed to require, -such as by intimate data communication or control flow between those -subprograms and other parts of the work. - - The Corresponding Source need not include anything that users -can regenerate automatically from other parts of the Corresponding -Source. - - The Corresponding Source for a work in source code form is that -same work. - - 2. Basic Permissions. - - All rights granted under this License are granted for the term of -copyright on the Program, and are irrevocable provided the stated -conditions are met. This License explicitly affirms your unlimited -permission to run the unmodified Program. The output from running a -covered work is covered by this License only if the output, given its -content, constitutes a covered work. This License acknowledges your -rights of fair use or other equivalent, as provided by copyright law. - - You may make, run and propagate covered works that you do not -convey, without conditions so long as your license otherwise remains -in force. You may convey covered works to others for the sole purpose -of having them make modifications exclusively for you, or provide you -with facilities for running those works, provided that you comply with -the terms of this License in conveying all material for which you do -not control copyright. Those thus making or running the covered works -for you must do so exclusively on your behalf, under your direction -and control, on terms that prohibit them from making any copies of -your copyrighted material outside their relationship with you. - - Conveying under any other circumstances is permitted solely under -the conditions stated below. Sublicensing is not allowed; section 10 -makes it unnecessary. - - 3. Protecting Users' Legal Rights From Anti-Circumvention Law. - - No covered work shall be deemed part of an effective technological -measure under any applicable law fulfilling obligations under article -11 of the WIPO copyright treaty adopted on 20 December 1996, or -similar laws prohibiting or restricting circumvention of such -measures. - - When you convey a covered work, you waive any legal power to forbid -circumvention of technological measures to the extent such circumvention -is effected by exercising rights under this License with respect to -the covered work, and you disclaim any intention to limit operation or -modification of the work as a means of enforcing, against the work's -users, your or third parties' legal rights to forbid circumvention of -technological measures. - - 4. Conveying Verbatim Copies. - - You may convey verbatim copies of the Program's source code as you -receive it, in any medium, provided that you conspicuously and -appropriately publish on each copy an appropriate copyright notice; -keep intact all notices stating that this License and any -non-permissive terms added in accord with section 7 apply to the code; -keep intact all notices of the absence of any warranty; and give all -recipients a copy of this License along with the Program. - - You may charge any price or no price for each copy that you convey, -and you may offer support or warranty protection for a fee. - - 5. Conveying Modified Source Versions. - - You may convey a work based on the Program, or the modifications to -produce it from the Program, in the form of source code under the -terms of section 4, provided that you also meet all of these conditions: - - a) The work must carry prominent notices stating that you modified - it, and giving a relevant date. - - b) The work must carry prominent notices stating that it is - released under this License and any conditions added under section - 7. This requirement modifies the requirement in section 4 to - "keep intact all notices". - - c) You must license the entire work, as a whole, under this - License to anyone who comes into possession of a copy. This - License will therefore apply, along with any applicable section 7 - additional terms, to the whole of the work, and all its parts, - regardless of how they are packaged. This License gives no - permission to license the work in any other way, but it does not - invalidate such permission if you have separately received it. - - d) If the work has interactive user interfaces, each must display - Appropriate Legal Notices; however, if the Program has interactive - interfaces that do not display Appropriate Legal Notices, your - work need not make them do so. - - A compilation of a covered work with other separate and independent -works, which are not by their nature extensions of the covered work, -and which are not combined with it such as to form a larger program, -in or on a volume of a storage or distribution medium, is called an -"aggregate" if the compilation and its resulting copyright are not -used to limit the access or legal rights of the compilation's users -beyond what the individual works permit. Inclusion of a covered work -in an aggregate does not cause this License to apply to the other -parts of the aggregate. - - 6. Conveying Non-Source Forms. - - You may convey a covered work in object code form under the terms -of sections 4 and 5, provided that you also convey the -machine-readable Corresponding Source under the terms of this License, -in one of these ways: - - a) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by the - Corresponding Source fixed on a durable physical medium - customarily used for software interchange. - - b) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by a - written offer, valid for at least three years and valid for as - long as you offer spare parts or customer support for that product - model, to give anyone who possesses the object code either (1) a - copy of the Corresponding Source for all the software in the - product that is covered by this License, on a durable physical - medium customarily used for software interchange, for a price no - more than your reasonable cost of physically performing this - conveying of source, or (2) access to copy the - Corresponding Source from a network server at no charge. - - c) Convey individual copies of the object code with a copy of the - written offer to provide the Corresponding Source. This - alternative is allowed only occasionally and noncommercially, and - only if you received the object code with such an offer, in accord - with subsection 6b. - - d) Convey the object code by offering access from a designated - place (gratis or for a charge), and offer equivalent access to the - Corresponding Source in the same way through the same place at no - further charge. You need not require recipients to copy the - Corresponding Source along with the object code. If the place to - copy the object code is a network server, the Corresponding Source - may be on a different server (operated by you or a third party) - that supports equivalent copying facilities, provided you maintain - clear directions next to the object code saying where to find the - Corresponding Source. Regardless of what server hosts the - Corresponding Source, you remain obligated to ensure that it is - available for as long as needed to satisfy these requirements. - - e) Convey the object code using peer-to-peer transmission, provided - you inform other peers where the object code and Corresponding - Source of the work are being offered to the general public at no - charge under subsection 6d. - - A separable portion of the object code, whose source code is excluded -from the Corresponding Source as a System Library, need not be -included in conveying the object code work. - - A "User Product" is either (1) a "consumer product", which means any -tangible personal property which is normally used for personal, family, -or household purposes, or (2) anything designed or sold for incorporation -into a dwelling. In determining whether a product is a consumer product, -doubtful cases shall be resolved in favor of coverage. For a particular -product received by a particular user, "normally used" refers to a -typical or common use of that class of product, regardless of the status -of the particular user or of the way in which the particular user -actually uses, or expects or is expected to use, the product. A product -is a consumer product regardless of whether the product has substantial -commercial, industrial or non-consumer uses, unless such uses represent -the only significant mode of use of the product. - - "Installation Information" for a User Product means any methods, -procedures, authorization keys, or other information required to install -and execute modified versions of a covered work in that User Product from -a modified version of its Corresponding Source. The information must -suffice to ensure that the continued functioning of the modified object -code is in no case prevented or interfered with solely because -modification has been made. - - If you convey an object code work under this section in, or with, or -specifically for use in, a User Product, and the conveying occurs as -part of a transaction in which the right of possession and use of the -User Product is transferred to the recipient in perpetuity or for a -fixed term (regardless of how the transaction is characterized), the -Corresponding Source conveyed under this section must be accompanied -by the Installation Information. But this requirement does not apply -if neither you nor any third party retains the ability to install -modified object code on the User Product (for example, the work has -been installed in ROM). - - The requirement to provide Installation Information does not include a -requirement to continue to provide support service, warranty, or updates -for a work that has been modified or installed by the recipient, or for -the User Product in which it has been modified or installed. Access to a -network may be denied when the modification itself materially and -adversely affects the operation of the network or violates the rules and -protocols for communication across the network. - - Corresponding Source conveyed, and Installation Information provided, -in accord with this section must be in a format that is publicly -documented (and with an implementation available to the public in -source code form), and must require no special password or key for -unpacking, reading or copying. - - 7. Additional Terms. - - "Additional permissions" are terms that supplement the terms of this -License by making exceptions from one or more of its conditions. -Additional permissions that are applicable to the entire Program shall -be treated as though they were included in this License, to the extent -that they are valid under applicable law. If additional permissions -apply only to part of the Program, that part may be used separately -under those permissions, but the entire Program remains governed by -this License without regard to the additional permissions. - - When you convey a copy of a covered work, you may at your option -remove any additional permissions from that copy, or from any part of -it. (Additional permissions may be written to require their own -removal in certain cases when you modify the work.) You may place -additional permissions on material, added by you to a covered work, -for which you have or can give appropriate copyright permission. - - Notwithstanding any other provision of this License, for material you -add to a covered work, you may (if authorized by the copyright holders of -that material) supplement the terms of this License with terms: - - a) Disclaiming warranty or limiting liability differently from the - terms of sections 15 and 16 of this License; or - - b) Requiring preservation of specified reasonable legal notices or - author attributions in that material or in the Appropriate Legal - Notices displayed by works containing it; or - - c) Prohibiting misrepresentation of the origin of that material, or - requiring that modified versions of such material be marked in - reasonable ways as different from the original version; or - - d) Limiting the use for publicity purposes of names of licensors or - authors of the material; or - - e) Declining to grant rights under trademark law for use of some - trade names, trademarks, or service marks; or - - f) Requiring indemnification of licensors and authors of that - material by anyone who conveys the material (or modified versions of - it) with contractual assumptions of liability to the recipient, for - any liability that these contractual assumptions directly impose on - those licensors and authors. - - All other non-permissive additional terms are considered "further -restrictions" within the meaning of section 10. If the Program as you -received it, or any part of it, contains a notice stating that it is -governed by this License along with a term that is a further -restriction, you may remove that term. If a license document contains -a further restriction but permits relicensing or conveying under this -License, you may add to a covered work material governed by the terms -of that license document, provided that the further restriction does -not survive such relicensing or conveying. - - If you add terms to a covered work in accord with this section, you -must place, in the relevant source files, a statement of the -additional terms that apply to those files, or a notice indicating -where to find the applicable terms. - - Additional terms, permissive or non-permissive, may be stated in the -form of a separately written license, or stated as exceptions; -the above requirements apply either way. - - 8. Termination. - - You may not propagate or modify a covered work except as expressly -provided under this License. Any attempt otherwise to propagate or -modify it is void, and will automatically terminate your rights under -this License (including any patent licenses granted under the third -paragraph of section 11). - - However, if you cease all violation of this License, then your -license from a particular copyright holder is reinstated (a) -provisionally, unless and until the copyright holder explicitly and -finally terminates your license, and (b) permanently, if the copyright -holder fails to notify you of the violation by some reasonable means -prior to 60 days after the cessation. - - Moreover, your license from a particular copyright holder is -reinstated permanently if the copyright holder notifies you of the -violation by some reasonable means, this is the first time you have -received notice of violation of this License (for any work) from that -copyright holder, and you cure the violation prior to 30 days after -your receipt of the notice. - - Termination of your rights under this section does not terminate the -licenses of parties who have received copies or rights from you under -this License. If your rights have been terminated and not permanently -reinstated, you do not qualify to receive new licenses for the same -material under section 10. - - 9. Acceptance Not Required for Having Copies. - - You are not required to accept this License in order to receive or -run a copy of the Program. Ancillary propagation of a covered work -occurring solely as a consequence of using peer-to-peer transmission -to receive a copy likewise does not require acceptance. However, -nothing other than this License grants you permission to propagate or -modify any covered work. These actions infringe copyright if you do -not accept this License. Therefore, by modifying or propagating a -covered work, you indicate your acceptance of this License to do so. - - 10. Automatic Licensing of Downstream Recipients. - - Each time you convey a covered work, the recipient automatically -receives a license from the original licensors, to run, modify and -propagate that work, subject to this License. You are not responsible -for enforcing compliance by third parties with this License. - - An "entity transaction" is a transaction transferring control of an -organization, or substantially all assets of one, or subdividing an -organization, or merging organizations. If propagation of a covered -work results from an entity transaction, each party to that -transaction who receives a copy of the work also receives whatever -licenses to the work the party's predecessor in interest had or could -give under the previous paragraph, plus a right to possession of the -Corresponding Source of the work from the predecessor in interest, if -the predecessor has it or can get it with reasonable efforts. - - You may not impose any further restrictions on the exercise of the -rights granted or affirmed under this License. For example, you may -not impose a license fee, royalty, or other charge for exercise of -rights granted under this License, and you may not initiate litigation -(including a cross-claim or counterclaim in a lawsuit) alleging that -any patent claim is infringed by making, using, selling, offering for -sale, or importing the Program or any portion of it. - - 11. Patents. - - A "contributor" is a copyright holder who authorizes use under this -License of the Program or a work on which the Program is based. The -work thus licensed is called the contributor's "contributor version". - - A contributor's "essential patent claims" are all patent claims -owned or controlled by the contributor, whether already acquired or -hereafter acquired, that would be infringed by some manner, permitted -by this License, of making, using, or selling its contributor version, -but do not include claims that would be infringed only as a -consequence of further modification of the contributor version. For -purposes of this definition, "control" includes the right to grant -patent sublicenses in a manner consistent with the requirements of -this License. - - Each contributor grants you a non-exclusive, worldwide, royalty-free -patent license under the contributor's essential patent claims, to -make, use, sell, offer for sale, import and otherwise run, modify and -propagate the contents of its contributor version. - - In the following three paragraphs, a "patent license" is any express -agreement or commitment, however denominated, not to enforce a patent -(such as an express permission to practice a patent or covenant not to -sue for patent infringement). To "grant" such a patent license to a -party means to make such an agreement or commitment not to enforce a -patent against the party. - - If you convey a covered work, knowingly relying on a patent license, -and the Corresponding Source of the work is not available for anyone -to copy, free of charge and under the terms of this License, through a -publicly available network server or other readily accessible means, -then you must either (1) cause the Corresponding Source to be so -available, or (2) arrange to deprive yourself of the benefit of the -patent license for this particular work, or (3) arrange, in a manner -consistent with the requirements of this License, to extend the patent -license to downstream recipients. "Knowingly relying" means you have -actual knowledge that, but for the patent license, your conveying the -covered work in a country, or your recipient's use of the covered work -in a country, would infringe one or more identifiable patents in that -country that you have reason to believe are valid. - - If, pursuant to or in connection with a single transaction or -arrangement, you convey, or propagate by procuring conveyance of, a -covered work, and grant a patent license to some of the parties -receiving the covered work authorizing them to use, propagate, modify -or convey a specific copy of the covered work, then the patent license -you grant is automatically extended to all recipients of the covered -work and works based on it. - - A patent license is "discriminatory" if it does not include within -the scope of its coverage, prohibits the exercise of, or is -conditioned on the non-exercise of one or more of the rights that are -specifically granted under this License. You may not convey a covered -work if you are a party to an arrangement with a third party that is -in the business of distributing software, under which you make payment -to the third party based on the extent of your activity of conveying -the work, and under which the third party grants, to any of the -parties who would receive the covered work from you, a discriminatory -patent license (a) in connection with copies of the covered work -conveyed by you (or copies made from those copies), or (b) primarily -for and in connection with specific products or compilations that -contain the covered work, unless you entered into that arrangement, -or that patent license was granted, prior to 28 March 2007. - - Nothing in this License shall be construed as excluding or limiting -any implied license or other defenses to infringement that may -otherwise be available to you under applicable patent law. - - 12. No Surrender of Others' Freedom. - - If conditions are imposed on you (whether by court order, agreement or -otherwise) that contradict the conditions of this License, they do not -excuse you from the conditions of this License. If you cannot convey a -covered work so as to satisfy simultaneously your obligations under this -License and any other pertinent obligations, then as a consequence you may -not convey it at all. For example, if you agree to terms that obligate you -to collect a royalty for further conveying from those to whom you convey -the Program, the only way you could satisfy both those terms and this -License would be to refrain entirely from conveying the Program. - - 13. Remote Network Interaction; Use with the GNU General Public License. - - Notwithstanding any other provision of this License, if you modify the -Program, your modified version must prominently offer all users -interacting with it remotely through a computer network (if your version -supports such interaction) an opportunity to receive the Corresponding -Source of your version by providing access to the Corresponding Source -from a network server at no charge, through some standard or customary -means of facilitating copying of software. This Corresponding Source -shall include the Corresponding Source for any work covered by version 3 -of the GNU General Public License that is incorporated pursuant to the -following paragraph. - - Notwithstanding any other provision of this License, you have -permission to link or combine any covered work with a work licensed -under version 3 of the GNU General Public License into a single -combined work, and to convey the resulting work. The terms of this -License will continue to apply to the part which is the covered work, -but the work with which it is combined will remain governed by version -3 of the GNU General Public License. - - 14. Revised Versions of this License. - - The Free Software Foundation may publish revised and/or new versions of -the GNU Affero General Public License from time to time. Such new versions -will be similar in spirit to the present version, but may differ in detail to -address new problems or concerns. - - Each version is given a distinguishing version number. If the -Program specifies that a certain numbered version of the GNU Affero General -Public License "or any later version" applies to it, you have the -option of following the terms and conditions either of that numbered -version or of any later version published by the Free Software -Foundation. If the Program does not specify a version number of the -GNU Affero General Public License, you may choose any version ever published -by the Free Software Foundation. - - If the Program specifies that a proxy can decide which future -versions of the GNU Affero General Public License can be used, that proxy's -public statement of acceptance of a version permanently authorizes you -to choose that version for the Program. - - Later license versions may give you additional or different -permissions. However, no additional obligations are imposed on any -author or copyright holder as a result of your choosing to follow a -later version. - - 15. Disclaimer of Warranty. - - THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY -APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT -HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY -OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, -THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR -PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM -IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF -ALL NECESSARY SERVICING, REPAIR OR CORRECTION. - - 16. Limitation of Liability. - - IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING -WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS -THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY -GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE -USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF -DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD -PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), -EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF -SUCH DAMAGES. - - 17. Interpretation of Sections 15 and 16. - - If the disclaimer of warranty and limitation of liability provided -above cannot be given local legal effect according to their terms, -reviewing courts shall apply local law that most closely approximates -an absolute waiver of all civil liability in connection with the -Program, unless a warranty or assumption of liability accompanies a -copy of the Program in return for a fee. - - END OF TERMS AND CONDITIONS - - How to Apply These Terms to Your New Programs - - If you develop a new program, and you want it to be of the greatest -possible use to the public, the best way to achieve this is to make it -free software which everyone can redistribute and change under these terms. - - To do so, attach the following notices to the program. It is safest -to attach them to the start of each source file to most effectively -state the exclusion of warranty; and each file should have at least -the "copyright" line and a pointer to where the full notice is found. - - - Copyright (C) - - This program is free software: you can redistribute it and/or modify - it under the terms of the GNU Affero General Public License as published by - the Free Software Foundation, either version 3 of the License, or - (at your option) any later version. - - This program is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU Affero General Public License for more details. - - You should have received a copy of the GNU Affero General Public License - along with this program. If not, see . - -Also add information on how to contact you by electronic and paper mail. - - If your software can interact with users remotely through a computer -network, you should also make sure that it provides a way for users to -get its source. For example, if your program is a web application, its -interface could display a "Source" link that leads users to an archive -of the code. There are many ways you could offer source, and different -solutions will be better for different programs; see section 13 for the -specific requirements. - - You should also get your employer (if you work as a programmer) or school, -if any, to sign a "copyright disclaimer" for the program, if necessary. -For more information on this, and how to apply and follow the GNU AGPL, see -. +MIT License + +Copyright (c) 2025 ClaraVerse Labs and Contributors + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/README.md b/README.md index faf3c5c8..7542832e 100644 --- a/README.md +++ b/README.md @@ -1,708 +1,917 @@
-Claraverse - Your Private AI Workspace +
+ + + + + ClaraVerse Logo + + +
+
+ +# **ClaraVerse** -### **Your Private AI Workspace** +

+ The Complete AI Ecosystem That Actually Respects Your Privacy +

-*One app replaces ChatGPT, Midjourney, and N8N. Local or cloud - your data stays yours.* +

+ + Star us + +

-

+[![Buy A Coffee](https://img.buymeacoffee.com/button-api/?text=Buy%20me%20a%20coffee&slug=claraverse&button_colour=FFDD00&font_colour=000000&font_family=Poppins&outline_colour=000000&coffee_colour=ffffff)](https://www.buymeacoffee.com/claraverse) -[![License](https://img.shields.io/badge/license-AGPL--3.0-blue.svg)](LICENSE) -[![GitHub Stars](https://img.shields.io/github/stars/claraverse-space/ClaraVerseAI?style=social)](https://github.com/claraverse-space/ClaraVerseAI/stargazers) -[![Docker Pulls](https://img.shields.io/docker/pulls/claraverseoss/claraverse?color=blue)](https://hub.docker.com/r/claraverseoss/claraverse) -[![Discord](https://img.shields.io/badge/Discord-Join%20Us-7289da?logo=discord&logoColor=white)](https://discord.com/invite/j633fsrAne) -[Website](https://claraverse.space) · [Documentation](#-documentation) · [Quick Start](#-quick-start) · [Community](#-community) · [Contributing](#-contributing) +
+ +[![GitHub Stars](https://img.shields.io/github/stars/badboysm890/ClaraVerse?style=flat&logo=github&color=gold&labelColor=1e1e2e)](https://github.com/badboysm890/ClaraVerse/stargazers) +[![Downloads](https://img.shields.io/github/downloads/badboysm890/ClaraVerse/total?style=flat&logo=github&color=00D4AA&labelColor=1e1e2e)](https://github.com/badboysm890/ClaraVerse/releases) +[![Discord](https://img.shields.io/badge/Discord-Join_Community-5865F2?style=flat&logo=discord&labelColor=1e1e2e)](https://discord.gg/j633fsrAne) +[![License](https://img.shields.io/github/license/badboysm890/ClaraVerse?style=flat&color=FF6B6B&labelColor=1e1e2e)](LICENSE) +[![Latest Release](https://img.shields.io/github/v/release/badboysm890/ClaraVerse?style=flat&color=9333EA&labelColor=1e1e2e)](https://github.com/badboysm890/ClaraVerse/releases/latest) + +
+ +

+ 8 AI Tools × + 1 App × + 100% Local = + ∞ Possibilities +

+ +
+ + + Download + +  + + Demo + + +
+
+ + + + + ClaraVerse Features Banner +
--- -## 🚀 Quick Start +
-**Install CLI:** -```bash -curl -fsSL https://get.claraverse.app | bash -``` +## 💭 **The Story Behind ClaraVerse** -**Start ClaraVerse:** -```bash -claraverse init -``` +> **"Why can't everything be in a single app? Why do we need to jump between different AI tools and pay multiple subscriptions?"** -Open **http://localhost** → Register → Add AI provider → Start chatting! +**Here's what happened to us (and probably you too):** -
-Other options +We found ourselves constantly jumping between different AI apps — tired of paying for Claude subscriptions, then installing LM Studio for local models, then N8N for automation, then Ollama for model management, then OpenWebUI for a better chat interface.. -**Docker (no CLI):** -```bash -docker run -d -p 80:80 -p 3001:3001 -v claraverse-data:/data claraverseoss/claraverse:latest -``` +When using Ollama, downloading models was the only way to get started — but then we were stuck with limited chat features. LM Studio was good for models, but the chat experience felt basic. We'd end up running everything in the background while building UIs and generating images separately. -**Clone & Run:** -```bash -git clone https://github.com/claraverse-space/ClaraVerseAI.git && cd ClaraVerseAI && ./quickstart.sh -``` +**That's when it hit us: Why can't everything be in a single app?** -
+ClaraVerse is our answer to that question. ---- +> ClaraVerse is not another OpenWebUI (which is excellent for chat) or Ollama - everything is just scattered around and I'm trying to build a single app where people can just install it and use it. -## ✨ What's Included +
-Everything runs locally - no external APIs needed: +## 🎯 **The Problem** -| Service | Purpose | -|---------|---------| -| **Frontend** | React app on port 80 | -| **Backend** | Go API on port 3001 | -| **MongoDB** | Conversations & workflows | -| **MySQL** | Providers & models | -| **Redis** | Job scheduling | -| **SearXNG** | Web search (no API key!) | -| **E2B** | Code execution (no API key!) | +
---- +| **The Subscription Hell** 💸 | **The Context-Loss Nightmare** 🔄 | +|:---|:---| +| • Claude Pro: **$20/month**
• ChatGPT Plus: **$20/month**
• GitHub Copilot: **$10/month**
• Midjourney: **$10/month**
• N8N Cloud: **$20/month**

**Total: $960/year** 😱 | • Chat in Claude → Code in VScode
• Prompt in LLM → Use it in ComfyUI
• Deploy with Ollama → Run in OpenWebUI

**Lost context with every switch** 🤦 | -## Star Why ClaraVerse? +
+ +
-**Self-hosting isn't enough.** Most "privacy-focused" chat UIs still store your conversations in MongoDB or PostgreSQL. ClaraVerse goes further with **browser-local storage**—even the server admin can't read your chats. +## ✨ **The Solution: ClaraVerse** -| Feature | ClaraVerse | ChatGPT/Claude | Open WebUI | LibreChat | -|---------|------------|----------------|------------|-----------| -| **Browser-Local Storage** | ✅ Never touches server | ❌ Cloud-only | ❌ Stored in MongoDB | ❌ Stored in MongoDB | -| **Server Can't Read Chats** | ✅ Zero-knowledge architecture | ❌ Full access | ❌ Admin has full access | ❌ Admin has full access | -| **Self-Hosting** | ✅ Optional | ❌ Cloud-only | ✅ Required | ✅ Required | -| **Works Offline** | ✅ Full offline mode | ❌ Internet required | ⚠️ Server required | ⚠️ Server required | -| **Multi-Provider** | ✅ OpenAI, Claude, Gemini, local | ❌ Single provider | ✅ Multi-provider | ✅ Multi-provider | -| **Visual Workflow Builder** | ✅ Chat + n8n combined | ❌ | ❌ | ❌ | -| **Interactive Prompts** | ✅ AI asks questions mid-chat | ❌ | ⚠️ Pre-defined only | ❌ | +
-> **50,000+ downloads** | The only AI platform where conversations never touch the server—even when self-hosted +### **One App. Six Tools. Zero Compromises.** -
-📋 Advanced Setup & Troubleshooting +
-### Prerequisites -- Docker & Docker Compose installed -- 4GB RAM minimum (8GB recommended) + + + + + + + + + + + + + + + + +
+ +
+ + + ### **Clara AI** + **Smart Assistant** + + `Advanced Chat` `Code Gen` + `Voice` `Files` `MCP Tools` `Model Manager` + +
+ +
+ +
+ + + ### **LumaUI** + **Code Builder** + + `Live Preview` `WebContainer` + `React/Vue/JS` `Hot Reload` + +
+ +
+ +
+ + + ### **ComfyUI** + **Image Studio** + + `SDXL/Flux` `LoRA` + `Batch Gen` `100% Private` + +
+ +
+ +
+ + + ### **N8N** + **Automation** + + `Visual Workflows` `1000+ APIs` + `Scheduling` `Data Pipelines` + +
+ +
+ +
+ + + ### **Agent Studio** + **AI Agents** + + `Autonomous Agents` `MCP Integration` + `Visual Designer` `Custom Nodes` + +
+ +
+ +
+ + + ### **Community** + **Sharing Hub** + + `Content Sharing` `Custom Nodes` + `Workflows` `Collaboration` + +
+ +
+ +
+ + + ### **Widgets** + **Dashboard** + + `Drag & Drop` `Live Updates` + `Persistent` `Customizable` + +
+ +
+ +
+ + + ### **MCP Ecosystem** + **Tool Integration** + + `20+ MCP Servers` `Desktop Automation` + `Browser Control` `File System` + +
+ +
+ +
+ +
-### Manual Installation +
-```bash -# 1. Clone -git clone https://github.com/claraverse-space/ClaraVerseAI.git -cd ClaraVerseAI +
-# 2. Configure -cp .env.default .env +## 🔐 **Privacy First, Always** -# 3. Start -docker compose up -d +
-# 4. Verify -docker compose ps +```diff ++ 100% Local Processing Your data never leaves your machine ++ Zero Telemetry We can't see what you're doing ++ Open Source Every line of code is auditable ++ Works Offline No internet? No problem! ``` -### Troubleshooting +
+ +
+ +## 🚀 **Quick Start** + +### **Option 1: Desktop App** (Recommended) + +
+ +[![Windows](https://img.shields.io/badge/Windows-0078D6?style=for-the-badge&logo=windows&logoColor=white)](https://github.com/badboysm890/ClaraVerse/releases/latest) +[![macOS](https://img.shields.io/badge/macOS-000000?style=for-the-badge&logo=apple&logoColor=white)](https://github.com/badboysm890/ClaraVerse/releases/latest) +[![Linux](https://img.shields.io/badge/Linux-FCC624?style=for-the-badge&logo=linux&logoColor=black)](https://github.com/badboysm890/ClaraVerse/releases/latest) + +
+ +### **Option 2: Development Build** ```bash -# Run diagnostics -./diagnose.sh # Linux/Mac -diagnose.bat # Windows +# Clone the repository +git clone https://github.com/badboysm890/ClaraVerse.git +cd ClaraVerse -# View logs -docker compose logs -f backend +# Install dependencies +npm install + +# Run in development mode +npm run electron:dev:hot +``` -# Restart -docker compose restart +### **Option 3: Docker** 🐳 -# Fresh start -docker compose down -v && docker compose up -d +```bash +# Coming soon! ``` - +
---- +## 📊 **Why ClaraVerse?** -## Shield Browser-Local Storage: True Zero-Knowledge Privacy +
-**The Problem with Traditional Self-Hosted Chat UIs:** +| Feature | ClaraVerse | Others | +|:--------|:----------:|:------:| +| **All-in-One Platform** | ✅ | ❌ | +| **100% Local Processing** | ✅ | ❌ | +| **No Subscriptions** | ✅ | ❌ | +| **Context Sharing** | ✅ | ❌ | +| **Community Hub** | ✅ | ❌ | +| **Autonomous Agents** | ✅ | ⚠️ | +| **MCP Tool Ecosystem** | ✅ | ❌ | +| **Open Source** | ✅ | ⚠️ | +| **Offline Mode** | ✅ | ❌ | +| **Custom Models** | ✅ | ⚠️ | +| **Enterprise Ready** | ⌛ | 💰 | -When you self-host Open WebUI or LibreChat, conversations are stored in your MongoDB database. You control the server, but the data still exists in a queryable database. +
-```python -# Traditional self-hosted architecture -User → Server → MongoDB - ↓ - db.conversations.find({user_id: "123"}) # Admin can read everything -``` +
-**ClaraVerse's Zero-Knowledge Architecture:** +## 🏗️ **Architecture** -Conversations stay in your browser's IndexedDB and **never touch the server or database**. The server only proxies API calls to LLM providers—it never sees or stores message content. +
-```python -# ClaraVerse browser-local mode -User → IndexedDB (browser only, never leaves device) - → Server (API proxy only, doesn't log or store) +```mermaid +graph LR + A[🎨 React UI] --> B[⚡ Unified API] + B --> C[🧠 Clara Core] + B --> D[🔧 LumaUI] + B --> E[🎨 ComfyUI] + B --> F[🔄 N8N] + B --> G[🤖 Agent Studio] + B --> H[📊 Widgets] + B --> I[👥 Community] + B --> J[🔗 MCP Ecosystem] + + C --> K[Llama.cpp] + C --> L[Vision] + C --> M[Voice] + + G --> N[Autonomous Agents] + G --> O[Visual Designer] + + J --> P[Desktop Automation] + J --> Q[Browser Control] + J --> R[File System] + + style A fill:#4A90E2 + style C fill:#FF6B6B + style E fill:#4ECDC4 + style G fill:#9B59B6 + style I fill:#E67E22 + style J fill:#27AE60 ``` -### Why This Matters +
-✅ **Host for Teams Without Liability**: Even as server admin, you **cannot** access user conversations -✅ **True Compliance**: No server-side message retention = simplified GDPR/HIPAA compliance -✅ **No Database Bloat**: Messages aren't stored in MongoDB—database only holds accounts and settings -✅ **Air-Gap Capable**: Browser caches conversations; works completely offline after initial load -✅ **Zero Backup Exposure**: Database backups don't contain sensitive chat content +
-### Per-Conversation Privacy Control +## 🌟 **Features Roadmap** -Unlike competitors that force you into one mode, ClaraVerse lets you choose **per conversation**: +
-- **Work Projects**: Browser-local mode (100% offline, zero server access) -- **Personal Chats**: Cloud-sync mode (encrypted backup for mobile access) -- **Switch Anytime**: Toggle privacy mode without losing conversation history +| Status | Feature | ETA | +|:-------|:--------|:----| +| ✅ | Clara AI Assistant | **Released** | +| ✅ | LumaUI Code Builder | **Released** | +| ✅ | ComfyUI Integration | **Released** | +| ✅ | N8N Workflows | **Released** | +| ✅ | Agent Studio (Advanced) | **Released** | +| ✅ | Community Hub | **Released** | +| ✅ | MCP Ecosystem (20+ Servers) | **Released** | +| 🚧 | Docker Image for Remote servers | Q3 2025 | +| 🚧 | Mobile App with Offline Support | Q3 2025 | +| 🚧 | Cloud Sync (Optional) | Q4 2025 | +| 📋 | Plugin Marketplace | Q4 2025 | +| 📋 | Team Collaboration | Q4 2025 | -**This is privacy-first architecture done right.** +
+ +> **🚧 Development Status** +> ClaraVerse is actively evolving! While core features are stable, some components may change as we work toward v1.0. We prioritize stability but welcome your feedback on improvements and new features. Join our [Discord](https://discord.gg/j633fsrAne) to stay updated! 🚀 ---- +
-## Images Feature Showcase in a Nutshell +## 🤝 **Community & Support**
-### Natural Chat Interface -Natural Chat with Multiple AI Models +[![Discord](https://img.shields.io/badge/Discord-5865F2?style=for-the-badge&logo=discord&logoColor=white)](https://discord.gg/j633fsrAne) +[![Reddit](https://img.shields.io/badge/Reddit-FF4500?style=for-the-badge&logo=reddit&logoColor=white)](https://www.reddit.com/r/claraverse/) +[![Twitter](https://img.shields.io/badge/Twitter-1DA1F2?style=for-the-badge&logo=twitter&logoColor=white)](https://twitter.com/claraverse) +[![YouTube](https://img.shields.io/badge/YouTube-FF0000?style=for-the-badge&logo=youtube&logoColor=white)](https://youtube.com/@claraverse) -*Chat naturally with GPT-4, Claude, Gemini, and more - all in one unified interface* +### **📚 Resources** + +[Documentation](https://github.com/badboysm890/ClaraVerse/tree/main/docs) • +[API Reference](https://github.com/badboysm890/ClaraVerse/tree/main/docs/api) • +[Tutorials](https://github.com/badboysm890/ClaraVerse/tree/main/docs/tutorials) • +[FAQ](https://github.com/badboysm890/ClaraVerse/wiki/FAQ) + +

-### Clara Memory - Context-Aware Conversations -Clara Memory Feature +## 💝 **Contributors** -*Clara remembers your preferences and conversation context across sessions* -

Clara's memory system: She can remember which is needed in Short Term Memory and
Archive rest of the Memories that's not used very often

+
-
+ + + + + + +### **How to Contribute** -### Smart Multi-Agent Orchestration, Chat with Clara to Create your crew of agents -Smart Agents Collaboration +We love contributions! Check our [Contributing Guide](CONTRIBUTING.md) to get started. -*Coordinate multiple specialized AI agents for complex workflows* -

Clara's Integrated Architecture allows Chat and Agents to use and
share Integration and Automate the chat workflow automatically

+```mermaid +graph LR + A[🍴 Fork] --> B[🔧 Code] + B --> C[✅ Test] + C --> D[📤 PR] + D --> E[🎉 Merge] +``` +

-### Private AI Processing - Nothing needs to be stored on the server -Privacy-First Architecture +## 🙏 **Acknowledgments** + +
+ +**Built on the shoulders of giants:** -*Browser-local storage ensures your data stays private - even server admins can't access your conversations* +[llama.cpp](https://github.com/ggml-org/llama.cpp) • +[llama-swap](https://github.com/mostlygeek/llama-swap) • +[faster-whisper](https://github.com/SYSTRAN/faster-whisper) • +[ComfyUI](https://github.com/comfyanonymous/ComfyUI) • +[N8N](https://github.com/n8n-io/n8n) + +### **Special Thanks** + +☕ **Coffee Supporters** • 🌟 **Star Gazers** • 🐛 **Bug Hunters** • 💡 **Feature Suggesters**
---- +
+ +## 📄 **License** + +
-## Features Key Features - -### Lock **Privacy & Security First** -- **Browser-Local Storage**: Conversations stored in IndexedDB, never touch server—even admins can't read your chats -- **Zero-Knowledge Architecture**: Server only proxies LLM API calls, doesn't log or store message content -- **Per-Conversation Privacy**: Choose browser-local (100% offline) or cloud-sync (encrypted backup) per chat -- **Local JWT Authentication**: Secure authentication with Argon2id password hashing -- **True Offline Mode**: Works completely air-gapped after initial load—no server dependency - -### AI **Universal AI Access** -- **Multi-Provider Support**: OpenAI, Anthropic Claude, Google Gemini, and any OpenAI-compatible endpoint -- **Bring Your Own Key (BYOK)**: Use existing API accounts or free local models -- **400+ Models Available**: From GPT-4o to Llama, Mistral, and specialized models -- **Unified Interface**: One workspace for all your AI needs - -### Rocket **Advanced Capabilities** -- **Visual Workflow Builder**: Drag-and-drop workflow designer with auto-layout—chat to create, visual editor to refine -- **Hybrid Block Architecture**: Variable blocks, LLM blocks, and Code blocks (execute tools without LLM overhead) -- **Interactive Prompts**: AI asks clarifying questions mid-conversation with typed forms (text, select, checkbox) -- **Real-Time Streaming**: WebSocket-based chat with automatic reconnection and conversation resume -- **Tool Execution**: Code generation, image creation, web search, file analysis with real-time status tracking -- **Response Versioning**: Generate, compare, and track multiple versions (add details, make concise, no search) - -### Globe **Cross-Platform & Flexible** -- **Desktop Apps**: Native Windows, macOS, and Linux applications -- **Web Interface**: Browser-based access via React frontend -- **Mobile Ready**: Responsive design for tablets and phones -- **P2P Sync**: Device-to-device synchronization without cloud storage -- **Enterprise Deployment**: Self-host for complete organizational control - -### Tools **Developer-Friendly** -- **MCP Bridge**: Native Model Context Protocol support—connect any MCP-compatible tool seamlessly -- **Open API**: RESTful + WebSocket APIs for custom integrations -- **Plugin System**: Extend functionality with custom tools and connectors -- **Docker Support**: One-command deployment with `docker compose` -- **GitHub, Slack, Notion Integration**: Pre-built connectors for your workflow -- **Database Connections**: Query and analyze data with AI assistance +**MIT License** - Use it, modify it, sell it, we don't mind! (If want you to give credit, that would be awesome but not mandatory 😉) + +See [LICENSE](LICENSE) for details. + +
--- -## Target Our Mission +
-**Building the best AI interface experience—without compromising your privacy.** +
-While other AI tools force you to choose between features and privacy, ClaraVerse refuses that trade-off. We believe you deserve both: a powerful, intuitive interface AND complete data sovereignty. +### **🚀 Ready to revolutionize your AI workflow?** -### Why ClaraVerse is Different +
-Most "privacy-focused" AI tools sacrifice usability for security. Open WebUI and others offer self-hosting, but you're still limited to basic chat interfaces. ClaraVerse goes further: + + Download + -Design **Best-in-Class Interface** -- Intuitive, polished UI that rivals ChatGPT and Claude -- Real-time streaming with automatic reconnection -- Smart context management across sessions -- Multi-modal support (text, images, code, files) -- Clara Memory: Remembers what matters, archives what doesn't +
+
-Lock **Privacy WITHOUT Compromise** -- **Browser-local storage**: Conversations in IndexedDB, never touch server/database -- **Zero-knowledge architecture**: Server admins cannot read user chats—even in self-hosted deployments -- **Per-conversation privacy**: Toggle browser-local (offline) vs cloud-sync (encrypted) per chat -- **Air-gap capable**: Works 100% offline after initial load, no server dependency -- **Local authentication**: JWT with Argon2id password hashing, no external auth services -- **Open source (AGPL-3.0)**: Verify and audit security yourself +**⭐ Star us** • **🍴 Fork us** • **💬 Join us** -Plugin **Extensibility That Matters** -- **MCP Bridge**: Native Model Context Protocol integration for seamless tool connections -- Multi-agent orchestration: Coordinate specialized AI agents for complex workflows -- 400+ models: OpenAI, Anthropic, Google, Gemini, and any OpenAI-compatible endpoint -- BYOK: Use your own API keys or completely free local models -- Plugin ecosystem: GitHub, Slack, Notion, databases, and custom integrations +
-Platform **All-in-One Platform** -- Replaces ChatGPT (conversations), Midjourney (image generation), n8n (workflows) -- **Visual workflow builder + chat** in one interface—chat to design, visual editor to execute -- **Interactive prompts**: AI asks clarifying questions mid-conversation with typed forms -- **Memory auto-archival**: Active memory management—keeps context focused without manual cleanup -- Cross-platform: Desktop apps, web interface, mobile-ready -- P2P sync: Device-to-device synchronization without cloud dependencies +Made with ❤️ by developers who believe privacy is a right, not a privilege -### Our Promise +
-**Privacy-first doesn't mean features-last.** Every interface decision, every feature, every line of code is designed with this dual commitment: +
-1. **Security by Default**: Your data, your keys, your control -2. **Excellence by Design**: Experience that makes privacy feel effortless +
+
-### Built For +
-- **Individuals**: Super-powered AI workspace without surveillance -- **Developers**: Open API, MCP bridge, plugin system, complete source access -- **Teams**: Collaborate with AI while keeping confidential data on-premises -- **Enterprises**: Deploy infrastructure that complies with strictest data sovereignty requirements (GDPR, HIPAA, SOC2) +## 💭 **A Note From The Developer regarding incomplete features, bugs and docs** -> **50,000+ downloads worldwide** | Join developers and privacy advocates who refuse to compromise +
---- +
-## Book Documentation +### **Building ClaraVerse** -| Resource | Description | -|----------|-------------| -| [ Architecture Guide](backend/docs/ARCHITECTURE.md) | System design and component overview | -| [ API Reference](backend/docs/API_REFERENCE.md) | REST and WebSocket API documentation | -| [ Docker Guide](docs/DOCKER.md) | Comprehensive Docker deployment | -| [ Security Guide](backend/docs/FINAL_SECURITY_INSPECTION.md) | Security features and best practices | -| [ Admin Guide](backend/docs/ADMIN_GUIDE.md) | System administration and configuration | -| [ Developer Guide](backend/docs/DEVELOPER_GUIDE.md) | Contributing and local development | -| [ Quick Reference](backend/docs/QUICK_REFERENCE.md) | Common commands and workflows | +
---- + + + + + +
-## Architecture Architecture + -ClaraVerse is built with modern, production-ready technologies: +### **Solo Developer** -``` -┌─────────────────────────────────────────────────────────────┐ -│ Frontend Layer │ -│ React 19 + TypeScript + Tailwind CSS 4 │ -│ Zustand State + React Router 7 │ -└────────────────────┬────────────────────────────────────────┘ - │ WebSocket + REST API -┌────────────────────▼────────────────────────────────────────┐ -│ Backend Layer │ -│ Go 1.24 + Fiber Framework │ -│ Real-time Streaming + Tool Execution │ -└────────────────────┬────────────────────────────────────────┘ - │ - ┌────────────┼────────────┐ - │ │ │ - ┌────▼───┐ ┌───▼────┐ ┌───▼────┐ - │MongoDB │ │ Redis │ │SearXNG │ - │Storage │ │ Jobs │ │ Search │ - └────────┘ └────────┘ └────────┘ -``` +`9-6 Day Job` `Night Coding` +`Weekend Builds` `3 AM Commits` -**Technology Stack:** -- **Frontend**: React 19, TypeScript, Vite 7, Tailwind CSS 4, Zustand 5 -- **Backend**: Go 1.24, Fiber (web framework), WebSocket streaming -- **Database**: MongoDB for persistence, MySQL for models/providers, Redis for caching/jobs -- **Services**: SearXNG (search), E2B Local Docker (code execution - no API key!) -- **Deployment**: Docker Compose, Nginx reverse proxy -- **Auth**: Local JWT with Argon2id password hashing (v2.0 - fully local, no Supabase) +**One person. Three platforms.** +**Just having fun in doing this.** ---- + -## Palette Features in Detail - -### Real-Time Streaming Chat - -Experience instant AI responses with our WebSocket-based architecture: - -- **Chunked Streaming**: See responses as they're generated -- **Connection Recovery**: Automatic reconnection with conversation resume -- **Heartbeat System**: Maintains stable connections through proxies -- **Multi-User Support**: Concurrent conversations without interference - -### Tool Execution Engine - -Extend AI capabilities beyond text: - -| Tool | Description | Example | -|------|-------------|---------| -| **Code Generation** | Execute Python, JavaScript, Go code in sandboxed E2B environment | "Write and run a script to analyze this CSV" | -| **Image Generation** | Create images with DALL-E, Stable Diffusion, or local models | "Generate a logo for my startup" | -| **Web Search** | Real-time internet search via SearXNG | "What are the latest AI developments?" | -| **File Analysis** | Process PDFs, images, documents with vision models | "Summarize this 50-page report" | -| **Data Query** | Connect to databases and run SQL queries | "Show sales trends from our PostgreSQL" | - -### Bring Your Own Key (BYOK) - -Use your existing AI subscriptions: - -1. Add your API keys in `backend/providers.json` -2. Configure model preferences and rate limits -3. Switch between providers seamlessly -4. Or use completely free local models (Ollama, LM Studio) - -```json -{ - "providers": [ - { - "name": "OpenAI", - "api_key": "sk-your-key", - "models": ["gpt-4o", "gpt-4o-mini"] - }, - { - "name": "Anthropic", - "api_key": "your-key", - "models": ["claude-3-5-sonnet", "claude-3-opus"] - } - ] -} -``` + -### Multi-Agent Orchestration +### **Community Heroes** -Coordinate multiple AI agents for complex workflows: +`Discord Testers` `Bug Hunters` +`Coffee Supporters` `PR Contributors` -- **Specialized Agents**: Create agents with specific roles (researcher, coder, analyst) -- **Agent Collaboration**: Agents can communicate and share context -- **Workflow Automation**: Chain agent tasks for multi-step processes -- **Custom Instructions**: Define agent behavior with natural language +**You make this possible.** +**Every single day.** ---- +
-## Chart Use Cases - -### For Developers -- **Code Review & Debugging**: Get instant feedback on code quality -- **Documentation Generation**: Auto-generate docs from codebases -- **API Integration**: Connect ClaraVerse to your development workflow -- **Database Analysis**: Query and visualize data with AI assistance - -### For Businesses -- **Zero-Liability Hosting**: Host for teams without server-side chat storage—admins can't access conversations -- **True Data Sovereignty**: Browser-local mode means data never leaves employee devices, even when self-hosted -- **Simplified Compliance**: No message retention in database = easier GDPR/HIPAA compliance -- **Team Collaboration**: Shared AI workspace with access control and privacy guarantees -- **Custom Integrations**: Connect to Slack, Notion, GitHub, CRMs via visual workflow builder -- **Cost Control**: BYOK means you control AI spending with your own API keys - -### For Privacy Advocates -- **Browser-Local Storage**: Conversations never touch server—even when self-hosted, admins can't read chats -- **Zero-Knowledge Architecture**: Server only proxies API calls, doesn't log or store message content -- **Per-Conversation Privacy**: Choose offline browser-local or encrypted cloud-sync per conversation -- **Air-Gapped Operation**: Works 100% offline after initial load—no server dependency -- **Open Source**: Verify zero-knowledge claims yourself or hire security auditors -- **No Database Retention**: Messages not stored in MongoDB—simplified compliance for GDPR/HIPAA - -### For Researchers -- **Experiment with Models**: Test 400+ models in one interface -- **Dataset Analysis**: Process large datasets with AI assistance -- **Literature Review**: Search and summarize academic papers -- **Reproducible Workflows**: Save and share AI-assisted research processes +
---- +
+📖 Read the Full Story -## Map Roadmap - -### ✅ Completed (v2.0 - Current Version) -- [x] **Browser-local storage** (IndexedDB) with zero-knowledge architecture -- [x] **Visual workflow builder** with drag-and-drop interface and auto-layout -- [x] **Interactive prompts** (AI asks questions mid-conversation with typed forms) -- [x] **Per-conversation privacy toggle** (browser-local vs cloud-sync) -- [x] Multi-provider LLM support (OpenAI, Anthropic, Google, OpenAI-compatible) -- [x] Real-time WebSocket streaming with automatic reconnection -- [x] Tool execution (code, image generation, web search) -- [x] Response versioning (regenerate, add details, make concise, etc.) -- [x] Memory system with auto-archival and scoring -- [x] Hybrid block architecture (Variable, LLM, Code blocks) -- [x] Docker-based deployment -- [x] BYOK (Bring Your Own Key) functionality -- [x] MongoDB + MySQL + Redis infrastructure -- [x] **Local JWT authentication** (v2.0 - replaced Supabase, fully offline) -- [x] **E2B Local Docker mode** (v2.0 - code execution without API key) -- [x] **Removed payment processing** (v2.0 - all users Pro tier by default) -- [x] **Removed CAPTCHA** (v2.0 - rate limiting only) -- [x] **100% offline core functionality** (v2.0 - no external service dependencies) -- [x] File upload support with previews (images, PDFs, documents, CSV, audio) -- [x] Markdown rendering with reasoning extraction - -### 🚧 In Progress (v1.1) -- [ ] Desktop applications (Windows, macOS, Linux) -- [ ] Mobile apps (iOS, Android) -- [ ] P2P device synchronization -- [ ] Enhanced multi-agent orchestration -- [ ] Plugin marketplace - -### 🔮 Planned (v2.0 and beyond) -- [ ] Local LLM integration (Ollama, LM Studio native support) -- [ ] Voice input/output -- [ ] Advanced RAG (Retrieval-Augmented Generation) -- [ ] Workspace collaboration features -- [ ] Browser extension -- [ ] Kubernetes deployment templates -- [ ] Enterprise SSO integration - -[View full roadmap →](https://github.com/claraverse-space/ClaraVerseAI/projects) +
---- +> **"Why are some docs incomplete? Why does this feature need work?"** -## Handshake Contributing +Here's the honest truth... -We welcome contributions from developers of all skill levels! ClaraVerse is built by the community, for the community. +I'm a **single developer** building ClaraVerse during nights and weekends while working a **9-6 day job**. I'm maintaining **3 platforms**, fixing bugs, adding features, answering Discord questions, reviewing PRs, and somehow trying to keep documentation updated. -### How to Contribute +
-1. **Fork** the repository -2. **Create** a feature branch: `git checkout -b feature/amazing-feature` -3. **Make** your changes and add tests -4. **Run** linting: `npm run lint && go vet ./...` -5. **Commit** with clear messages: `git commit -m 'Add amazing feature'` -6. **Push** to your fork: `git push origin feature/amazing-feature` -7. **Open** a Pull Request with a detailed description +### **🌟 The Amazing People Making This Possible** -### Contribution Areas + + + + + + + +
-We especially welcome help in these areas: +**💬 Discord Community** +
+UI Testing • Feature Ideas • Bug Reports -- Bug **Bug Fixes**: Check [open issues](https://github.com/claraverse-space/ClaraVerseAI/issues) -- Docs **Documentation**: Improve guides, add examples, fix typos -- Language **Translations**: Help us reach non-English speakers -- Design **UI/UX**: Design improvements and accessibility -- Testing **Testing**: Add unit tests, integration tests, E2E tests -- Plugin **Integrations**: Build connectors for new tools and services -- AI **Models**: Add support for new LLM providers +
-### Development Setup +**👨‍💻 @aruntemme** +
+ClaraCore Dev • Notebook Features -See [DEVELOPER_GUIDE.md](backend/docs/DEVELOPER_GUIDE.md) for detailed instructions. +
-Quick start for contributors: +**☕ Coffee Supporters** +
+Keeping 3 AM Sessions Alive -```bash -# Install dependencies -make install +
-# Start development environment with hot reload -./dev.sh +**🐛 Issue Reporters** +
+Detailed Reports • Patience -# Run tests -cd frontend && npm run test -cd backend && go test ./... +
-# Check code quality -npm run lint && npm run format -go vet ./... && go fmt ./... -``` +
-### Code of Conduct +**The Reality Check:** +- Docs written with **Claude & GPT** during my **2-3 hours of free time** +- Some nights too exhausted to write proper commit messages +- Every feature request matters, but time is limited +- Your patience literally keeps this project alive -We are committed to providing a welcoming and inclusive environment. Please read our [Code of Conduct](CODE_OF_CONDUCT.md) before participating. +
---- +
-## Community Community +**🙏 Thank you for being part of this crazy journey** -Join thousands of privacy-conscious developers and AI enthusiasts: +*Building an AI platform alone is insane.* +*Your support makes it possible.* -- Discord **[Discord](https://discord.com/invite/j633fsrAne)**: Real-time chat and support -- X **[Twitter/X](https://x.com/clara_verse_)**: Updates and announcements -- TikTok **[TikTok](https://www.tiktok.com/@claraversehq)**: Short-form content and demos -- Newsletter **[Newsletter](https://claraverse.space/newsletter)**: Monthly updates and tips -- YouTube **[YouTube](https://www.youtube.com/@ClaraVerseAI)**: Tutorials and demos -- LinkedIn **[LinkedIn](https://linkedin.com/company/claraverse)**: Professional updates +— @badboysm890, probably debugging at 3 AM ☕ -### Show Your Support +
-If ClaraVerse has helped you, consider: +
-- Star **Star** this repository -- Bug **Report bugs** and suggest features -- Share **Share** with colleagues and on social media -- Sponsor **Sponsor** development ([GitHub Sponsors](https://github.com/sponsors/claraverse-space)) -- Code **Contribute** code, docs, or designs +
+ +
--- +
-## License License +## ❓ **Frequently Asked Questions** -ClaraVerse is licensed under the **GNU Affero General Public License v3.0 (AGPL-3.0)** - see the [LICENSE](LICENSE) file for details. +
-### What This Means: +### **Quick Answers to Common Questions** -**You ARE free to:** -- ✅ **Use commercially** - Host ClaraVerse as a service, even for profit -- ✅ **Modify** - Customize and improve the software -- ✅ **Distribute** - Share with others -- ✅ **Private use** - Use internally in your organization +

+ Installation × + Models × + Features × + Troubleshooting +

-**BUT you MUST:** -- 📤 **Share modifications** - Any changes must be open-sourced under AGPL-3.0 -- 🌐 **Network copyleft** - If you host ClaraVerse as a service, users must have access to your source code -- 📝 **Credit developers** - Preserve copyright and license notices -- 🔓 **Give back to the community** - Improvements benefit everyone +
-### Why AGPL-3.0? +
-We chose AGPL-3.0 to ensure that: -1. **ClaraVerse remains free forever** - No one can take it private -2. **The community benefits from all improvements** - Even from hosted/SaaS deployments -3. **Developers get credit** - Your contributions are always attributed -4. **Big tech gives back** - Companies using ClaraVerse must contribute improvements +
+ + 🚀 Installation & Setup + -**ClaraVerse is and will remain free and open-source forever.** +
---- + + + + + +
-## Thanks Acknowledgments +### **System Requirements** -ClaraVerse is built on the shoulders of giants. Special thanks to: +| Platform | Requirements | +|:---------|:------------| +| **Windows** | Win 10/11 • 8GB RAM | +| **macOS** | 10.15+ • M1/Intel | +| **Linux** | Ubuntu 18.04+ • FUSE | -- **[Go Fiber](https://gofiber.io/)** - Lightning-fast web framework -- **[React](https://react.dev/)** - UI library -- **[Anthropic](https://anthropic.com/)**, **[OpenAI](https://openai.com/)**, **[Google](https://ai.google.dev/)** - AI model providers -- **[SearXNG](https://github.com/searxng/searxng)** - Privacy-respecting search -- **[E2B](https://e2b.dev/)** - Code execution sandboxes (now running in local Docker mode!) -- **[Argon2](https://github.com/P-H-C/phc-winner-argon2)** - Password hashing library -- All our [contributors](https://github.com/claraverse-space/ClaraVerseAI/graphs/contributors) and community members + -**Note**: v2.0 moved from Supabase to local JWT authentication for complete offline capability +### **Common Issues** ---- +**🛡️ Windows Protection Warning?** +``` +More Info → Run Anyway +(App is safe but unsigned) +``` -## Help Troubleshooting +**🍎 macOS Won't Open?** +```bash +sudo xattr -r -d com.apple.quarantine \ + /Applications/ClaraVerse.app +``` -### Common Issues +
-
-WebSocket connection drops frequently - -**Solution**: Check nginx/proxy timeout settings. Ensure `proxy_read_timeout` is at least 300s: - -```nginx -location /ws/ { - proxy_pass http://localhost:3001; - proxy_http_version 1.1; - proxy_set_header Upgrade $http_upgrade; - proxy_set_header Connection "upgrade"; - proxy_read_timeout 300s; -} -```
-Docker containers won't start + + 🤖 AI Models & Performance + -**Solution**: Check for port conflicts and validate compose file: +
-```bash -# Check logs -docker compose logs backend + + + + + +
-# Validate configuration -docker compose config +### **Model Compatibility** + +✅ **Auto-Import From:** +- Ollama models +- LM Studio models +- GGUF files +- Custom models + +❌ **Not Required:** +- OpenAI API keys +- Cloud subscriptions +- Internet connection + + + +### **Performance Tips** + +**Clara AI Not Responding?** +1. Check AI provider settings +2. Switch to different model +3. Ensure Ollama running: `ollama serve` +4. Restart ClaraVerse + +**Memory Optimization:** +- Use smaller models (7B vs 70B) +- Close unused tabs +- Restart periodically + +
-# Check port usage -lsof -i :3001 -```
-Models not appearing in UI + + 🎨 Features & Development + + +
+ + + + + + +
+ +### **LumaUI Issues** -**Solution**: Verify `backend/providers.json` configuration: +**Project Won't Start?** +- Restart ClaraVerse +- Clear app cache +- Check 4GB+ RAM available +- Close heavy applications + +**Build Errors?** +- Update dependencies +- Check console logs +- Try simpler project first + + + +### **Image Generation** + +**Generation Fails?** +- Reduce resolution (512x512) +- Check GPU memory (6GB+) +- Close GPU applications +- Try CPU mode (slower) + +**Quality Issues?** +- Increase steps (20-50) +- Try different models +- Adjust CFG scale + +
+ +
+💡 Pro Tip: Multi-Tool Usage + +
+ +You can run **all tools simultaneously** - that's the magic of ClaraVerse! +Clara AI + LumaUI + ComfyUI can all share context and work together. + +
-1. Ensure file exists (copy from `providers.example.json`) -2. Check API keys are valid -3. Verify `enabled: true` for each provider -4. Restart backend: `docker compose restart backend`
-Build failures + + 🤖 Autonomous Agents & MCP + + +
-**Solution**: Ensure you have the correct versions: + + + + + +
-```bash -go version # Should be 1.24+ -node --version # Should be 20+ -python --version # Should be 3.11+ -``` +### **Autonomous Agent Features** + +✅ **Self-Executing Workflows** +- Autonomous task execution +- Multi-step reasoning +- Error self-correction +- Chain-of-thought processing + +✅ **MCP Tool Integration** +- 20+ available MCP servers +- Desktop automation +- Browser control +- File system access + + + +### **Community Features** + +**🌟 Content Sharing:** +- Share custom nodes +- Download workflows +- Community templates +- Security scanning + +**🔧 Agent Studio:** +- Visual workflow builder +- Drag-and-drop interface +- Custom node development +- Real-time testing + +
-Clear caches and reinstall: -```bash -make clean -make install -```
-For more help: -- Guide [Full troubleshooting guide](backend/docs/TROUBLESHOOTING.md) -- Discord [Discord support channel](https://discord.com/invite/j633fsrAne) -- Bug [Report an issue](https://github.com/claraverse-space/ClaraVerseAI/issues) +
---- +
-## Contact Contact +### **Quick Fixes for Common Problems** -- Website **Website**: [claraverse.space](https://claraverse.space) -- Email **Email**: [hello@claraverse.space](mailto:hello@claraverse.space) -- Enterprise **Enterprise**: [enterprise@claraverse.space](mailto:enterprise@claraverse.space) -- Bug **Bug Reports**: [GitHub Issues](https://github.com/claraverse-space/ClaraVerseAI/issues) -- Ideas **Feature Requests**: [GitHub Discussions](https://github.com/claraverse-space/ClaraVerseAI/discussions) +
---- +
+ + + + + + + + +
+ + + +**High Memory?** + + +• Close unused tabs
+• Use smaller models
+• Reduce batch sizes
+• Restart periodically +
+ +
+ + + +**Network Issues?** + + +• Check firewall
+• Try different network
+• Use offline mode
+• Verify antivirus +
+ +
+ + + +**GPU Problems?** + + +• Update drivers
+• Check CUDA version
+• Monitor VRAM usage
+• Try CPU fallback +
+ +
+ + + +**App Crashes?** + + +• Check error logs
+• Reinstall app
+• Clear cache/data
+• Report on GitHub +
+ +
+ +
+ +
-**Built with ❤️ by the ClaraVerse Community** +### **🆘 Still Need Help?** + + + + +  + + + +  + + + + +
+ +
-*Pioneering the new age of private, powerful AI for the Super Individual* +
-[⬆ Back to Top](#️-claraverse) +
+ +Can't find your answer? Join our Discord for real-time help!
+ +
diff --git a/assets/entitlements.mac.plist b/assets/entitlements.mac.plist new file mode 100644 index 00000000..8d30e7bc --- /dev/null +++ b/assets/entitlements.mac.plist @@ -0,0 +1,22 @@ + + + + + com.apple.security.cs.allow-jit + + com.apple.security.cs.allow-unsigned-executable-memory + + com.apple.security.cs.allow-dyld-environment-variables + + com.apple.security.network.client + + com.apple.security.network.server + + com.apple.security.files.user-selected.read-write + + com.apple.security.inherit + + com.apple.security.automation.apple-events + + + diff --git a/assets/icons/128x128.png b/assets/icons/128x128.png new file mode 100644 index 00000000..6de372a7 Binary files /dev/null and b/assets/icons/128x128.png differ diff --git a/assets/icons/16x16.png b/assets/icons/16x16.png new file mode 100644 index 00000000..7e46356d Binary files /dev/null and b/assets/icons/16x16.png differ diff --git a/assets/icons/24x24.png b/assets/icons/24x24.png new file mode 100644 index 00000000..0fcb05cf Binary files /dev/null and b/assets/icons/24x24.png differ diff --git a/assets/icons/256x256.png b/assets/icons/256x256.png new file mode 100644 index 00000000..87b906d2 Binary files /dev/null and b/assets/icons/256x256.png differ diff --git a/assets/icons/32x32.png b/assets/icons/32x32.png new file mode 100644 index 00000000..75fb4899 Binary files /dev/null and b/assets/icons/32x32.png differ diff --git a/assets/icons/48x48.png b/assets/icons/48x48.png new file mode 100644 index 00000000..d5c0c4ec Binary files /dev/null and b/assets/icons/48x48.png differ diff --git a/assets/icons/512x512.png b/assets/icons/512x512.png new file mode 100644 index 00000000..5da5eb41 Binary files /dev/null and b/assets/icons/512x512.png differ diff --git a/assets/icons/64x64.png b/assets/icons/64x64.png new file mode 100644 index 00000000..61f78c45 Binary files /dev/null and b/assets/icons/64x64.png differ diff --git a/docs/images/logo.png b/assets/icons/logo.png similarity index 100% rename from docs/images/logo.png rename to assets/icons/logo.png diff --git a/assets/icons/png/128x128.png b/assets/icons/png/128x128.png new file mode 100644 index 00000000..6de372a7 Binary files /dev/null and b/assets/icons/png/128x128.png differ diff --git a/assets/icons/png/16x16.png b/assets/icons/png/16x16.png new file mode 100644 index 00000000..7e46356d Binary files /dev/null and b/assets/icons/png/16x16.png differ diff --git a/assets/icons/png/24x24.png b/assets/icons/png/24x24.png new file mode 100644 index 00000000..0fcb05cf Binary files /dev/null and b/assets/icons/png/24x24.png differ diff --git a/assets/icons/png/256x256.png b/assets/icons/png/256x256.png new file mode 100644 index 00000000..87b906d2 Binary files /dev/null and b/assets/icons/png/256x256.png differ diff --git a/assets/icons/png/32x32.png b/assets/icons/png/32x32.png new file mode 100644 index 00000000..75fb4899 Binary files /dev/null and b/assets/icons/png/32x32.png differ diff --git a/assets/icons/png/48x48.png b/assets/icons/png/48x48.png new file mode 100644 index 00000000..d5c0c4ec Binary files /dev/null and b/assets/icons/png/48x48.png differ diff --git a/assets/icons/png/512x512.png b/assets/icons/png/512x512.png new file mode 100644 index 00000000..5da5eb41 Binary files /dev/null and b/assets/icons/png/512x512.png differ diff --git a/assets/icons/png/64x64.png b/assets/icons/png/64x64.png new file mode 100644 index 00000000..61f78c45 Binary files /dev/null and b/assets/icons/png/64x64.png differ diff --git a/assets/icons/png/README.md b/assets/icons/png/README.md new file mode 100644 index 00000000..06635b6a --- /dev/null +++ b/assets/icons/png/README.md @@ -0,0 +1,29 @@ +# Icon Requirements for Linux + +For proper icon display in Linux distributions, the following icon sizes are required: + +- 16x16.png +- 24x24.png +- 32x32.png +- 48x48.png +- 64x64.png +- 128x128.png +- 256x256.png +- 512x512.png + +Please ensure all PNG icons are properly optimized and follow these standard sizes. + +You can create these from your original logo.png using tools like ImageMagick: + +```bash +convert logo.png -resize 16x16 16x16.png +convert logo.png -resize 24x24 24x24.png +convert logo.png -resize 32x32 32x32.png +convert logo.png -resize 48x48 48x48.png +convert logo.png -resize 64x64 64x64.png +convert logo.png -resize 128x128 128x128.png +convert logo.png -resize 256x256 256x256.png +convert logo.png -resize 512x512 512x512.png +``` + +This directory structure is specifically referenced in the electron-builder configuration for Linux builds. diff --git a/frontend/src/assets/logo.png b/assets/icons/png/logo.png similarity index 100% rename from frontend/src/assets/logo.png rename to assets/icons/png/logo.png diff --git a/assets/icons/win/icon.ico b/assets/icons/win/icon.ico new file mode 100644 index 00000000..763db1ec Binary files /dev/null and b/assets/icons/win/icon.ico differ diff --git a/backend/.air.toml b/backend/.air.toml deleted file mode 100644 index 01128182..00000000 --- a/backend/.air.toml +++ /dev/null @@ -1,44 +0,0 @@ -root = "." -testdata_dir = "testdata" -tmp_dir = "tmp" - -[build] - args_bin = [] - bin = "./tmp/main" - cmd = "go build -o ./tmp/main ./cmd/server" - delay = 1000 - exclude_dir = ["assets", "tmp", "vendor", "testdata"] - exclude_file = [] - exclude_regex = ["_test.go"] - exclude_unchanged = false - follow_symlink = false - full_bin = "" - include_dir = [] - include_ext = ["go", "tpl", "tmpl", "html"] - include_file = [] - kill_delay = "0s" - log = "build-errors.log" - poll = false - poll_interval = 0 - rerun = false - rerun_delay = 500 - send_interrupt = false - stop_on_error = false - -[color] - app = "" - build = "yellow" - main = "magenta" - runner = "green" - watcher = "cyan" - -[log] - main_only = false - time = false - -[misc] - clean_on_exit = false - -[screen] - clear_on_rebuild = false - keep_scroll = true diff --git a/backend/.dockerignore b/backend/.dockerignore deleted file mode 100644 index 76cfcce8..00000000 --- a/backend/.dockerignore +++ /dev/null @@ -1,66 +0,0 @@ -# Git -.git -.gitignore -.gitattributes - -# Documentation -*.md -docs/ -README.md - -# Environment files (will be mounted at runtime) -.env -.env.local -.env.*.local - -# Database files (will be in persistent volumes) -*.db -*.db-shm -*.db-wal - -# Secure files (not for container) -secure_files/ - -# Provider configuration (will be mounted at runtime) -providers.json - -# Uploads directory (will be in persistent volumes) -uploads/ -*.tmp - -# Logs -*.log -logs/ - -# Build artifacts -claraverse -*.exe -*.exe~ -*.so -*.dylib -dist/ -build/ - -# MCP bridge builds (separate binary) -mcp-bridge/mcp-client -mcp-bridge/mcp-client.exe -mcp-bridge/*.exe~ - -# IDE and editor files -.vscode/ -.idea/ -*.swp -*.swo -*~ -.DS_Store - -# Testing -*_test.go -test/ -coverage.txt -*.out - -# Temporary files -tmp/ -temp/ -*.tmp diff --git a/backend/.env.example b/backend/.env.example deleted file mode 100644 index b296b730..00000000 --- a/backend/.env.example +++ /dev/null @@ -1,74 +0,0 @@ -# Server Configuration -PORT=3001 -ENVIRONMENT=development # Options: development, testing, production (REQUIRED) - -# MongoDB Configuration -# Flexible deployment - use any MongoDB instance: -# - Local Docker: mongodb://localhost:27017/claraverse -# - MongoDB Atlas: mongodb+srv://user:pass@cluster.mongodb.net/claraverse -# - Self-hosted: mongodb://user:pass@your-server:27017/claraverse -MONGODB_URI=mongodb://localhost:27017/claraverse - -# Encryption Configuration -# Generate a secure 32-byte hex key: openssl rand -hex 32 -# This key is used to encrypt user data (workflows, conversations, executions) -# WARNING: Losing this key means losing access to all encrypted data! -ENCRYPTION_MASTER_KEY= - -# Provider Configuration -PROVIDERS_FILE=providers.json - -# Supabase Authentication (REQUIRED in production, optional in development) -# Get these from https://app.supabase.com/project/_/settings/api -# SUPABASE_URL: Your project URL (e.g., https://xxxxx.supabase.co) -# SUPABASE_KEY: Your service role key (not anon key) for backend validation -# WARNING: Leaving these empty in production will cause the server to terminate -SUPABASE_URL= -SUPABASE_KEY= - -# SearXNG Web Search (Optional) -# Docker: Use http://searxng:8080 (automatically provided in docker-compose) -# Local dev: Set up your own SearXNG instance at http://localhost:8080 -SEARXNG_URL=http://searxng:8080 - -# File Upload Configuration -# Directory for storing uploaded files -UPLOAD_DIR=./uploads -# Maximum file size in bytes (20MB default) -MAX_FILE_SIZE=20971520 - -# Security Configuration -# CORS: Comma-separated list of allowed origins (REQUIRED in production) -# Development: http://localhost:5173,http://localhost:3000 -# Production: https://yourdomain.com,https://www.yourdomain.com -ALLOWED_ORIGINS=http://localhost:5173,http://localhost:3000 - -# FRONTEND_URL: The URL of your frontend (REQUIRED for payment redirects) -# This is where users will be redirected after completing payment -# Development: http://localhost:5173 or http://localhost:3001 -# Production: https://yourdomain.com -# Ngrok: https://your-ngrok-url.ngrok-free.app -FRONTEND_URL=http://localhost:5173 -# Backend Public URL (for generating absolute download URLs) -# Used by tools to generate full URLs that LLMs can use directly -# Docker: http://localhost:3001 (or your public domain in production) -# Production: https://api.yourdomain.com -BACKEND_URL=http://localhost:3001 - -# E2B Code Interpreter Service (REQUIRED for Python code execution) -# Get your API key from https://e2b.dev/dashboard -# Example: E2B_API_KEY=e2b_xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx -E2B_API_KEY= -# E2B Service URL (automatically set in Docker, use http://localhost:8001 for local dev) -E2B_SERVICE_URL=http://e2b-service:8001 - -# Promotional Campaign Configuration -# Enable this to give new signups a temporary pro plan during the promotional period -# Example: January 2025 - give all new users 30 days of pro plan -PROMO_ENABLED=false -# Start date in RFC3339 format (UTC timezone) -PROMO_START_DATE=2025-01-01T00:00:00Z -# End date in RFC3339 format (UTC timezone) - exclusive -PROMO_END_DATE=2025-02-01T00:00:00Z -# Duration of promotional pro plan in days (from signup date) -PROMO_DURATION_DAYS=30 diff --git a/backend/.gitignore b/backend/.gitignore deleted file mode 100644 index a01d610f..00000000 --- a/backend/.gitignore +++ /dev/null @@ -1,33 +0,0 @@ -# Configuration files with secrets -providers.json -.env - -# Database -model_capabilities.db -*.db - -# Logs -*.log - -# IDE -.vscode/ -.idea/ -.claude/ - -# Build artifacts -claraverse-server -claraverse-server.exe -*.exe - -# OS -.DS_Store -Thumbs.db - -# Air hot reload -tmp/ -*.md - -*.venv/ -# Air hot reload temp directory -tmp/ -build-errors.log diff --git a/backend/Dockerfile b/backend/Dockerfile deleted file mode 100644 index 21a5f9c5..00000000 --- a/backend/Dockerfile +++ /dev/null @@ -1,112 +0,0 @@ -# Stage 1: Builder -FROM golang:1.25.5-alpine AS builder - -# Install build dependencies for CGO (required for SQLite) -RUN apk add --no-cache gcc musl-dev - -# Set working directory -WORKDIR /build - -# Copy dependency files first (for better layer caching) -COPY go.mod go.sum ./ - -# Download dependencies -RUN go mod download - -# Copy source code -COPY . . - -# Run critical tests before building (fail fast if tests fail) -# This ensures no broken code makes it into production -ARG SKIP_TESTS=false -RUN if [ "$SKIP_TESTS" = "false" ]; then \ - echo "=== Running test suite ===" && \ - echo "--- Testing database ---" && \ - CGO_ENABLED=1 go test -race -timeout 120s ./internal/database/... && \ - echo "--- Testing models ---" && \ - CGO_ENABLED=1 go test -race -timeout 120s ./internal/models/... && \ - echo "--- Testing services ---" && \ - CGO_ENABLED=1 go test -race -timeout 120s ./internal/services/... && \ - echo "--- Testing tools (file I/O) ---" && \ - CGO_ENABLED=1 go test -race -timeout 120s ./internal/tools/... && \ - echo "--- Testing execution (workflows) ---" && \ - CGO_ENABLED=1 go test -race -timeout 120s ./internal/execution/... && \ - echo "--- Testing filecache ---" && \ - CGO_ENABLED=1 go test -race -timeout 120s ./internal/filecache/... && \ - echo "--- Testing audio service ---" && \ - CGO_ENABLED=1 go test -race -timeout 120s ./internal/audio/... && \ - echo "--- Testing vision service ---" && \ - CGO_ENABLED=1 go test -race -timeout 120s ./internal/vision/... && \ - echo "--- Testing securefile service ---" && \ - CGO_ENABLED=1 go test -race -timeout 120s ./internal/securefile/... && \ - echo "--- Testing preflight ---" && \ - CGO_ENABLED=1 go test -race -timeout 120s ./internal/preflight/... && \ - echo "--- Testing integration ---" && \ - CGO_ENABLED=1 go test -race -timeout 120s ./tests/... && \ - echo "=== All tests passed ==="; \ - else echo "=== Skipping tests ==="; fi - -# Build the application with CGO enabled (required for modernc.org/sqlite) -# -ldflags "-s -w" strips debug info to reduce binary size -RUN CGO_ENABLED=1 GOOS=linux go build -ldflags "-s -w" -o claraverse ./cmd/server - -# Stage 2: Runtime -FROM alpine:latest - -# Install runtime dependencies including Chromium for PDF generation -# Includes emoji fonts and extended Unicode symbol support for PDF rendering -RUN apk add --no-cache \ - ca-certificates \ - tzdata \ - wget \ - chromium \ - nss \ - freetype \ - harfbuzz \ - ttf-freefont \ - font-noto \ - font-noto-emoji \ - font-noto-cjk \ - fontconfig - -# Rebuild font cache to register all installed fonts -RUN fc-cache -f -v - -# Create non-root user and group -RUN addgroup -g 1000 claraverse && \ - adduser -D -u 1000 -G claraverse claraverse - -# Set working directory -WORKDIR /app - -# Create necessary directories with proper permissions -RUN mkdir -p /app/data /app/config /app/uploads /app/logs /app/generated /app/secure_files && \ - chown -R claraverse:claraverse /app - -# Set environment variables for Chromium -ENV CHROME_BIN=/usr/bin/chromium-browser \ - CHROME_PATH=/usr/lib/chromium/ - -# Copy binary from builder stage -COPY --from=builder --chown=claraverse:claraverse /build/claraverse /app/claraverse - -# Copy example configuration files (if they exist) -COPY --chown=claraverse:claraverse providers.example.json /app/providers.example.json - -# Copy and set permissions for entrypoint script -COPY --chown=claraverse:claraverse docker-entrypoint.sh /app/docker-entrypoint.sh -RUN sed -i 's/\r$//' /app/docker-entrypoint.sh && chmod +x /app/docker-entrypoint.sh - -# Switch to non-root user -USER claraverse - -# Expose port -EXPOSE 3001 - -# Health check -HEALTHCHECK --interval=30s --timeout=3s --start-period=10s --retries=3 \ - CMD wget --no-verbose --tries=1 --spider http://localhost:3001/health || exit 1 - -# Set entrypoint and command -ENTRYPOINT ["/app/docker-entrypoint.sh"] -CMD ["/app/claraverse"] diff --git a/backend/Dockerfile.dev b/backend/Dockerfile.dev deleted file mode 100644 index c0673fc1..00000000 --- a/backend/Dockerfile.dev +++ /dev/null @@ -1,22 +0,0 @@ -FROM golang:1.25.5-alpine - -# Install air for hot reload -RUN go install github.com/air-verse/air@latest - -# Install build dependencies -RUN apk add --no-cache gcc musl-dev - -WORKDIR /app - -# Copy go mod files -COPY go.mod go.sum ./ -RUN go mod download - -# Copy source -COPY . . - -# Expose port -EXPOSE 3001 - -# Run with air for hot reload -CMD ["air", "-c", ".air.toml"] diff --git a/backend/cmd/server/main.go b/backend/cmd/server/main.go deleted file mode 100644 index 571ac8cf..00000000 --- a/backend/cmd/server/main.go +++ /dev/null @@ -1,1686 +0,0 @@ -package main - -import ( - "claraverse/internal/config" - "claraverse/internal/crypto" - "claraverse/internal/database" - "claraverse/internal/document" - "claraverse/internal/execution" - "claraverse/internal/filecache" - "claraverse/internal/handlers" - "claraverse/internal/jobs" - "claraverse/internal/middleware" - "claraverse/internal/models" - "claraverse/internal/preflight" - "claraverse/internal/services" - "claraverse/internal/tools" - "claraverse/pkg/auth" - "context" - "fmt" - "log" - "os" - "os/signal" - "path/filepath" - "strings" - "syscall" - "time" - - "github.com/ansrivas/fiberprometheus/v2" - "github.com/fsnotify/fsnotify" - "github.com/gofiber/contrib/websocket" - "github.com/gofiber/fiber/v2" - "github.com/gofiber/fiber/v2/middleware/cors" - "github.com/gofiber/fiber/v2/middleware/limiter" - "github.com/gofiber/fiber/v2/middleware/logger" - "github.com/gofiber/fiber/v2/middleware/recover" - "github.com/joho/godotenv" -) - -func main() { - log.SetFlags(log.LstdFlags | log.Lshortfile) - log.Println("🚀 Starting ClaraVerse Server...") - - // Load .env file (ignore error if file doesn't exist) - if err := godotenv.Load(); err != nil { - log.Printf("⚠️ No .env file found or error loading it: %v", err) - } else { - log.Println("✅ .env file loaded successfully") - } - - // Load configuration - cfg := config.Load() - log.Printf("📋 Configuration loaded (Port: %s, DB: MySQL)", cfg.Port) - - // Initialize MySQL database - if cfg.DatabaseURL == "" { - log.Fatal("❌ DATABASE_URL environment variable is required (mysql://user:pass@host:port/dbname?parseTime=true)") - } - db, err := database.New(cfg.DatabaseURL) - if err != nil { - log.Fatalf("❌ Failed to connect to database: %v", err) - } - defer db.Close() - - if err := db.Initialize(); err != nil { - log.Fatalf("❌ Failed to initialize database: %v", err) - } - - // Initialize MongoDB (optional - for builder conversations and user data) - var mongoDB *database.MongoDB - var encryptionService *crypto.EncryptionService - var userService *services.UserService - var builderConvService *services.BuilderConversationService - - mongoURI := os.Getenv("MONGODB_URI") - if mongoURI != "" { - log.Println("🔗 Connecting to MongoDB...") - var err error - mongoDB, err = database.NewMongoDB(mongoURI) - if err != nil { - log.Printf("⚠️ Failed to connect to MongoDB: %v (builder features disabled)", err) - } else { - defer mongoDB.Close(context.Background()) - log.Println("✅ MongoDB connected successfully") - - // Initialize encryption service - masterKey := os.Getenv("ENCRYPTION_MASTER_KEY") - if masterKey != "" { - encryptionService, err = crypto.NewEncryptionService(masterKey) - if err != nil { - log.Printf("⚠️ Failed to initialize encryption: %v", err) - } else { - log.Println("✅ Encryption service initialized") - } - } else { - // SECURITY: In production, encryption is required when MongoDB is enabled - environment := os.Getenv("ENVIRONMENT") - if environment == "production" { - log.Fatal("❌ CRITICAL SECURITY ERROR: ENCRYPTION_MASTER_KEY is required in production when MongoDB is enabled. Generate with: openssl rand -hex 32") - } - log.Println("⚠️ ENCRYPTION_MASTER_KEY not set - conversation encryption disabled (development mode only)") - } - - // Initialize user service - userService = services.NewUserService(mongoDB, cfg, nil) // usageLimiter set later - log.Println("✅ User service initialized") - - // Initialize builder conversation service - if encryptionService != nil { - builderConvService = services.NewBuilderConversationService(mongoDB, encryptionService) - log.Println("✅ Builder conversation service initialized") - } - } - } else { - log.Println("⚠️ MONGODB_URI not set - builder conversation persistence disabled") - } - - // Initialize chat sync service (requires MongoDB + EncryptionService) - var chatSyncService *services.ChatSyncService - if mongoDB != nil && encryptionService != nil { - chatSyncService = services.NewChatSyncService(mongoDB, encryptionService) - // Ensure indexes - if err := chatSyncService.EnsureIndexes(context.Background()); err != nil { - log.Printf("⚠️ Failed to ensure chat sync indexes: %v", err) - } - log.Println("✅ Chat sync service initialized (encrypted cloud storage)") - } - - // Initialize Redis service (for scheduler + pub/sub) - var redisService *services.RedisService - var schedulerService *services.SchedulerService - var executionLimiter *middleware.ExecutionLimiter - - if cfg.RedisURL != "" { - log.Println("🔗 Connecting to Redis...") - var err error - redisService, err = services.NewRedisService(cfg.RedisURL) - if err != nil { - log.Printf("⚠️ Failed to connect to Redis: %v (scheduler disabled)", err) - } else { - log.Println("✅ Redis connected successfully") - } - } else { - log.Println("⚠️ REDIS_URL not set - scheduler disabled") - } - - // Run preflight checks - checker := preflight.NewChecker(db) - results := checker.RunAll() - - // Exit if critical checks failed - if preflight.HasFailures(results) { - log.Println("\n❌ Pre-flight checks failed. Please fix the issues above before starting the server.") - os.Exit(1) - } - - log.Println("✅ All pre-flight checks passed") - - // Initialize services - providerService := services.NewProviderService(db) - modelService := services.NewModelService(db) - connManager := services.NewConnectionManager() - - // Initialize Prometheus metrics - services.InitMetrics(connManager) - log.Println("✅ Prometheus metrics initialized") - - // Initialize MCP bridge service - mcpBridge := services.NewMCPBridgeService(db, tools.GetRegistry()) - log.Println("✅ MCP bridge service initialized") - - chatService := services.NewChatService(db, providerService, mcpBridge, nil) // toolService set later after credential service init - - // Initialize agent service (requires MongoDB for scalable storage) - var agentService *services.AgentService - if mongoDB != nil { - agentService = services.NewAgentService(mongoDB) - // Ensure indexes for agents, workflows, and workflow_versions - if err := agentService.EnsureIndexes(context.Background()); err != nil { - log.Printf("⚠️ Failed to ensure agent indexes: %v", err) - } - log.Println("✅ Agent service initialized (MongoDB)") - } else { - log.Println("⚠️ MongoDB not available - agent builder features disabled") - } - - workflowGeneratorService := services.NewWorkflowGeneratorService(db, providerService, chatService) - log.Println("✅ Workflow generator service initialized") - - workflowGeneratorV2Service := services.NewWorkflowGeneratorV2Service(db, providerService, chatService) - log.Println("✅ Workflow generator v2 service initialized (multi-step with tool selection)") - - // Initialize tier service (requires MongoDB) - var tierService *services.TierService - if mongoDB != nil { - tierService = services.NewTierService(mongoDB) - log.Println("✅ Tier service initialized") - } - - // Initialize execution limiter (requires TierService + Redis) - if tierService != nil && redisService != nil { - executionLimiter = middleware.NewExecutionLimiter(tierService, redisService.Client()) - log.Println("✅ Execution limiter initialized") - } else { - log.Println("⚠️ Execution limiter disabled (requires TierService and Redis)") - } - - // Initialize usage limiter service (requires TierService + Redis + MongoDB) - var usageLimiter *services.UsageLimiterService - if tierService != nil && redisService != nil && mongoDB != nil { - usageLimiter = services.NewUsageLimiterService(tierService, redisService.Client(), mongoDB) - log.Println("✅ Usage limiter service initialized") - - // Inject usage limiter into user service for promo user counter reset - if userService != nil { - userService.SetUsageLimiter(usageLimiter) - log.Println("✅ Usage limiter injected into user service") - } - } else { - log.Println("⚠️ Usage limiter disabled (requires TierService, Redis, and MongoDB)") - } - - // Initialize execution service (requires MongoDB + TierService) - var executionService *services.ExecutionService - if mongoDB != nil { - executionService = services.NewExecutionService(mongoDB, tierService) - // Ensure indexes - if err := executionService.EnsureIndexes(context.Background()); err != nil { - log.Printf("⚠️ Failed to ensure execution indexes: %v", err) - } - log.Println("✅ Execution service initialized") - } - - // Initialize analytics service (minimal, non-invasive usage tracking) - var analyticsService *services.AnalyticsService - if mongoDB != nil { - analyticsService = services.NewAnalyticsService(mongoDB) - // Ensure indexes - if err := analyticsService.EnsureIndexes(context.Background()); err != nil { - log.Printf("⚠️ Failed to ensure analytics indexes: %v", err) - } - log.Println("✅ Analytics service initialized (minimal tracking)") - } - - // Initialize API key service (requires MongoDB + TierService) - var apiKeyService *services.APIKeyService - if mongoDB != nil { - apiKeyService = services.NewAPIKeyService(mongoDB, tierService) - // Ensure indexes - if err := apiKeyService.EnsureIndexes(context.Background()); err != nil { - log.Printf("⚠️ Failed to ensure API key indexes: %v", err) - } - log.Println("✅ API key service initialized") - } - - // Initialize credential service (requires MongoDB + EncryptionService) - var credentialService *services.CredentialService - if mongoDB != nil && encryptionService != nil { - credentialService = services.NewCredentialService(mongoDB, encryptionService) - // Ensure indexes - if err := credentialService.EnsureIndexes(context.Background()); err != nil { - log.Printf("⚠️ Failed to ensure credential indexes: %v", err) - } - log.Println("✅ Credential service initialized") - } - - // Initialize tool service (provides credential-filtered tools) - toolService := services.NewToolService(tools.GetRegistry(), credentialService) - log.Println("✅ Tool service initialized") - - // Set tool service on chat service (was initialized with nil earlier) - chatService.SetToolService(toolService) - - // Initialize and set tool predictor service for dynamic tool selection - toolPredictorService := services.NewToolPredictorService(db, providerService, chatService) - chatService.SetToolPredictorService(toolPredictorService) - log.Println("✅ Tool predictor service initialized") - - // Initialize memory services (requires MongoDB + EncryptionService) - var memoryStorageService *services.MemoryStorageService - var memoryExtractionService *services.MemoryExtractionService - var memorySelectionService *services.MemorySelectionService - var memoryDecayService *services.MemoryDecayService - var memoryModelPool *services.MemoryModelPool - if mongoDB != nil && encryptionService != nil { - memoryStorageService = services.NewMemoryStorageService(mongoDB, encryptionService) - log.Println("✅ Memory storage service initialized") - - // Initialize model pool for dynamic memory model selection - var err error - memoryModelPool, err = services.NewMemoryModelPool(chatService, db.DB) - if err != nil { - log.Printf("⚠️ Failed to initialize memory model pool: %v", err) - log.Println("⚠️ Memory extraction/selection services disabled (requires valid memory models)") - } else { - log.Println("✅ Memory model pool initialized") - - memoryExtractionService = services.NewMemoryExtractionService( - mongoDB, - encryptionService, - providerService, - memoryStorageService, - chatService, - memoryModelPool, - ) - log.Println("✅ Memory extraction service initialized") - - memorySelectionService = services.NewMemorySelectionService( - mongoDB, - encryptionService, - providerService, - memoryStorageService, - chatService, - memoryModelPool, - ) - log.Println("✅ Memory selection service initialized") - - // Set memory services on chat service - chatService.SetMemoryExtractionService(memoryExtractionService) - chatService.SetMemorySelectionService(memorySelectionService) - chatService.SetUserService(userService) - } - - memoryDecayService = services.NewMemoryDecayService(mongoDB) - log.Println("✅ Memory decay service initialized") - } else { - log.Println("⚠️ Memory services disabled (requires MongoDB + EncryptionService)") - } - - // Initialize scheduler service (requires Redis + MongoDB + AgentService + ExecutionService) - if redisService != nil && mongoDB != nil { - var err error - schedulerService, err = services.NewSchedulerService(mongoDB, redisService, agentService, executionService) - if err != nil { - log.Printf("⚠️ Failed to initialize scheduler: %v", err) - } else { - log.Println("✅ Scheduler service initialized") - } - } - - // Initialize PubSub service (requires Redis) - var pubsubService *services.PubSubService - if redisService != nil { - instanceID := fmt.Sprintf("instance-%d", time.Now().UnixNano()%10000) - pubsubService = services.NewPubSubService(redisService, instanceID) - if err := pubsubService.Start(); err != nil { - log.Printf("⚠️ Failed to start PubSub service: %v", err) - } else { - log.Printf("✅ PubSub service initialized (instance: %s)", instanceID) - } - } - - // Initialize workflow execution engine with block checker support - executorRegistry := execution.NewExecutorRegistry(chatService, providerService, tools.GetRegistry(), credentialService) - workflowEngine := execution.NewWorkflowEngineWithChecker(executorRegistry, providerService) - log.Println("✅ Workflow execution engine initialized (with block checker)") - - // Set workflow executor on scheduler and start it - if schedulerService != nil { - // Create a workflow executor callback that wraps the workflow engine - workflowExecutor := func(workflow *models.Workflow, inputs map[string]interface{}) (*models.WorkflowExecuteResult, error) { - // Create a dummy status channel (scheduled jobs don't need real-time updates) - statusChan := make(chan models.ExecutionUpdate, 100) - go func() { - for range statusChan { - // Drain channel - in Phase 4, this will publish to Redis pub/sub - } - }() - - result, err := workflowEngine.Execute(context.Background(), workflow, inputs, statusChan) - close(statusChan) - - if err != nil { - return &models.WorkflowExecuteResult{ - Status: "failed", - Error: err.Error(), - }, err - } - return &models.WorkflowExecuteResult{ - Status: result.Status, - Output: result.Output, - BlockStates: result.BlockStates, - Error: result.Error, - }, nil - } - - schedulerService.SetWorkflowExecutor(workflowExecutor) - if err := schedulerService.Start(context.Background()); err != nil { - log.Printf("⚠️ Failed to start scheduler: %v", err) - } else { - log.Println("✅ Scheduler started successfully") - } - } - - // Initialize authentication (Local JWT - v2.0) - var jwtAuth *auth.LocalJWTAuth - jwtSecret := os.Getenv("JWT_SECRET") - if jwtSecret == "" { - environment := os.Getenv("ENVIRONMENT") - if environment == "production" { - log.Fatal("❌ CRITICAL SECURITY ERROR: JWT_SECRET is required in production. Generate with: openssl rand -hex 64") - } - log.Println("⚠️ JWT_SECRET not set - authentication disabled (development mode)") - } else { - // Parse JWT expiry durations from environment variables - accessTokenExpiry := 15 * time.Minute // Default: 15 minutes - refreshTokenExpiry := 7 * 24 * time.Hour // Default: 7 days - - if accessExpiryStr := os.Getenv("JWT_ACCESS_TOKEN_EXPIRY"); accessExpiryStr != "" { - if parsed, err := time.ParseDuration(accessExpiryStr); err == nil { - accessTokenExpiry = parsed - } else { - log.Printf("⚠️ Invalid JWT_ACCESS_TOKEN_EXPIRY: %v, using default 15m", err) - } - } - - if refreshExpiryStr := os.Getenv("JWT_REFRESH_TOKEN_EXPIRY"); refreshExpiryStr != "" { - if parsed, err := time.ParseDuration(refreshExpiryStr); err == nil { - refreshTokenExpiry = parsed - } else { - log.Printf("⚠️ Invalid JWT_REFRESH_TOKEN_EXPIRY: %v, using default 7d", err) - } - } - - var err error - jwtAuth, err = auth.NewLocalJWTAuth(jwtSecret, accessTokenExpiry, refreshTokenExpiry) - if err != nil { - log.Fatalf("❌ Failed to initialize JWT authentication: %v", err) - } - log.Printf("✅ Local JWT authentication initialized (access: %v, refresh: %v)", accessTokenExpiry, refreshTokenExpiry) - } - - // Try loading configuration from database first - _, err = loadConfigFromDatabase(modelService, chatService, providerService) - if err != nil { - log.Printf("⚠️ Warning: Could not load config from database: %v", err) - } - - // Database starts empty - use admin UI to add providers and models - - // Initialize vision service (for describe_image tool) - // Must be after provider sync so model aliases are available - services.SetVisionDependencies(providerService, db) - services.InitVisionService() - - // Initialize audio service (for transcribe_audio tool) - // Uses the same provider service dependency set above - services.InitAudioService() - - // NOTE: providers.json file watcher removed - all provider management now in MySQL - - // Start background model refresh job (refreshes from database) - go startModelRefreshJob(providerService, modelService, chatService) - - // Run startup cleanup to delete orphaned files from previous runs - // This ensures zero retention policy is enforced even after server restarts - uploadDir := "./uploads" - fileCache := filecache.GetService() - fileCache.RunStartupCleanup(uploadDir) - - // Start background image cleanup job (also cleans orphaned files) - go startImageCleanupJob(uploadDir) - - // Start background document cleanup job - go startDocumentCleanupJob() - - // Start memory extraction worker (requires memory extraction service) - if memoryExtractionService != nil { - go startMemoryExtractionWorker(memoryExtractionService) - } - - // Start memory decay worker (requires memory decay service) - if memoryDecayService != nil { - go startMemoryDecayWorker(memoryDecayService) - } - - // Initialize Fiber app - app := fiber.New(fiber.Config{ - AppName: "ClaraVerse v1.0", - ReadTimeout: 360 * time.Second, // 6 minutes to handle long tool executions - WriteTimeout: 360 * time.Second, // 6 minutes to handle long tool executions - IdleTimeout: 360 * time.Second, // 6 minutes to handle long tool executions - BodyLimit: 50 * 1024 * 1024, // 50MB limit for chat messages with images and large conversations - }) - - // Middleware - app.Use(recover.New()) - app.Use(logger.New()) - - // Prometheus metrics middleware - prometheus := fiberprometheus.New("claraverse") - prometheus.RegisterAt(app, "/metrics") - app.Use(prometheus.Middleware) - log.Println("📊 Prometheus metrics endpoint enabled at /metrics") - - // Load rate limiting configuration - rateLimitConfig := middleware.LoadRateLimitConfig() - log.Printf("🛡️ [RATE-LIMIT] Loaded config: Global=%d/min, Public=%d/min, Auth=%d/min, WS=%d/min", - rateLimitConfig.GlobalAPIMax, - rateLimitConfig.PublicReadMax, - rateLimitConfig.AuthenticatedMax, - rateLimitConfig.WebSocketMax, - ) - - // CORS configuration with environment-based origins - allowedOrigins := os.Getenv("ALLOWED_ORIGINS") - if allowedOrigins == "" { - // Default to localhost for development - allowedOrigins = "http://localhost:5173,http://localhost:3000" - log.Println("⚠️ ALLOWED_ORIGINS not set, using development defaults") - } - - app.Use(cors.New(cors.Config{ - AllowOrigins: allowedOrigins, - AllowMethods: "GET,POST,PUT,DELETE,OPTIONS", - AllowHeaders: "Origin,Content-Type,Accept,Authorization", - AllowCredentials: true, // Required for cookies (JWT refresh tokens) - // Skip CORS check for external access endpoints - they have their own permissive CORS - Next: func(c *fiber.Ctx) bool { - path := c.Path() - return strings.HasPrefix(path, "/api/trigger") || strings.HasPrefix(path, "/api/external") - }, - })) - - log.Printf("🔒 [SECURITY] CORS allowed origins: %s (excluding /api/trigger and /api/external)", allowedOrigins) - - // Global API rate limiter - first line of DDoS defense - // Applies to all /api/* routes, excludes health checks and metrics - app.Use("/api", middleware.GlobalAPIRateLimiter(rateLimitConfig)) - log.Println("🛡️ [RATE-LIMIT] Global API rate limiter enabled") - - // Initialize handlers - healthHandler := handlers.NewHealthHandler(connManager) - providerHandler := handlers.NewProviderHandler(providerService) - modelHandler := handlers.NewModelHandler(modelService) - uploadHandler := handlers.NewUploadHandler("./uploads", usageLimiter) - downloadHandler := handlers.NewDownloadHandler() - secureDownloadHandler := handlers.NewSecureDownloadHandler() - conversationHandler := handlers.NewConversationHandler(chatService, builderConvService) - userHandler := handlers.NewUserHandler(chatService, userService) - wsHandler := handlers.NewWebSocketHandler(connManager, chatService, analyticsService, usageLimiter) - - // Initialize local auth handler (v2.0) - var localAuthHandler *handlers.LocalAuthHandler - if jwtAuth != nil && mongoDB != nil && userService != nil { - localAuthHandler = handlers.NewLocalAuthHandler(jwtAuth, userService) - log.Println("✅ Local auth handler initialized") - } - - // Initialize memory handler (requires memory services) - var memoryHandler *handlers.MemoryHandler - if memoryStorageService != nil && memoryExtractionService != nil { - memoryHandler = handlers.NewMemoryHandler(memoryStorageService, memoryExtractionService, chatService) - log.Println("✅ Memory handler initialized") - } - - // Inject usage limiter into chat service for tool execution - if usageLimiter != nil { - chatService.SetUsageLimiter(usageLimiter) - } - mcpWSHandler := handlers.NewMCPWebSocketHandler(mcpBridge) - configHandler := handlers.NewConfigHandler() - // Initialize agent handler (requires agentService) - var agentHandler *handlers.AgentHandler - var workflowWSHandler *handlers.WorkflowWebSocketHandler - if agentService != nil { - agentHandler = handlers.NewAgentHandler(agentService, workflowGeneratorService) - // Wire up builder conversation service for sync endpoint - if builderConvService != nil { - agentHandler.SetBuilderConversationService(builderConvService) - } - // Wire up v2 workflow generator service (multi-step with tool selection) - agentHandler.SetWorkflowGeneratorV2Service(workflowGeneratorV2Service) - // Wire up provider service for Ask mode - agentHandler.SetProviderService(providerService) - workflowWSHandler = handlers.NewWorkflowWebSocketHandler(agentService, workflowEngine, executionLimiter) - // Wire up execution service for workflow execution tracking - if executionService != nil { - workflowWSHandler.SetExecutionService(executionService) - } - log.Println("✅ Agent handler initialized") - } - toolsHandler := handlers.NewToolsHandler(tools.GetRegistry(), toolService) - imageProxyHandler := handlers.NewImageProxyHandler() - audioHandler := handlers.NewAudioHandler() - log.Println("✅ Audio handler initialized") - - // Initialize schedule handler (requires scheduler service) - var scheduleHandler *handlers.ScheduleHandler - if schedulerService != nil { - scheduleHandler = handlers.NewScheduleHandler(schedulerService, agentService) - log.Println("✅ Schedule handler initialized") - } - - // Initialize execution handler (requires execution service) - var executionHandler *handlers.ExecutionHandler - if executionService != nil { - executionHandler = handlers.NewExecutionHandler(executionService) - log.Println("✅ Execution handler initialized") - } - - // Initialize API key handler (requires API key service) - var apiKeyHandler *handlers.APIKeyHandler - if apiKeyService != nil { - apiKeyHandler = handlers.NewAPIKeyHandler(apiKeyService) - log.Println("✅ API key handler initialized") - } - - // Initialize trigger handler (requires agent service + execution service + workflow engine) - var triggerHandler *handlers.TriggerHandler - if executionService != nil { - triggerHandler = handlers.NewTriggerHandler(agentService, executionService, workflowEngine) - log.Println("✅ Trigger handler initialized") - } - - // Initialize credential handler (requires credential service) - var credentialHandler *handlers.CredentialHandler - var composioAuthHandler *handlers.ComposioAuthHandler - if credentialService != nil { - credentialHandler = handlers.NewCredentialHandler(credentialService) - log.Println("✅ Credential handler initialized") - - // Initialize Composio OAuth handler - composioAuthHandler = handlers.NewComposioAuthHandler(credentialService) - log.Println("✅ Composio OAuth handler initialized") - } - - // Initialize chat sync handler (requires chat sync service) - var chatSyncHandler *handlers.ChatSyncHandler - if chatSyncService != nil { - chatSyncHandler = handlers.NewChatSyncHandler(chatSyncService) - log.Println("✅ Chat sync handler initialized") - } - - // Initialize user preferences handler (requires userService) - var userPreferencesHandler *handlers.UserPreferencesHandler - if userService != nil { - userPreferencesHandler = handlers.NewUserPreferencesHandler(userService) - log.Println("✅ User preferences handler initialized") - } - - // Payment service removed in v2.0 - all users default to Pro tier - - // Wire up GDPR services for complete account deletion - userHandler.SetGDPRServices( - agentService, - executionService, - apiKeyService, - credentialService, - chatSyncService, - schedulerService, - builderConvService, - ) - log.Println("✅ GDPR services wired up for account deletion") - - // Routes - - // Health check (public) - app.Get("/health", healthHandler.Handle) - - // Rate limiter for upload endpoint (10 uploads per minute per user) - uploadLimiter := limiter.New(limiter.Config{ - Max: 10, - Expiration: 1 * time.Minute, - KeyGenerator: func(c *fiber.Ctx) string { - // Rate limit by user ID - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - // Fallback to IP if no user ID - return c.IP() - } - return "upload:" + userID - }, - LimitReached: func(c *fiber.Ctx) error { - log.Printf("⚠️ [RATE-LIMIT] Upload limit reached for user: %v", c.Locals("user_id")) - return c.Status(fiber.StatusTooManyRequests).JSON(fiber.Map{ - "error": "Too many upload requests. Please wait before uploading again.", - }) - }, - }) - - // Create endpoint-specific rate limiters - publicReadLimiter := middleware.PublicReadRateLimiter(rateLimitConfig) - imageProxyLimiter := middleware.ImageProxyRateLimiter(rateLimitConfig) - transcribeLimiter := middleware.TranscribeRateLimiter(rateLimitConfig) - - // API routes (public read-only) - api := app.Group("/api") - { - // Authentication routes (v2.0 - Local JWT) - if localAuthHandler != nil { - auth := api.Group("/auth") - auth.Get("/status", localAuthHandler.GetStatus) // Public - check if users exist - auth.Post("/register", localAuthHandler.Register) - auth.Post("/login", localAuthHandler.Login) - auth.Post("/refresh", localAuthHandler.RefreshToken) - auth.Post("/logout", middleware.LocalAuthMiddleware(jwtAuth), localAuthHandler.Logout) - auth.Get("/me", middleware.LocalAuthMiddleware(jwtAuth), localAuthHandler.GetCurrentUser) - log.Println("✅ Local auth routes registered (/api/auth/*)") - } - - // Image proxy endpoint (public - used by frontend for image search results) - // Has its own rate limiter to prevent bandwidth abuse - api.Get("/proxy/image", imageProxyLimiter, imageProxyHandler.ProxyImage) - - // Public read-only endpoints with rate limiting - api.Get("/providers", publicReadLimiter, providerHandler.List) - api.Get("/models", publicReadLimiter, middleware.OptionalLocalAuthMiddleware(jwtAuth), modelHandler.List) - api.Get("/models/tool-predictors", publicReadLimiter, modelHandler.ListToolPredictorModels) // Specific routes before parameterized ones - api.Get("/models/provider/:id", publicReadLimiter, modelHandler.ListByProvider) - - // Configuration endpoints (public) - api.Get("/config/recommended-models", publicReadLimiter, configHandler.GetRecommendedModels) - - // Conversation status check (public) - api.Get("/conversations/:id/status", publicReadLimiter, conversationHandler.GetStatus) - - // File upload (requires authentication via JWT or API key + rate limiting) - // API key users need "upload" scope to upload files - if apiKeyService != nil { - api.Post("/upload", - middleware.APIKeyOrJWTMiddleware(apiKeyService, middleware.OptionalLocalAuthMiddleware(jwtAuth)), - middleware.RequireScope("upload"), - uploadLimiter, - uploadHandler.Upload, - ) - } else { - api.Post("/upload", - middleware.OptionalLocalAuthMiddleware(jwtAuth), - uploadLimiter, - uploadHandler.Upload, - ) - } - api.Delete("/upload/:id", middleware.OptionalLocalAuthMiddleware(jwtAuth), uploadHandler.Delete) - - // File status check endpoint (for pre-execution validation) - api.Get("/upload/:id/status", middleware.OptionalLocalAuthMiddleware(jwtAuth), uploadHandler.CheckFileStatus) - - // Audio transcription endpoint (requires authentication + rate limiting for expensive GPU operation) - api.Post("/audio/transcribe", middleware.OptionalLocalAuthMiddleware(jwtAuth), transcribeLimiter, audioHandler.Transcribe) - - // Document download (requires authentication for access control) - api.Get("/download/:id", middleware.OptionalLocalAuthMiddleware(jwtAuth), downloadHandler.Download) - - // Secure file downloads (access code based - no auth required for download) - api.Get("/files/:id", secureDownloadHandler.Download) // Download with access code - api.Get("/files/:id/info", secureDownloadHandler.GetInfo) // Get file info with access code - api.Get("/files", middleware.LocalAuthMiddleware(jwtAuth), secureDownloadHandler.ListUserFiles) // List user's files - api.Delete("/files/:id", middleware.LocalAuthMiddleware(jwtAuth), secureDownloadHandler.Delete) // Delete file (owner only) - - // User preferences endpoints (requires authentication) - api.Get("/user/preferences", middleware.LocalAuthMiddleware(jwtAuth), userHandler.GetPreferences) - api.Put("/user/preferences", middleware.LocalAuthMiddleware(jwtAuth), userHandler.UpdatePreferences) - api.Post("/user/welcome-popup-seen", middleware.LocalAuthMiddleware(jwtAuth), userHandler.MarkWelcomePopupSeen) - - // GDPR Compliance endpoints (requires authentication) - api.Get("/user/data", middleware.LocalAuthMiddleware(jwtAuth), userHandler.ExportData) - api.Delete("/user/account", middleware.LocalAuthMiddleware(jwtAuth), userHandler.DeleteAccount) - - // Privacy policy (public) - api.Get("/privacy-policy", userHandler.GetPrivacyPolicy) - - // Memory management routes (requires authentication + memory services) - if memoryHandler != nil { - memories := api.Group("/memories", middleware.LocalAuthMiddleware(jwtAuth)) - memories.Get("/", memoryHandler.ListMemories) - memories.Get("/stats", memoryHandler.GetMemoryStats) // Must be before /:id to avoid route conflict - memories.Get("/:id", memoryHandler.GetMemory) - memories.Post("/", memoryHandler.CreateMemory) - memories.Put("/:id", memoryHandler.UpdateMemory) - memories.Delete("/:id", memoryHandler.DeleteMemory) - memories.Post("/:id/archive", memoryHandler.ArchiveMemory) - memories.Post("/:id/unarchive", memoryHandler.UnarchiveMemory) - - // Conversation memory extraction (manual trigger) - api.Post("/conversations/:id/extract-memories", middleware.LocalAuthMiddleware(jwtAuth), memoryHandler.TriggerMemoryExtraction) - } - - // Agent builder routes (requires authentication + MongoDB) - if agentHandler != nil { - agents := api.Group("/agents", middleware.LocalAuthMiddleware(jwtAuth)) - agents.Post("/", agentHandler.Create) - agents.Get("/", agentHandler.List) - agents.Get("/recent", agentHandler.ListRecent) // Must be before /:id to avoid route conflict - agents.Post("/ask", agentHandler.Ask) // Ask mode - must be before /:id to avoid route conflict - agents.Get("/:id", agentHandler.Get) - agents.Put("/:id", agentHandler.Update) - agents.Delete("/:id", agentHandler.Delete) - agents.Post("/:id/sync", agentHandler.SyncAgent) // Sync local agent to backend - - // Workflow version routes - MUST be before /:id/workflow to avoid route conflict - agents.Get("/:id/workflow/versions", agentHandler.ListWorkflowVersions) - agents.Get("/:id/workflow/versions/:version", agentHandler.GetWorkflowVersion) - agents.Post("/:id/workflow/restore/:version", agentHandler.RestoreWorkflowVersion) - - // Workflow routes (less specific, must come after /versions routes) - agents.Put("/:id/workflow", agentHandler.SaveWorkflow) - agents.Get("/:id/workflow", agentHandler.GetWorkflow) - agents.Post("/:id/generate-workflow", agentHandler.GenerateWorkflow) - agents.Post("/:id/generate-workflow-v2", agentHandler.GenerateWorkflowV2) // Multi-step with tool selection - agents.Post("/:id/select-tools", agentHandler.SelectTools) // Tool selection only (step 1) - agents.Post("/:id/generate-with-tools", agentHandler.GenerateWithTools) // Generate with pre-selected tools (step 2) - agents.Post("/:id/generate-sample-input", agentHandler.GenerateSampleInput) // Generate sample JSON input for testing - - // Builder conversation routes (under agents) - agents.Get("/:id/conversations", conversationHandler.ListBuilderConversations) - agents.Post("/:id/conversations", conversationHandler.CreateBuilderConversation) - agents.Get("/:id/conversations/current", conversationHandler.GetOrCreateBuilderConversation) - agents.Get("/:id/conversations/:convId", conversationHandler.GetBuilderConversation) - agents.Delete("/:id/conversations/:convId", conversationHandler.DeleteBuilderConversation) - agents.Post("/:id/conversations/:convId/messages", conversationHandler.AddBuilderMessage) - - // Schedule routes (under agents) - requires scheduler service - if scheduleHandler != nil { - agents.Post("/:id/schedule", scheduleHandler.Create) - agents.Get("/:id/schedule", scheduleHandler.Get) - agents.Put("/:id/schedule", scheduleHandler.Update) - agents.Delete("/:id/schedule", scheduleHandler.Delete) - agents.Post("/:id/schedule/run", scheduleHandler.TriggerNow) - } - - // Execution routes (under agents) - if executionHandler != nil { - agents.Get("/:id/executions", executionHandler.ListByAgent) - agents.Get("/:id/executions/stats", executionHandler.GetStats) - } - } - - // Execution routes (top-level, authenticated) - MongoDB only - if executionHandler != nil { - executions := api.Group("/executions", middleware.LocalAuthMiddleware(jwtAuth)) - executions.Get("/", executionHandler.ListAll) - executions.Get("/:id", executionHandler.GetByID) - } - - // Schedule routes (top-level, authenticated) - for usage stats - if scheduleHandler != nil { - schedules := api.Group("/schedules", middleware.LocalAuthMiddleware(jwtAuth)) - schedules.Get("/usage", scheduleHandler.GetUsage) - } - - // Tool routes (requires authentication) - tools := api.Group("/tools", middleware.LocalAuthMiddleware(jwtAuth)) - tools.Get("/", toolsHandler.ListTools) - tools.Get("/available", toolsHandler.GetAvailableTools) // Returns tools filtered by user's credentials - tools.Post("/recommend", toolsHandler.RecommendTools) - if agentHandler != nil { - tools.Get("/registry", agentHandler.GetToolRegistry) // Tool registry for workflow builder - } - - // API Key management routes (requires authentication) - if apiKeyHandler != nil { - keys := api.Group("/keys", middleware.LocalAuthMiddleware(jwtAuth)) - keys.Post("/", apiKeyHandler.Create) - keys.Get("/", apiKeyHandler.List) - keys.Get("/:id", apiKeyHandler.Get) - keys.Post("/:id/revoke", apiKeyHandler.Revoke) - keys.Delete("/:id", apiKeyHandler.Delete) - } - - // Credential management routes (requires authentication) - if credentialHandler != nil { - // Integration registry (public read) - api.Get("/integrations", credentialHandler.GetIntegrations) - api.Get("/integrations/:id", credentialHandler.GetIntegration) - - // Credential CRUD (authenticated) - credentials := api.Group("/credentials", middleware.LocalAuthMiddleware(jwtAuth)) - credentials.Post("/", credentialHandler.Create) - credentials.Get("/", credentialHandler.List) - credentials.Get("/by-integration", credentialHandler.GetCredentialsByIntegration) - credentials.Get("/references", credentialHandler.GetCredentialReferences) - credentials.Get("/:id", credentialHandler.Get) - credentials.Put("/:id", credentialHandler.Update) - credentials.Delete("/:id", credentialHandler.Delete) - credentials.Post("/:id/test", credentialHandler.Test) - - // Composio OAuth routes (authenticated) - if composioAuthHandler != nil { - composio := api.Group("/integrations/composio", middleware.LocalAuthMiddleware(jwtAuth)) - composio.Get("/googlesheets/authorize", composioAuthHandler.InitiateGoogleSheetsAuth) - composio.Get("/gmail/authorize", composioAuthHandler.InitiateGmailAuth) - composio.Get("/connected-account", composioAuthHandler.GetConnectedAccount) - composio.Post("/complete-setup", composioAuthHandler.CompleteComposioSetup) - - // Callback endpoint (unauthenticated - Composio calls this) - api.Get("/integrations/composio/callback", composioAuthHandler.HandleComposioCallback) - } - } - - // Chat sync routes (requires authentication + chat sync service) - if chatSyncHandler != nil { - chats := api.Group("/chats", middleware.LocalAuthMiddleware(jwtAuth)) - chats.Get("/sync", chatSyncHandler.SyncAll) // Get all chats for initial sync (must be before /:id) - chats.Post("/sync", chatSyncHandler.BulkSync) // Bulk upload chats - chats.Get("/", chatSyncHandler.List) // List chats (paginated) - chats.Post("/", chatSyncHandler.CreateOrUpdate) // Create or update a chat - chats.Get("/:id", chatSyncHandler.Get) // Get single chat - chats.Put("/:id", chatSyncHandler.Update) // Partial update - chats.Delete("/:id", chatSyncHandler.Delete) // Delete single chat - chats.Post("/:id/messages", chatSyncHandler.AddMessage) // Add message to chat - chats.Delete("/", chatSyncHandler.DeleteAll) // Delete all chats (GDPR) - log.Println("✅ Chat sync routes registered") - } - - // User preferences routes (requires authentication + userService) - if userPreferencesHandler != nil { - prefs := api.Group("/preferences", middleware.LocalAuthMiddleware(jwtAuth)) - prefs.Get("/", userPreferencesHandler.Get) // Get preferences - prefs.Put("/", userPreferencesHandler.Update) // Update preferences - log.Println("✅ User preferences routes registered") - } - - // Subscription routes removed in v2.0 - no payment processing - - // Admin routes (protected by admin middleware - superadmin only) - if userService != nil && tierService != nil { - adminHandler := handlers.NewAdminHandler(userService, tierService, analyticsService, providerService, modelService) - adminRoutes := api.Group("/admin", middleware.LocalAuthMiddleware(jwtAuth), middleware.AdminMiddleware(cfg)) - - // Admin status - adminRoutes.Get("/me", adminHandler.GetAdminStatus) - - // User management - adminRoutes.Get("/users/:userID", adminHandler.GetUserDetails) - adminRoutes.Post("/users/:userID/overrides", adminHandler.SetLimitOverrides) - adminRoutes.Delete("/users/:userID/overrides", adminHandler.RemoveAllOverrides) - adminRoutes.Get("/users", adminHandler.ListUsers) - - // Analytics - adminRoutes.Get("/analytics/overview", adminHandler.GetOverviewAnalytics) - adminRoutes.Get("/analytics/providers", adminHandler.GetProviderAnalytics) - adminRoutes.Get("/analytics/chats", adminHandler.GetChatAnalytics) - adminRoutes.Get("/analytics/models", adminHandler.GetModelAnalytics) - adminRoutes.Get("/analytics/agents", adminHandler.GetAgentAnalytics) - adminRoutes.Post("/analytics/migrate-timestamps", adminHandler.MigrateChatSessionTimestamps) - - // Provider management (CRUD) - adminRoutes.Get("/providers", adminHandler.GetProviders) - adminRoutes.Post("/providers", adminHandler.CreateProvider) - adminRoutes.Put("/providers/:id", adminHandler.UpdateProvider) - adminRoutes.Delete("/providers/:id", adminHandler.DeleteProvider) - adminRoutes.Put("/providers/:id/toggle", adminHandler.ToggleProvider) - - // Model management (CRUD, testing, benchmarking, aliases) - if modelService != nil && providerService != nil { - modelMgmtService := services.NewModelManagementService(db) - modelMgmtHandler := handlers.NewModelManagementHandler(modelMgmtService, modelService, providerService) - - // Bulk operations (MUST be before parameterized routes to avoid :modelId matching) - adminRoutes.Post("/models/import-aliases", modelMgmtHandler.ImportAliasesFromJSON) - adminRoutes.Put("/models/bulk/agents-enabled", modelMgmtHandler.BulkUpdateAgentsEnabled) - adminRoutes.Put("/models/bulk/visibility", modelMgmtHandler.BulkUpdateVisibility) - - // Model CRUD (list and create don't conflict with specific paths) - adminRoutes.Get("/models", modelMgmtHandler.GetAllModels) - adminRoutes.Post("/models", modelMgmtHandler.CreateModel) - - // Global tier management (specific paths before :modelId) - adminRoutes.Get("/tiers", modelMgmtHandler.GetTiers) - - // Model fetching from provider API - adminRoutes.Post("/providers/:providerId/fetch", modelMgmtHandler.FetchModelsFromProvider) - adminRoutes.Post("/providers/:providerId/sync", modelMgmtHandler.SyncProviderToJSON) - - // Parameterized model routes (MUST come after all specific /models/* paths) - adminRoutes.Put("/models/:modelId", modelMgmtHandler.UpdateModel) - adminRoutes.Delete("/models/:modelId", modelMgmtHandler.DeleteModel) - adminRoutes.Post("/models/:modelId/tier", modelMgmtHandler.SetModelTier) - adminRoutes.Delete("/models/:modelId/tier", modelMgmtHandler.ClearModelTier) - - // Model testing - adminRoutes.Post("/models/:modelId/test/connection", modelMgmtHandler.TestModelConnection) - adminRoutes.Post("/models/:modelId/test/capability", modelMgmtHandler.TestModelCapability) - adminRoutes.Post("/models/:modelId/benchmark", modelMgmtHandler.RunModelBenchmark) - adminRoutes.Get("/models/:modelId/test-results", modelMgmtHandler.GetModelTestResults) - - // Alias management (parameterized routes must come after specific paths) - adminRoutes.Get("/models/:modelId/aliases", modelMgmtHandler.GetModelAliases) - adminRoutes.Post("/models/:modelId/aliases", modelMgmtHandler.CreateModelAlias) - adminRoutes.Put("/models/:modelId/aliases/:alias", modelMgmtHandler.UpdateModelAlias) - adminRoutes.Delete("/models/:modelId/aliases/:alias", modelMgmtHandler.DeleteModelAlias) - - log.Println("✅ Model management routes registered (CRUD, testing, tiers, aliases)") - } - - // Legacy stats endpoint - adminRoutes.Get("/stats", adminHandler.GetSystemStats) - - log.Println("✅ Admin routes registered (status, analytics, user management, providers)") - } - - // Webhook endpoint removed in v2.0 - no payment processing - - } - - // Trigger endpoints (API key authenticated, CORS open for external access) - // These are meant to be called from anywhere (webhooks, external services, etc.) - if triggerHandler != nil && apiKeyService != nil { - trigger := app.Group("/api/trigger") - - // Apply permissive CORS for trigger endpoints only - trigger.Use(cors.New(cors.Config{ - AllowOrigins: "*", - AllowMethods: "GET,POST,OPTIONS", - AllowHeaders: "Origin,Content-Type,Accept,Authorization,X-API-Key", - AllowCredentials: false, - })) - - // Apply API key authentication - trigger.Use(middleware.APIKeyMiddleware(apiKeyService)) - - trigger.Post("/:agentId", triggerHandler.TriggerAgent) - trigger.Get("/status/:executionId", triggerHandler.GetExecutionStatus) - - log.Println("✅ Trigger endpoints registered with open CORS (external access enabled)") - } - - // External upload endpoint (API key authenticated, CORS open for external access) - // Allows external services to upload files before triggering agents - if apiKeyService != nil { - externalUpload := app.Group("/api/external") - - // Apply permissive CORS for external upload - externalUpload.Use(cors.New(cors.Config{ - AllowOrigins: "*", - AllowMethods: "POST,OPTIONS", - AllowHeaders: "Origin,Content-Type,Accept,Authorization,X-API-Key", - AllowCredentials: false, - })) - - // Apply API key authentication with upload scope requirement - externalUpload.Use(middleware.APIKeyMiddleware(apiKeyService)) - externalUpload.Use(middleware.RequireScope("upload")) - - externalUpload.Post("/upload", uploadLimiter, uploadHandler.Upload) - - log.Println("✅ External upload endpoint registered with open CORS (/api/external/upload)") - } - - // Serve uploaded files (authenticated - replaced static serving for security) - app.Get("/uploads/:filename", middleware.OptionalLocalAuthMiddleware(jwtAuth), func(c *fiber.Ctx) error { - filename := c.Params("filename") - - // Get user ID from auth middleware - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" || userID == "anonymous" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required to access files", - }) - } - - // Extract file ID from filename (UUID before extension) - fileID := strings.TrimSuffix(filename, filepath.Ext(filename)) - - // Get file from cache and verify ownership - fileCache := filecache.GetService() - file, found := fileCache.Get(fileID) - - if !found { - log.Printf("⚠️ [FILE-ACCESS] File not found or expired: %s (user: %s)", fileID, userID) - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "File not found or has expired", - }) - } - - // Verify ownership - if file.UserID != userID { - log.Printf("🚫 [SECURITY] User %s denied access to file %s (owned by %s)", userID, fileID, file.UserID) - return c.Status(fiber.StatusForbidden).JSON(fiber.Map{ - "error": "Access denied to this file", - }) - } - - // Verify file exists on disk - if file.FilePath == "" || !strings.HasSuffix(file.FilePath, filename) { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "File not found", - }) - } - - log.Printf("✅ [FILE-ACCESS] User %s accessing file %s", userID, filename) - - // Serve file - return c.SendFile(file.FilePath) - }) - - // WebSocket route (requires auth) - app.Use("/ws", func(c *fiber.Ctx) error { - if websocket.IsWebSocketUpgrade(c) { - c.Locals("allowed", true) - return c.Next() - } - return fiber.ErrUpgradeRequired - }) - - // Rate limiter for WebSocket connections (configurable via RATE_LIMIT_WEBSOCKET env var) - wsConnectionLimiter := middleware.WebSocketRateLimiter(rateLimitConfig) - - app.Use("/ws/chat", wsConnectionLimiter) - app.Use("/ws/chat", middleware.OptionalLocalAuthMiddleware(jwtAuth)) - app.Get("/ws/chat", websocket.New(wsHandler.Handle)) - - // MCP WebSocket endpoint (requires authentication) - app.Use("/mcp/connect", func(c *fiber.Ctx) error { - if websocket.IsWebSocketUpgrade(c) { - c.Locals("allowed", true) - return c.Next() - } - return fiber.ErrUpgradeRequired - }) - - // Rate limiter for MCP WebSocket connections (uses same config as WebSocket) - mcpConnectionLimiter := middleware.WebSocketRateLimiter(rateLimitConfig) - - app.Use("/mcp/connect", mcpConnectionLimiter) - app.Use("/mcp/connect", middleware.OptionalLocalAuthMiddleware(jwtAuth)) - app.Get("/mcp/connect", websocket.New(mcpWSHandler.HandleConnection)) - - // Workflow execution WebSocket endpoint (requires authentication + MongoDB) - if workflowWSHandler != nil { - app.Use("/ws/workflow", func(c *fiber.Ctx) error { - if websocket.IsWebSocketUpgrade(c) { - c.Locals("allowed", true) - return c.Next() - } - return fiber.ErrUpgradeRequired - }) - - app.Use("/ws/workflow", wsConnectionLimiter) - app.Use("/ws/workflow", middleware.LocalAuthMiddleware(jwtAuth)) - app.Get("/ws/workflow", websocket.New(workflowWSHandler.Handle)) - } - - // Initialize background jobs - var jobScheduler *jobs.JobScheduler - if mongoDB != nil && tierService != nil && userService != nil { - jobScheduler = jobs.NewJobScheduler() - - // Register retention cleanup job (runs daily at 2 AM UTC) - retentionJob := jobs.NewRetentionCleanupJob(mongoDB, tierService) - jobScheduler.Register("retention_cleanup", retentionJob) - - // Register grace period checker (runs hourly) - gracePeriodJob := jobs.NewGracePeriodChecker(mongoDB, userService, tierService, 7) // 7 day grace period - jobScheduler.Register("grace_period_check", gracePeriodJob) - - // Register promo expiration checker (runs hourly) - promoExpirationJob := jobs.NewPromoExpirationChecker(mongoDB, userService, tierService) - jobScheduler.Register("promo_expiration_check", promoExpirationJob) - - // Start job scheduler - if err := jobScheduler.Start(); err != nil { - log.Printf("⚠️ Failed to start job scheduler: %v", err) - } else { - log.Println("✅ Background job scheduler started") - } - } else { - log.Println("⚠️ Background jobs disabled (requires MongoDB, TierService, UserService)") - } - - // Start server - log.Printf("✅ Server ready on port %s", cfg.Port) - log.Printf("🔗 WebSocket endpoint: ws://localhost:%s/ws/chat", cfg.Port) - log.Printf("🔌 MCP endpoint: ws://localhost:%s/mcp/connect", cfg.Port) - log.Printf("⚡ Workflow endpoint: ws://localhost:%s/ws/workflow", cfg.Port) - log.Printf("📡 Health check: http://localhost:%s/health", cfg.Port) - if schedulerService != nil { - log.Printf("⏰ Scheduler enabled with Redis") - } - if jobScheduler != nil { - log.Printf("🕐 Background jobs: retention cleanup (daily 2 AM), grace period check (hourly)") - } - - // Handle graceful shutdown - go func() { - sigChan := make(chan os.Signal, 1) - signal.Notify(sigChan, syscall.SIGINT, syscall.SIGTERM) - <-sigChan - - log.Println("\n🛑 Shutting down server...") - - // Stop background jobs - if jobScheduler != nil { - jobScheduler.Stop() - } - - // Stop scheduler first - if schedulerService != nil { - if err := schedulerService.Stop(); err != nil { - log.Printf("⚠️ Error stopping scheduler: %v", err) - } - } - - // Stop PubSub service - if pubsubService != nil { - if err := pubsubService.Stop(); err != nil { - log.Printf("⚠️ Error stopping PubSub: %v", err) - } - } - - // Shutdown Fiber - if err := app.Shutdown(); err != nil { - log.Printf("⚠️ Error shutting down server: %v", err) - } - }() - - if err := app.Listen(":" + cfg.Port); err != nil { - log.Fatalf("❌ Failed to start server: %v", err) - } -} - -// syncProviders syncs providers from JSON file to database -func syncProviders(filePath string, providerService *services.ProviderService, modelService *services.ModelService, chatService *services.ChatService) error { - log.Println("🔄 Syncing providers from providers.json...") - - providersConfig, err := config.LoadProviders(filePath) - if err != nil { - return fmt.Errorf("failed to load providers config: %w", err) - } - - log.Printf("📋 Syncing %d providers from providers.json...", len(providersConfig.Providers)) - - // Build a set of provider names from config - configProviderNames := make(map[string]bool) - for _, providerConfig := range providersConfig.Providers { - configProviderNames[providerConfig.Name] = true - } - - // Clean up stale providers that are no longer in providers.json - existingProviders, err := providerService.GetAllIncludingDisabled() - if err != nil { - log.Printf("⚠️ Could not check for stale providers: %v", err) - } else { - for _, existingProvider := range existingProviders { - if !configProviderNames[existingProvider.Name] { - log.Printf(" 🗑️ Removing stale provider: %s (ID %d) - no longer in providers.json", existingProvider.Name, existingProvider.ID) - if err := providerService.Delete(existingProvider.ID); err != nil { - log.Printf(" ⚠️ Failed to delete stale provider %s: %v", existingProvider.Name, err) - } else { - log.Printf(" ✅ Deleted stale provider: %s and its models", existingProvider.Name) - } - } - } - } - - for _, providerConfig := range providersConfig.Providers { - // Check if provider exists - existingProvider, err := providerService.GetByName(providerConfig.Name) - if err != nil { - return fmt.Errorf("failed to check provider: %w", err) - } - - var provider *models.Provider - if existingProvider == nil { - // Create new provider - log.Printf(" ➕ Creating provider: %s", providerConfig.Name) - provider, err = providerService.Create(providerConfig) - if err != nil { - return fmt.Errorf("failed to create provider: %w", err) - } - } else { - // Update existing provider - log.Printf(" ♻️ Updating provider: %s (ID %d)", providerConfig.Name, existingProvider.ID) - if err := providerService.Update(existingProvider.ID, providerConfig); err != nil { - return fmt.Errorf("failed to update provider: %w", err) - } - provider = existingProvider - } - - // Get config service instance - configService := services.GetConfigService() - - // Load model aliases into both chat service and config service - if len(providerConfig.ModelAliases) > 0 { - log.Printf(" 🔄 Loading %d model aliases for %s...", len(providerConfig.ModelAliases), providerConfig.Name) - chatService.SetModelAliases(provider.ID, providerConfig.ModelAliases) - configService.SetModelAliases(provider.ID, providerConfig.ModelAliases) - - // Save aliases to database - if err := modelService.SaveAliasesToDB(provider.ID, providerConfig.ModelAliases); err != nil { - log.Printf(" ⚠️ Failed to save aliases to database for %s: %v", providerConfig.Name, err) - } - } - - // Store recommended models - if providerConfig.RecommendedModels != nil { - configService.SetRecommendedModels(provider.ID, providerConfig.RecommendedModels) - - // Save recommended models to database - if err := modelService.SaveRecommendedModelsToDB(provider.ID, providerConfig.RecommendedModels); err != nil { - log.Printf(" ⚠️ Failed to save recommended models to database for %s: %v", providerConfig.Name, err) - } - } - - // Store provider security flag - configService.SetProviderSecure(provider.ID, providerConfig.Secure) - - // Sync filters - if len(providerConfig.Filters) > 0 { - log.Printf(" 🔧 Syncing %d filters for %s...", len(providerConfig.Filters), providerConfig.Name) - if err := providerService.SyncFilters(provider.ID, providerConfig.Filters); err != nil { - return fmt.Errorf("failed to sync filters: %w", err) - } - } - - // Fetch models if provider is enabled - if providerConfig.Enabled { - if err := modelService.FetchFromProvider(provider); err != nil { - log.Printf(" ⚠️ Failed to fetch models for %s: %v", provider.Name, err) - } else { - // Sync model alias metadata to database (smart_tool_router, agents, etc.) - if len(providerConfig.ModelAliases) > 0 { - if err := modelService.SyncModelAliasMetadata(provider.ID, providerConfig.ModelAliases); err != nil { - log.Printf(" ⚠️ Failed to sync model alias metadata for %s: %v", provider.Name, err) - } - } - - // Apply filters - if err := providerService.ApplyFilters(provider.ID); err != nil { - log.Printf(" ⚠️ Failed to apply filters for %s: %v", provider.Name, err) - } - } - } - } - - // After syncing providers to database, load image providers from database - log.Println("🎨 Loading image providers from database...") - allProviders, err := providerService.GetAll() - if err != nil { - log.Printf("⚠️ Failed to load providers from database: %v", err) - } else { - // Convert database Provider models to ProviderConfig - var providerConfigs []models.ProviderConfig - for _, p := range allProviders { - providerConfigs = append(providerConfigs, models.ProviderConfig{ - Name: p.Name, - BaseURL: p.BaseURL, - APIKey: p.APIKey, - Enabled: p.Enabled, - Secure: p.Secure, - AudioOnly: p.AudioOnly, - ImageOnly: p.ImageOnly, - ImageEditOnly: p.ImageEditOnly, - DefaultModel: p.DefaultModel, - SystemPrompt: p.SystemPrompt, - Favicon: p.Favicon, - }) - } - - // Load image providers into the image provider service - imageProviderService := services.GetImageProviderService() - imageProviderService.LoadFromProviders(providerConfigs) - - // Load image edit providers into the image edit provider service - imageEditProviderService := services.GetImageEditProviderService() - imageEditProviderService.LoadFromProviders(providerConfigs) - } - - log.Println("✅ Provider sync completed") - return nil -} - -// loadConfigFromDatabase loads model aliases and recommended models from database -// Returns true if data was successfully loaded, false if database is empty (first run) -func loadConfigFromDatabase(modelService *services.ModelService, chatService *services.ChatService, providerService *services.ProviderService) (bool, error) { - log.Println("🔄 Loading configuration from database...") - - // Load aliases from database - aliases, err := modelService.LoadAllAliasesFromDB() - if err != nil { - return false, fmt.Errorf("failed to load aliases from database: %w", err) - } - - // Load recommended models from database - recommendedModels, err := modelService.LoadAllRecommendedModelsFromDB() - if err != nil { - return false, fmt.Errorf("failed to load recommended models from database: %w", err) - } - - // If database is empty, that's fine - admin will configure via UI - if len(aliases) == 0 && len(recommendedModels) == 0 { - log.Println("📋 Database is empty - use admin UI to configure providers and models") - return false, nil - } - - // Load into ConfigService - configService := services.GetConfigService() - - // Load all providers to get security flags - _, err = providerService.GetAllIncludingDisabled() - if err != nil { - return false, fmt.Errorf("failed to load providers: %w", err) - } - - // Load aliases into both chat service and config service - for providerID, providerAliases := range aliases { - if len(providerAliases) > 0 { - chatService.SetModelAliases(providerID, providerAliases) - configService.SetModelAliases(providerID, providerAliases) - log.Printf(" ✅ Loaded %d aliases for provider %d", len(providerAliases), providerID) - } - } - - // Load recommended models into config service - for providerID, recommended := range recommendedModels { - configService.SetRecommendedModels(providerID, recommended) - log.Printf(" ✅ Loaded recommended models for provider %d", providerID) - } - - // Load image providers from database - log.Println("🎨 Loading image providers from database...") - allProviders, err := providerService.GetAll() - if err != nil { - log.Printf("⚠️ Failed to load providers from database: %v", err) - } else { - // Convert database Provider models to ProviderConfig - var providerConfigs []models.ProviderConfig - for _, p := range allProviders { - providerConfigs = append(providerConfigs, models.ProviderConfig{ - Name: p.Name, - BaseURL: p.BaseURL, - APIKey: p.APIKey, - Enabled: p.Enabled, - Secure: p.Secure, - AudioOnly: p.AudioOnly, - ImageOnly: p.ImageOnly, - ImageEditOnly: p.ImageEditOnly, - DefaultModel: p.DefaultModel, - SystemPrompt: p.SystemPrompt, - Favicon: p.Favicon, - }) - } - - // Load image providers into the image provider service - imageProviderService := services.GetImageProviderService() - imageProviderService.LoadFromProviders(providerConfigs) - - // Load image edit providers into the image edit provider service - imageEditProviderService := services.GetImageEditProviderService() - imageEditProviderService.LoadFromProviders(providerConfigs) - } - - log.Printf("✅ Loaded configuration from database: %d provider aliases, %d recommended model sets", - len(aliases), len(recommendedModels)) - return true, nil -} - -// startModelRefreshJob starts a background job to refresh models every 24 hours -func startModelRefreshJob(providerService *services.ProviderService, modelService *services.ModelService, chatService *services.ChatService) { - ticker := time.NewTicker(24 * time.Hour) - defer ticker.Stop() - - log.Println("⏰ Model refresh job started (every 24 hours)") - - for range ticker.C { - log.Println("🔄 Running scheduled model refresh...") - - // Reload aliases from database to ensure they stay in sync - log.Println("🔄 Reloading model aliases from database...") - if err := reloadModelAliases(providerService, modelService, chatService); err != nil { - log.Printf("⚠️ Failed to reload model aliases: %v", err) - } else { - log.Println("✅ Model aliases reloaded successfully") - } - - providers, err := providerService.GetAll() - if err != nil { - log.Printf("❌ Failed to get providers for refresh: %v", err) - continue - } - - for _, provider := range providers { - if !provider.Enabled { - continue - } - - if err := modelService.FetchFromProvider(&provider); err != nil { - log.Printf("❌ Error refreshing models for %s: %v", provider.Name, err) - } else { - // Apply filters after refresh - if err := providerService.ApplyFilters(provider.ID); err != nil { - log.Printf("⚠️ Failed to apply filters for %s: %v", provider.Name, err) - } - } - } - - log.Println("✅ Scheduled model refresh completed") - } -} - -// reloadModelAliases reloads model aliases from database into memory -// This is called by the background refresh job to keep in-memory cache fresh -func reloadModelAliases(providerService *services.ProviderService, modelService *services.ModelService, chatService *services.ChatService) error { - log.Println("🔄 [ALIAS-RELOAD] Loading model aliases from database...") - - // Load aliases from database - aliases, err := modelService.LoadAllAliasesFromDB() - if err != nil { - return fmt.Errorf("failed to load aliases from database: %w", err) - } - - // Load recommended models from database - recommendedModels, err := modelService.LoadAllRecommendedModelsFromDB() - if err != nil { - return fmt.Errorf("failed to load recommended models from database: %w", err) - } - - configService := services.GetConfigService() - - // Load aliases into both chat service and config service - for providerID, providerAliases := range aliases { - if len(providerAliases) > 0 { - chatService.SetModelAliases(providerID, providerAliases) - configService.SetModelAliases(providerID, providerAliases) - log.Printf(" ✅ Reloaded %d aliases for provider %d", len(providerAliases), providerID) - } - } - - // Load recommended models into config service - for providerID, recommended := range recommendedModels { - configService.SetRecommendedModels(providerID, recommended) - } - - log.Printf("✅ [ALIAS-RELOAD] Configuration reloaded from database") - return nil -} - -// startImageCleanupJob starts a background job to clean up expired images every 10 minutes -func startImageCleanupJob(uploadDir string) { - ticker := time.NewTicker(10 * time.Minute) - defer ticker.Stop() - - log.Println("⏰ File cleanup job started (every 10 minutes) - handles images, CSV, Excel, JSON, etc.") - - for range ticker.C { - log.Println("🧹 Running scheduled file cleanup...") - - // Get file cache service - fileCache := filecache.GetService() - - // Cleanup expired files tracked in cache (images, CSV, Excel, JSON, etc.) - fileCache.CleanupExpiredFiles() - - // Cleanup orphaned files on disk (not in cache - e.g., from server restarts) - // Max age of 1 hour matches our retention policy - fileCache.CleanupOrphanedFiles(uploadDir, 1*time.Hour) - - log.Println("✅ Scheduled file cleanup completed") - } -} - -// startDocumentCleanupJob starts a background job to clean up downloaded documents every 5 minutes -func startDocumentCleanupJob() { - ticker := time.NewTicker(5 * time.Minute) - defer ticker.Stop() - - log.Println("⏰ Document cleanup job started (every 5 minutes)") - - for range ticker.C { - log.Println("🧹 Running scheduled document cleanup...") - - // Get document service - documentService := document.GetService() - - // Cleanup downloaded documents - documentService.CleanupDownloadedDocuments() - - log.Println("✅ Scheduled document cleanup completed") - } -} - -// startMemoryExtractionWorker processes pending memory extraction jobs -func startMemoryExtractionWorker(memoryExtractionService *services.MemoryExtractionService) { - ticker := time.NewTicker(30 * time.Second) - defer ticker.Stop() - - log.Println("⏰ Memory extraction worker started (every 30 seconds)") - - for range ticker.C { - ctx, cancel := context.WithTimeout(context.Background(), 25*time.Second) - if err := memoryExtractionService.ProcessPendingJobs(ctx); err != nil { - log.Printf("⚠️ [MEMORY-WORKER] Failed to process jobs: %v", err) - } - cancel() - } -} - -func startMemoryDecayWorker(memoryDecayService *services.MemoryDecayService) { - // Run immediately on startup - log.Println("🔄 [MEMORY-DECAY] Running initial decay job") - ctx, cancel := context.WithTimeout(context.Background(), 5*time.Minute) - if err := memoryDecayService.RunDecayJob(ctx); err != nil { - log.Printf("⚠️ [MEMORY-DECAY] Initial decay job failed: %v", err) - } - cancel() - - // Then run every 6 hours - ticker := time.NewTicker(6 * time.Hour) - defer ticker.Stop() - - log.Println("⏰ Memory decay worker started (every 6 hours)") - - for range ticker.C { - ctx, cancel := context.WithTimeout(context.Background(), 5*time.Minute) - if err := memoryDecayService.RunDecayJob(ctx); err != nil { - log.Printf("⚠️ [MEMORY-DECAY] Decay job failed: %v", err) - } - cancel() - } -} - -// startProvidersFileWatcher watches providers.json for changes and auto-syncs -func startProvidersFileWatcher( - filePath string, - providerService *services.ProviderService, - modelService *services.ModelService, - chatService *services.ChatService, -) { - watcher, err := fsnotify.NewWatcher() - if err != nil { - log.Printf("⚠️ Failed to create file watcher: %v", err) - return - } - - // Get absolute path for the file - absPath, err := filepath.Abs(filePath) - if err != nil { - log.Printf("⚠️ Failed to get absolute path for %s: %v", filePath, err) - watcher.Close() - return - } - - // Watch the directory containing the file (more reliable than watching the file directly) - dir := filepath.Dir(absPath) - filename := filepath.Base(absPath) - - if err := watcher.Add(dir); err != nil { - log.Printf("⚠️ Failed to watch directory %s: %v", dir, err) - watcher.Close() - return - } - - log.Printf("👁️ Watching %s for changes (hot-reload enabled)", filePath) - - // Debounce timer to avoid multiple syncs for rapid file changes - var debounceTimer *time.Timer - debounceDuration := 500 * time.Millisecond - - for { - select { - case event, ok := <-watcher.Events: - if !ok { - return - } - - // Only react to changes to our specific file - if filepath.Base(event.Name) != filename { - continue - } - - // React to write and create events - if event.Op&fsnotify.Write == fsnotify.Write || event.Op&fsnotify.Create == fsnotify.Create { - // Debounce: cancel previous timer and set a new one - if debounceTimer != nil { - debounceTimer.Stop() - } - - debounceTimer = time.AfterFunc(debounceDuration, func() { - log.Printf("🔄 Detected changes in %s, re-syncing providers...", filePath) - - if err := syncProviders(filePath, providerService, modelService, chatService); err != nil { - log.Printf("❌ Failed to sync providers after file change: %v", err) - } else { - log.Printf("✅ Providers synced successfully from %s", filePath) - } - }) - } - - case err, ok := <-watcher.Errors: - if !ok { - return - } - log.Printf("⚠️ File watcher error: %v", err) - } - } -} diff --git a/backend/docker-entrypoint.sh b/backend/docker-entrypoint.sh deleted file mode 100755 index 8147260f..00000000 --- a/backend/docker-entrypoint.sh +++ /dev/null @@ -1,67 +0,0 @@ -#!/bin/sh -set -e - -# ClaraVerse Backend Docker Entrypoint Script -# This script handles initialization before starting the application - -echo "=== ClaraVerse Backend Starting ===" - -# Check if .env file exists -if [ ! -f "/app/.env" ]; then - echo "WARNING: .env file not found. Using example configuration." - echo "Please create .env file from .env.example before running in production." - - # Create .env from example if available - if [ -f "/app/.env.example" ]; then - cp /app/.env.example /app/.env - echo "Created .env from .env.example" - fi -fi - -# Check if providers.json exists at configured location -PROVIDERS_PATH="${PROVIDERS_FILE:-/app/providers.json}" -if [ ! -f "$PROVIDERS_PATH" ]; then - echo "WARNING: providers.json not found at $PROVIDERS_PATH. Using example configuration." - - # Create providers.json from example if available - if [ -f "/app/providers.example.json" ]; then - cp /app/providers.example.json "$PROVIDERS_PATH" - echo "Created providers.json from providers.example.json" - else - echo "ERROR: No providers configuration found!" - echo "Please mount providers.json or create one from providers.example.json" - exit 1 - fi -fi - -# Ensure data directories exist with proper permissions -mkdir -p /app/data /app/uploads /app/logs - -# Display configuration summary -echo "" -echo "Configuration:" -echo " - Database: ${DATABASE_PATH:-model_capabilities.db}" -echo " - Providers: ${PROVIDERS_FILE:-providers.json}" -echo " - Upload Dir: ${UPLOAD_DIR:-./uploads}" -echo " - Environment: ${ENVIRONMENT:-development}" -echo " - Port: ${PORT:-3001}" -echo "" - -# Check for required environment variables in production -if [ "$ENVIRONMENT" = "production" ]; then - echo "Running in PRODUCTION mode" - - if [ -z "$SUPABASE_URL" ] || [ -z "$SUPABASE_KEY" ]; then - echo "WARNING: Supabase configuration is missing!" - echo "SUPABASE_URL and SUPABASE_KEY are required in production mode." - echo "The server will terminate if authentication is not properly configured." - fi -fi - -echo "" -echo "Starting ClaraVerse backend..." -echo "===================================" -echo "" - -# Execute the main application -exec "$@" diff --git a/backend/e2b-service/.dockerignore b/backend/e2b-service/.dockerignore deleted file mode 100644 index 582f6978..00000000 --- a/backend/e2b-service/.dockerignore +++ /dev/null @@ -1,18 +0,0 @@ -__pycache__ -*.pyc -*.pyo -*.pyd -.Python -*.so -*.egg -*.egg-info -dist -build -.pytest_cache -.coverage -htmlcov -.tox -.env -.venv -venv -ENV diff --git a/backend/e2b-service/Dockerfile b/backend/e2b-service/Dockerfile deleted file mode 100644 index ffb80752..00000000 --- a/backend/e2b-service/Dockerfile +++ /dev/null @@ -1,20 +0,0 @@ -FROM python:3.11-slim - -WORKDIR /app - -# Install dependencies -COPY requirements.txt . -RUN pip install --no-cache-dir -r requirements.txt - -# Copy application -COPY main.py . - -# Expose port -EXPOSE 8001 - -# Health check -HEALTHCHECK --interval=30s --timeout=10s --start-period=5s --retries=3 \ - CMD python -c "import requests; requests.get('http://localhost:8001/health')" - -# Run the application -CMD ["uvicorn", "main:app", "--host", "0.0.0.0", "--port", "8001", "--log-level", "info"] diff --git a/backend/e2b-service/main.py b/backend/e2b-service/main.py deleted file mode 100644 index 61304f28..00000000 --- a/backend/e2b-service/main.py +++ /dev/null @@ -1,475 +0,0 @@ -#!/usr/bin/env python3 -""" -E2B Code Executor Microservice for ClaraVerse -Now using E2B in LOCAL mode - no cloud API required! -Provides REST API for executing Python code in E2B sandboxes -""" - -import os -import base64 -from typing import List, Optional, Dict, Any -from fastapi import FastAPI, HTTPException, UploadFile, File, Form -from fastapi.middleware.cors import CORSMiddleware -from pydantic import BaseModel -from e2b_code_interpreter import Sandbox -import logging - -# Configure logging -logging.basicConfig(level=logging.INFO) -logger = logging.getLogger(__name__) - -# Configure E2B for local execution (v2.0 - no API key needed!) -E2B_MODE = os.getenv("E2B_MODE", "local") # local or production -os.environ['E2B_MODE'] = E2B_MODE - -if E2B_MODE == "local": - logger.info("🐳 E2B running in LOCAL Docker mode - no API key required!") - os.environ['E2B_LOCAL_USE_DOCKER'] = os.getenv("E2B_LOCAL_USE_DOCKER", "true") - os.environ['E2B_SANDBOX_POOL_SIZE'] = os.getenv("E2B_SANDBOX_POOL_SIZE", "3") - os.environ['E2B_EXECUTION_TIMEOUT'] = os.getenv("E2B_EXECUTION_TIMEOUT", "30000") - os.environ['E2B_RATE_LIMIT_PER_MIN'] = os.getenv("E2B_RATE_LIMIT_PER_MIN", "20") - logger.info(f" Pool size: {os.environ['E2B_SANDBOX_POOL_SIZE']} warm sandboxes") - logger.info(f" Timeout: {os.environ['E2B_EXECUTION_TIMEOUT']}ms") -else: - # Production mode - requires E2B API key - E2B_API_KEY = os.getenv("E2B_API_KEY") - if not E2B_API_KEY: - logger.error("E2B_API_KEY environment variable required for production mode") - raise RuntimeError("E2B_API_KEY is required when E2B_MODE=production") - logger.info("☁️ E2B running in CLOUD mode with API key") - -app = FastAPI( - title="E2B Code Executor Service", - description="Microservice for executing Python code in isolated E2B sandboxes", - version="1.0.0" -) - -# CORS middleware -app.add_middleware( - CORSMiddleware, - allow_origins=["*"], # In production, restrict to backend service - allow_credentials=True, - allow_methods=["*"], - allow_headers=["*"], -) - - -# Request/Response Models -class ExecuteRequest(BaseModel): - code: str - timeout: Optional[int] = 30 # seconds - - -class PlotResult(BaseModel): - format: str # "png", "svg", etc. - data: str # base64 encoded - - -class ExecuteResponse(BaseModel): - success: bool - stdout: str - stderr: str - error: Optional[str] = None - plots: List[PlotResult] = [] - execution_time: Optional[float] = None - - -class FileUploadRequest(BaseModel): - code: str - timeout: Optional[int] = 30 - - -# Advanced execution models (with dependencies and output files) -class AdvancedExecuteRequest(BaseModel): - code: str - timeout: Optional[int] = 30 - dependencies: List[str] = [] # pip packages to install - output_files: List[str] = [] # files to retrieve after execution - - -class FileResult(BaseModel): - filename: str - data: str # base64 encoded - size: int - - -class AdvancedExecuteResponse(BaseModel): - success: bool - stdout: str - stderr: str - error: Optional[str] = None - plots: List[PlotResult] = [] - files: List[FileResult] = [] - execution_time: Optional[float] = None - install_output: str = "" - - -# Health check endpoint -@app.get("/health") -async def health_check(): - """Health check endpoint""" - return { - "status": "healthy", - "service": "e2b-executor", - "mode": E2B_MODE, - "e2b_api_key_set": bool(os.getenv("E2B_API_KEY")) if E2B_MODE == "production" else False - } - - -# Execute Python code endpoint -@app.post("/execute", response_model=ExecuteResponse) -async def execute_code(request: ExecuteRequest): - """ - Execute Python code in an E2B sandbox (local Docker mode) - - Returns: - - stdout: Standard output - - stderr: Standard error - - plots: List of generated plots (base64 encoded) - - error: Error message if execution failed - """ - logger.info(f"Executing code in {E2B_MODE.upper()} mode (length: {len(request.code)} chars)") - - try: - # Create sandbox - with Sandbox.create() as sandbox: - # Run code - execution = sandbox.run_code(request.code) - - # Collect stdout - stdout = "" - if execution.logs.stdout: - stdout = "\n".join(execution.logs.stdout) - - # Collect stderr - stderr = "" - if execution.logs.stderr: - stderr = "\n".join(execution.logs.stderr) - - # Check for execution errors - error_msg = None - if execution.error: - error_msg = str(execution.error) - logger.warning(f"Execution error: {error_msg}") - - # Collect plots and text results - plots = [] - result_texts = [] - for i, result in enumerate(execution.results): - if hasattr(result, 'png') and result.png: - plots.append(PlotResult( - format="png", - data=result.png # Already base64 encoded - )) - logger.info(f"Found plot {i}: {len(result.png)} bytes (base64)") - elif hasattr(result, 'text') and result.text: - result_texts.append(result.text) - logger.info(f"Found text result {i}: {result.text[:100]}...") - - # Append result texts to stdout (captures last expression like Jupyter) - if result_texts: - result_output = "\n".join(result_texts) - if stdout: - stdout = stdout + "\n" + result_output - else: - stdout = result_output - - response = ExecuteResponse( - success=error_msg is None, - stdout=stdout, - stderr=stderr, - error=error_msg, - plots=plots - ) - - logger.info(f"Execution completed: success={response.success}, plots={len(plots)}") - return response - - except Exception as e: - logger.error(f"Sandbox execution failed: {str(e)}") - raise HTTPException( - status_code=500, - detail=f"Sandbox execution failed: {str(e)}" - ) - - -# Execute with file upload endpoint -@app.post("/execute-with-files", response_model=ExecuteResponse) -async def execute_with_files( - code: str = Form(...), - files: List[UploadFile] = File(...), - timeout: int = Form(30) -): - """ - Execute Python code with uploaded files - - Files are uploaded to the sandbox and can be accessed by filename in the code - """ - logger.info(f"Executing code with {len(files)} files") - - try: - # Create sandbox - with Sandbox.create() as sandbox: - # Upload files to sandbox - for file in files: - content = await file.read() - sandbox.files.write(file.filename, content) - logger.info(f"Uploaded file: {file.filename} ({len(content)} bytes)") - - # Run code - execution = sandbox.run_code(code) - - # Collect stdout - stdout = "" - if execution.logs.stdout: - stdout = "\n".join(execution.logs.stdout) - - # Collect stderr - stderr = "" - if execution.logs.stderr: - stderr = "\n".join(execution.logs.stderr) - - # Check for errors - error_msg = None - if execution.error: - error_msg = str(execution.error) - logger.warning(f"Execution error: {error_msg}") - - # Collect plots - plots = [] - for i, result in enumerate(execution.results): - if hasattr(result, 'png') and result.png: - plots.append(PlotResult( - format="png", - data=result.png - )) - logger.info(f"Found plot {i}") - - response = ExecuteResponse( - success=error_msg is None, - stdout=stdout, - stderr=stderr, - error=error_msg, - plots=plots - ) - - logger.info(f"Execution with files completed: success={response.success}") - return response - - except Exception as e: - logger.error(f"Sandbox execution failed: {str(e)}") - raise HTTPException( - status_code=500, - detail=f"Sandbox execution failed: {str(e)}" - ) - - -# Execute with dependencies and output file retrieval -@app.post("/execute-advanced", response_model=AdvancedExecuteResponse) -async def execute_advanced(request: AdvancedExecuteRequest): - """ - Execute Python code with pip dependencies and output file retrieval. - - - Install pip packages before running code - - Run user code (max 30 seconds) - - Auto-detect and retrieve ALL generated files (plus any explicitly specified) - """ - import time - - logger.info(f"Advanced execution: code={len(request.code)} chars, deps={request.dependencies}, output_files={request.output_files}") - - try: - with Sandbox.create() as sandbox: - start_time = time.time() - install_output = "" - - # 1. Install dependencies (if any) - if request.dependencies: - deps_str = " ".join(request.dependencies) - logger.info(f"Installing dependencies: {deps_str}") - try: - result = sandbox.commands.run(f"pip install -q {deps_str}", timeout=60) - install_output = (result.stdout or "") + (result.stderr or "") - logger.info(f"Dependencies installed: {install_output[:200]}") - except Exception as e: - logger.error(f"Dependency installation failed: {e}") - return AdvancedExecuteResponse( - success=False, - stdout="", - stderr="", - error=f"Failed to install dependencies: {str(e)}", - plots=[], - files=[], - execution_time=time.time() - start_time, - install_output=str(e) - ) - - # 2. List files BEFORE execution to detect new files later - files_before = set() - try: - result = sandbox.commands.run("find /home/user -maxdepth 2 -type f 2>/dev/null || ls -la /home/user", timeout=10) - if result.stdout: - for line in result.stdout.strip().split('\n'): - line = line.strip() - if line and not line.startswith('total'): - # Handle both find output (full paths) and ls output - if line.startswith('/'): - files_before.add(line) - else: - # ls -la format: permissions links owner group size date name - parts = line.split() - if len(parts) >= 9: - files_before.add(parts[-1]) - logger.info(f"Files before execution: {len(files_before)}") - except Exception as e: - logger.warning(f"Could not list files before execution: {e}") - - # 3. Run user code - execution = sandbox.run_code(request.code) - - # Collect stdout - stdout = "" - if execution.logs.stdout: - stdout = "\n".join(execution.logs.stdout) - - # Collect stderr - stderr = "" - if execution.logs.stderr: - stderr = "\n".join(execution.logs.stderr) - - # Check for errors - error_msg = None - if execution.error: - error_msg = str(execution.error) - logger.warning(f"Execution error: {error_msg}") - - # Collect plots and text results from execution.results - # E2B results contain last expression value (like Jupyter) - plots = [] - result_texts = [] - for i, result in enumerate(execution.results): - if hasattr(result, 'png') and result.png: - plots.append(PlotResult( - format="png", - data=result.png - )) - logger.info(f"Found plot {i}") - elif hasattr(result, 'text') and result.text: - # Capture text output from last expression (like Jupyter Out[]) - result_texts.append(result.text) - logger.info(f"Found text result {i}: {result.text[:100]}...") - - # Append result texts to stdout if no explicit print was used - if result_texts: - result_output = "\n".join(result_texts) - if stdout: - stdout = stdout + "\n" + result_output - else: - stdout = result_output - - # 4. List files AFTER execution to detect new files - files_after = set() - new_files = [] - try: - result = sandbox.commands.run("find /home/user -maxdepth 2 -type f 2>/dev/null || ls -la /home/user", timeout=10) - if result.stdout: - for line in result.stdout.strip().split('\n'): - line = line.strip() - if line and not line.startswith('total'): - if line.startswith('/'): - files_after.add(line) - else: - parts = line.split() - if len(parts) >= 9: - files_after.add(parts[-1]) - # Find new files (created during execution) - new_files = list(files_after - files_before) - # Filter out common unwanted files - excluded_patterns = ['.pyc', '__pycache__', '.ipynb_checkpoints', '.cache'] - new_files = [f for f in new_files if not any(p in f for p in excluded_patterns)] - logger.info(f"Files after execution: {len(files_after)}, new files detected: {new_files}") - except Exception as e: - logger.warning(f"Could not list files after execution: {e}") - - # 5. Collect output files (auto-detected + explicitly requested) - files = [] - collected_filenames = set() - - # First collect explicitly requested files - for filepath in request.output_files: - try: - content = sandbox.files.read(filepath) - # Handle both string and bytes - if isinstance(content, str): - content = content.encode('utf-8') - filename = os.path.basename(filepath) - files.append(FileResult( - filename=filename, - data=base64.b64encode(content).decode('utf-8'), - size=len(content) - )) - collected_filenames.add(filename) - logger.info(f"Retrieved requested file: {filepath} ({len(content)} bytes)") - except Exception as e: - logger.warning(f"Could not retrieve file {filepath}: {e}") - - # Then collect auto-detected new files (if not already collected) - for filepath in new_files: - filename = os.path.basename(filepath) - if filename in collected_filenames: - continue # Already collected - try: - # Try both the full path and just the filename - content = None - for try_path in [filepath, f"/home/user/{filename}", filename]: - try: - content = sandbox.files.read(try_path) - break - except: - continue - - if content is not None: - if isinstance(content, str): - content = content.encode('utf-8') - files.append(FileResult( - filename=filename, - data=base64.b64encode(content).decode('utf-8'), - size=len(content) - )) - collected_filenames.add(filename) - logger.info(f"Retrieved auto-detected file: {filename} ({len(content)} bytes)") - except Exception as e: - logger.warning(f"Could not retrieve auto-detected file {filepath}: {e}") - - execution_time = time.time() - start_time - - response = AdvancedExecuteResponse( - success=error_msg is None, - stdout=stdout, - stderr=stderr, - error=error_msg, - plots=plots, - files=files, - execution_time=execution_time, - install_output=install_output - ) - - logger.info(f"Advanced execution completed: success={response.success}, plots={len(plots)}, files={len(files)}, time={execution_time:.2f}s") - return response - - except Exception as e: - logger.error(f"Advanced sandbox execution failed: {str(e)}") - raise HTTPException( - status_code=500, - detail=f"Sandbox execution failed: {str(e)}" - ) - - -if __name__ == "__main__": - import uvicorn - uvicorn.run( - app, - host="0.0.0.0", - port=8001, - log_level="info" - ) diff --git a/backend/e2b-service/main_simple.py b/backend/e2b-service/main_simple.py deleted file mode 100644 index a302cdb0..00000000 --- a/backend/e2b-service/main_simple.py +++ /dev/null @@ -1,417 +0,0 @@ -#!/usr/bin/env python3 -""" -Simple Python Code Executor for ClaraVerse All-in-One Docker Image - -This is a lightweight alternative to the E2B-based executor that runs Python code -directly using subprocess. It's designed for the all-in-one Docker image where -Docker-in-Docker isn't available for E2B local mode. - -Features: -- Subprocess-based Python execution with timeout -- Basic sandboxing via restricted builtins and resource limits -- Support for matplotlib plots (auto-saved as PNG) -- File upload and retrieval support -- No external dependencies (E2B API key or Docker) - -Note: This is less secure than E2B sandboxes. For production use with untrusted -code, use the regular E2B service with proper sandboxing. -""" - -import os -import sys -import base64 -import tempfile -import subprocess -import shutil -import signal -from typing import List, Optional -from pathlib import Path -from fastapi import FastAPI, HTTPException, UploadFile, File, Form -from fastapi.middleware.cors import CORSMiddleware -from pydantic import BaseModel -import logging -import time -import uuid - -# Configure logging -logging.basicConfig(level=logging.INFO) -logger = logging.getLogger(__name__) - -logger.info("🚀 Starting Simple Python Executor (All-in-One Mode)") -logger.info(" This executor runs Python code directly without E2B sandboxes") - -app = FastAPI( - title="Simple Python Executor Service", - description="Lightweight Python code executor for ClaraVerse All-in-One", - version="1.0.0" -) - -# CORS middleware -app.add_middleware( - CORSMiddleware, - allow_origins=["*"], - allow_credentials=True, - allow_methods=["*"], - allow_headers=["*"], -) - -# Execution settings -EXECUTION_TIMEOUT = int(os.getenv("EXECUTION_TIMEOUT", "30")) # seconds -MAX_OUTPUT_SIZE = 100 * 1024 # 100KB max output -WORK_DIR = Path(os.getenv("WORK_DIR", "/tmp/code-executor")) - -# Ensure work directory exists -WORK_DIR.mkdir(parents=True, exist_ok=True) - - -# Request/Response Models -class ExecuteRequest(BaseModel): - code: str - timeout: Optional[int] = 30 - - -class PlotResult(BaseModel): - format: str - data: str # base64 encoded - - -class ExecuteResponse(BaseModel): - success: bool - stdout: str - stderr: str - error: Optional[str] = None - plots: List[PlotResult] = [] - execution_time: Optional[float] = None - - -class AdvancedExecuteRequest(BaseModel): - code: str - timeout: Optional[int] = 30 - dependencies: List[str] = [] - output_files: List[str] = [] - - -class FileResult(BaseModel): - filename: str - data: str # base64 encoded - size: int - - -class AdvancedExecuteResponse(BaseModel): - success: bool - stdout: str - stderr: str - error: Optional[str] = None - plots: List[PlotResult] = [] - files: List[FileResult] = [] - execution_time: Optional[float] = None - install_output: str = "" - - -def create_wrapper_code(user_code: str, plot_dir: str) -> str: - """ - Wrap user code with matplotlib backend setup for headless plot generation. - """ - wrapper = f''' -import sys -import os - -# Set matplotlib to use non-interactive backend BEFORE importing pyplot -import matplotlib -matplotlib.use('Agg') - -# Configure plot output directory -_plot_dir = {repr(plot_dir)} -_plot_counter = [0] - -# Patch plt.show() to save plots instead -import matplotlib.pyplot as plt -_original_show = plt.show - -def _patched_show(*args, **kwargs): - _plot_counter[0] += 1 - plot_path = os.path.join(_plot_dir, f"plot_{{_plot_counter[0]}}.png") - plt.savefig(plot_path, format='png', dpi=100, bbox_inches='tight') - print(f"[PLOT_SAVED]{{plot_path}}") - plt.close() - -plt.show = _patched_show - -# Also save figures when plt.savefig is called -_original_savefig = plt.savefig - -def _patched_savefig(fname, *args, **kwargs): - # Call original savefig - result = _original_savefig(fname, *args, **kwargs) - # Also save to our plot dir if it's a new file - if not str(fname).startswith(_plot_dir): - _plot_counter[0] += 1 - copy_path = os.path.join(_plot_dir, f"plot_{{_plot_counter[0]}}.png") - _original_savefig(copy_path, format='png', dpi=100, bbox_inches='tight') - print(f"[PLOT_SAVED]{{copy_path}}") - return result - -# Don't patch savefig - let user control where files go -# plt.savefig = _patched_savefig - -# Change to work directory -os.chdir({repr(plot_dir)}) - -# Execute user code -{user_code} -''' - return wrapper - - -def execute_python_code(code: str, timeout: int, work_dir: Path, dependencies: List[str] = None) -> dict: - """ - Execute Python code in a subprocess with timeout. - Returns dict with stdout, stderr, error, plots, files. - """ - start_time = time.time() - result = { - "stdout": "", - "stderr": "", - "error": None, - "plots": [], - "files": [], - "install_output": "", - "execution_time": 0 - } - - # Create temporary directory for this execution - exec_id = str(uuid.uuid4())[:8] - exec_dir = work_dir / exec_id - exec_dir.mkdir(parents=True, exist_ok=True) - plot_dir = exec_dir / "plots" - plot_dir.mkdir(exist_ok=True) - - try: - # Install dependencies if requested - if dependencies: - logger.info(f"Installing dependencies: {dependencies}") - try: - install_result = subprocess.run( - [sys.executable, "-m", "pip", "install", "-q"] + dependencies, - capture_output=True, - text=True, - timeout=60 - ) - result["install_output"] = install_result.stdout + install_result.stderr - if install_result.returncode != 0: - result["error"] = f"Failed to install dependencies: {result['install_output']}" - result["execution_time"] = time.time() - start_time - return result - logger.info(f"Dependencies installed: {result['install_output'][:200]}") - except subprocess.TimeoutExpired: - result["error"] = "Dependency installation timed out" - result["execution_time"] = time.time() - start_time - return result - - # Wrap code with matplotlib setup - wrapped_code = create_wrapper_code(code, str(exec_dir)) - - # Write code to temp file - code_file = exec_dir / "script.py" - code_file.write_text(wrapped_code) - - # Execute with timeout - try: - proc = subprocess.run( - [sys.executable, str(code_file)], - capture_output=True, - text=True, - timeout=timeout, - cwd=str(exec_dir) - ) - - result["stdout"] = proc.stdout[:MAX_OUTPUT_SIZE] if proc.stdout else "" - result["stderr"] = proc.stderr[:MAX_OUTPUT_SIZE] if proc.stderr else "" - - if proc.returncode != 0: - # Extract just the error message, not the full traceback if possible - stderr = result["stderr"] - if "Error:" in stderr: - result["error"] = stderr.split("\n")[-2] if stderr else "Execution failed" - else: - result["error"] = stderr or "Execution failed with non-zero exit code" - - except subprocess.TimeoutExpired: - result["error"] = f"Execution timed out after {timeout} seconds" - result["stderr"] = f"TimeoutError: Code execution exceeded {timeout} second limit" - - # Collect plots - plot_files = list(plot_dir.glob("*.png")) + list(exec_dir.glob("*.png")) - for plot_file in plot_files: - try: - with open(plot_file, "rb") as f: - plot_data = base64.b64encode(f.read()).decode("utf-8") - result["plots"].append({ - "format": "png", - "data": plot_data - }) - logger.info(f"Collected plot: {plot_file.name}") - except Exception as e: - logger.warning(f"Failed to read plot {plot_file}: {e}") - - # Remove [PLOT_SAVED] messages from stdout - if result["stdout"]: - lines = result["stdout"].split("\n") - result["stdout"] = "\n".join( - line for line in lines if not line.startswith("[PLOT_SAVED]") - ) - - # Collect any other generated files (excluding script.py and plots) - for file_path in exec_dir.iterdir(): - if file_path.is_file() and file_path.name != "script.py": - if file_path.suffix.lower() not in [".png", ".pyc"]: - try: - with open(file_path, "rb") as f: - file_data = f.read() - result["files"].append({ - "filename": file_path.name, - "data": base64.b64encode(file_data).decode("utf-8"), - "size": len(file_data) - }) - logger.info(f"Collected file: {file_path.name} ({len(file_data)} bytes)") - except Exception as e: - logger.warning(f"Failed to read file {file_path}: {e}") - - finally: - # Clean up execution directory - try: - shutil.rmtree(exec_dir) - except Exception as e: - logger.warning(f"Failed to cleanup exec dir: {e}") - - result["execution_time"] = time.time() - start_time - - return result - - -# Health check endpoint -@app.get("/health") -async def health_check(): - """Health check endpoint""" - return { - "status": "healthy", - "service": "simple-python-executor", - "mode": "subprocess", - "e2b_api_key_set": False - } - - -# Execute Python code endpoint -@app.post("/execute", response_model=ExecuteResponse) -async def execute_code(request: ExecuteRequest): - """ - Execute Python code using subprocess. - """ - logger.info(f"Executing code (length: {len(request.code)} chars)") - - timeout = min(request.timeout or EXECUTION_TIMEOUT, EXECUTION_TIMEOUT) - result = execute_python_code(request.code, timeout, WORK_DIR) - - response = ExecuteResponse( - success=result["error"] is None, - stdout=result["stdout"], - stderr=result["stderr"], - error=result["error"], - plots=[PlotResult(**p) for p in result["plots"]], - execution_time=result["execution_time"] - ) - - logger.info(f"Execution completed: success={response.success}, plots={len(response.plots)}") - return response - - -# Execute with file upload endpoint -@app.post("/execute-with-files", response_model=ExecuteResponse) -async def execute_with_files( - code: str = Form(...), - files: List[UploadFile] = File(...), - timeout: int = Form(30) -): - """ - Execute Python code with uploaded files. - """ - logger.info(f"Executing code with {len(files)} files") - - # Create temp directory and save uploaded files - exec_id = str(uuid.uuid4())[:8] - exec_dir = WORK_DIR / exec_id - exec_dir.mkdir(parents=True, exist_ok=True) - - try: - # Save uploaded files - for file in files: - content = await file.read() - file_path = exec_dir / file.filename - file_path.write_bytes(content) - logger.info(f"Uploaded file: {file.filename} ({len(content)} bytes)") - - # Prepend code to change to the directory with uploaded files - full_code = f"import os; os.chdir({repr(str(exec_dir))})\n{code}" - - timeout = min(timeout, EXECUTION_TIMEOUT) - result = execute_python_code(full_code, timeout, exec_dir) - - response = ExecuteResponse( - success=result["error"] is None, - stdout=result["stdout"], - stderr=result["stderr"], - error=result["error"], - plots=[PlotResult(**p) for p in result["plots"]], - execution_time=result["execution_time"] - ) - - logger.info(f"Execution with files completed: success={response.success}") - return response - - finally: - # Cleanup - try: - shutil.rmtree(exec_dir) - except: - pass - - -# Advanced execution endpoint -@app.post("/execute-advanced", response_model=AdvancedExecuteResponse) -async def execute_advanced(request: AdvancedExecuteRequest): - """ - Execute Python code with pip dependencies and output file retrieval. - """ - logger.info(f"Advanced execution: code={len(request.code)} chars, deps={request.dependencies}, output_files={request.output_files}") - - timeout = min(request.timeout or EXECUTION_TIMEOUT, EXECUTION_TIMEOUT) - result = execute_python_code( - request.code, - timeout, - WORK_DIR, - dependencies=request.dependencies - ) - - response = AdvancedExecuteResponse( - success=result["error"] is None, - stdout=result["stdout"], - stderr=result["stderr"], - error=result["error"], - plots=[PlotResult(**p) for p in result["plots"]], - files=[FileResult(**f) for f in result["files"]], - execution_time=result["execution_time"], - install_output=result["install_output"] - ) - - logger.info(f"Advanced execution completed: success={response.success}, plots={len(response.plots)}, files={len(response.files)}") - return response - - -if __name__ == "__main__": - import uvicorn - uvicorn.run( - app, - host="0.0.0.0", - port=8001, - log_level="info" - ) diff --git a/backend/e2b-service/requirements.txt b/backend/e2b-service/requirements.txt deleted file mode 100644 index baf49fea..00000000 --- a/backend/e2b-service/requirements.txt +++ /dev/null @@ -1,11 +0,0 @@ -fastapi>=0.109.0 -uvicorn>=0.27.0 -pydantic>=2.5.0 -e2b-code-interpreter>=0.0.11 -python-multipart>=0.0.6 -requests>=2.31.0 -# For simple executor (all-in-one mode) - plot generation -matplotlib>=3.8.0 -numpy>=1.26.0 -pandas>=2.1.0 - diff --git a/backend/go.mod b/backend/go.mod deleted file mode 100644 index 6d4affc0..00000000 --- a/backend/go.mod +++ /dev/null @@ -1,118 +0,0 @@ -module claraverse - -go 1.25.5 - -require ( - github.com/ansrivas/fiberprometheus/v2 v2.14.0 - github.com/chromedp/cdproto v0.0.0-20250724212937-08a3db8b4327 - github.com/chromedp/chromedp v0.14.2 - github.com/dodopayments/dodopayments-go v1.70.0 - github.com/fsnotify/fsnotify v1.9.0 - github.com/go-co-op/gocron/v2 v2.14.0 - github.com/gofiber/contrib/websocket v1.3.4 - github.com/gofiber/fiber/v2 v2.52.9 - github.com/google/uuid v1.6.0 - github.com/invopop/jsonschema v0.13.0 - github.com/joho/godotenv v1.5.1 - github.com/ledongthuc/pdf v0.0.0-20250511090121-5959a4027728 - github.com/markusmobius/go-trafilatura v1.12.2 - github.com/patrickmn/go-cache v2.1.0+incompatible - github.com/prometheus/client_golang v1.23.2 - github.com/redis/go-redis/v9 v9.7.0 - github.com/robfig/cron/v3 v3.0.1 - github.com/sirupsen/logrus v1.9.3 - github.com/temoto/robotstxt v1.1.2 - github.com/xuri/excelize/v2 v2.10.0 - github.com/yuin/goldmark v1.7.13 - go.mongodb.org/mongo-driver v1.17.1 - golang.org/x/crypto v0.47.0 - golang.org/x/time v0.14.0 - modernc.org/sqlite v1.40.1 -) - -require ( - filippo.io/edwards25519 v1.1.0 // indirect - github.com/RadhiFadlillah/whatlanggo v0.0.0-20240916001553-aac1f0f737fc // indirect - github.com/andybalholm/brotli v1.2.0 // indirect - github.com/andybalholm/cascadia v1.3.3 // indirect - github.com/araddon/dateparse v0.0.0-20210429162001-6b43995a97de // indirect - github.com/bahlo/generic-list-go v0.2.0 // indirect - github.com/beorn7/perks v1.0.1 // indirect - github.com/buger/jsonparser v1.1.1 // indirect - github.com/cespare/xxhash/v2 v2.3.0 // indirect - github.com/chromedp/sysutil v1.1.0 // indirect - github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect - github.com/dustin/go-humanize v1.0.1 // indirect - github.com/elliotchance/pie/v2 v2.9.0 // indirect - github.com/fasthttp/websocket v1.5.8 // indirect - github.com/forPelevin/gomoji v1.2.0 // indirect - github.com/go-json-experiment/json v0.0.0-20250725192818-e39067aee2d2 // indirect - github.com/go-shiori/dom v0.0.0-20230515143342-73569d674e1c // indirect - github.com/go-shiori/go-readability v0.0.0-20241012063810-92284fa8a71f // indirect - github.com/go-sql-driver/mysql v1.9.3 // indirect - github.com/gobwas/httphead v0.1.0 // indirect - github.com/gobwas/pool v0.2.1 // indirect - github.com/gobwas/ws v1.4.0 // indirect - github.com/gogs/chardet v0.0.0-20211120154057-b7413eaefb8f // indirect - github.com/golang-jwt/jwt/v5 v5.3.0 // indirect - github.com/golang/snappy v0.0.4 // indirect - github.com/hablullah/go-hijri v1.0.2 // indirect - github.com/hablullah/go-juliandays v1.0.0 // indirect - github.com/jalaali/go-jalaali v0.0.0-20210801064154-80525e88d958 // indirect - github.com/jonboulle/clockwork v0.4.0 // indirect - github.com/klauspost/compress v1.18.0 // indirect - github.com/mailru/easyjson v0.7.7 // indirect - github.com/markusmobius/go-dateparser v1.2.3 // indirect - github.com/markusmobius/go-domdistiller v0.0.0-20240926050704-25b8d046ffb4 // indirect - github.com/markusmobius/go-htmldate v1.9.1 // indirect - github.com/mattn/go-colorable v0.1.13 // indirect - github.com/mattn/go-isatty v0.0.20 // indirect - github.com/mattn/go-runewidth v0.0.16 // indirect - github.com/montanaflynn/stats v0.7.1 // indirect - github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 // indirect - github.com/ncruces/go-strftime v1.0.0 // indirect - github.com/philhofer/fwd v1.1.3-0.20240916144458-20a13a1f6b7c // indirect - github.com/prometheus/client_model v0.6.2 // indirect - github.com/prometheus/common v0.66.1 // indirect - github.com/prometheus/procfs v0.16.1 // indirect - github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec // indirect - github.com/richardlehane/mscfb v1.0.4 // indirect - github.com/richardlehane/msoleps v1.0.4 // indirect - github.com/rivo/uniseg v0.4.7 // indirect - github.com/rs/zerolog v1.33.0 // indirect - github.com/savsgio/gotils v0.0.0-20240303185622-093b76447511 // indirect - github.com/sergi/go-diff v1.4.0 // indirect - github.com/standard-webhooks/standard-webhooks/libraries v0.0.0-20251210175704-b03a68fe8b19 // indirect - github.com/tetratelabs/wazero v1.8.1 // indirect - github.com/tidwall/gjson v1.14.4 // indirect - github.com/tidwall/match v1.1.1 // indirect - github.com/tidwall/pretty v1.2.1 // indirect - github.com/tidwall/sjson v1.2.5 // indirect - github.com/tiendc/go-deepcopy v1.7.1 // indirect - github.com/tinylib/msgp v1.2.5 // indirect - github.com/valyala/bytebufferpool v1.0.0 // indirect - github.com/valyala/fasthttp v1.65.0 // indirect - github.com/wasilibs/go-re2 v1.7.0 // indirect - github.com/wasilibs/wazero-helpers v0.0.0-20240620070341-3dff1577cd52 // indirect - github.com/wk8/go-ordered-map/v2 v2.1.8 // indirect - github.com/xdg-go/pbkdf2 v1.0.0 // indirect - github.com/xdg-go/scram v1.1.2 // indirect - github.com/xdg-go/stringprep v1.0.4 // indirect - github.com/xuri/efp v0.0.1 // indirect - github.com/xuri/nfp v0.0.2-0.20250530014748-2ddeb826f9a9 // indirect - github.com/yosssi/gohtml v0.0.0-20201013000340-ee4748c638f4 // indirect - github.com/youmark/pkcs8 v0.0.0-20240726163527-a2c0da244d78 // indirect - go.opentelemetry.io/otel v1.37.0 // indirect - go.opentelemetry.io/otel/trace v1.37.0 // indirect - go.yaml.in/yaml/v2 v2.4.2 // indirect - golang.org/x/exp v0.0.0-20251023183803-a4bb9ffd2546 // indirect - golang.org/x/net v0.48.0 // indirect - golang.org/x/sync v0.19.0 // indirect - golang.org/x/sys v0.40.0 // indirect - golang.org/x/text v0.33.0 // indirect - google.golang.org/protobuf v1.36.8 // indirect - gopkg.in/yaml.v3 v3.0.1 // indirect - modernc.org/libc v1.66.10 // indirect - modernc.org/mathutil v1.7.1 // indirect - modernc.org/memory v1.11.0 // indirect -) diff --git a/backend/go.sum b/backend/go.sum deleted file mode 100644 index f267d970..00000000 --- a/backend/go.sum +++ /dev/null @@ -1,396 +0,0 @@ -filippo.io/edwards25519 v1.1.0 h1:FNf4tywRC1HmFuKW5xopWpigGjJKiJSV0Cqo0cJWDaA= -filippo.io/edwards25519 v1.1.0/go.mod h1:BxyFTGdWcka3PhytdK4V28tE5sGfRvvvRV7EaN4VDT4= -github.com/RadhiFadlillah/whatlanggo v0.0.0-20240916001553-aac1f0f737fc h1:6aA31zw7fnfJ/G1ebisIesCDl44slkIVFqk3YTSadd8= -github.com/RadhiFadlillah/whatlanggo v0.0.0-20240916001553-aac1f0f737fc/go.mod h1:PgrPWaMBxL1lyq1k5DEMqC0Y67R3pG1vEsHzxFXeDxc= -github.com/andybalholm/brotli v1.2.0 h1:ukwgCxwYrmACq68yiUqwIWnGY0cTPox/M94sVwToPjQ= -github.com/andybalholm/brotli v1.2.0/go.mod h1:rzTDkvFWvIrjDXZHkuS16NPggd91W3kUSvPlQ1pLaKY= -github.com/andybalholm/cascadia v1.3.3 h1:AG2YHrzJIm4BZ19iwJ/DAua6Btl3IwJX+VI4kktS1LM= -github.com/andybalholm/cascadia v1.3.3/go.mod h1:xNd9bqTn98Ln4DwST8/nG+H0yuB8Hmgu1YHNnWw0GeA= -github.com/ansrivas/fiberprometheus/v2 v2.14.0 h1:4DhjAk+zA2cRA8VSlZBLjCms40AITc9Cbs8Y/ovq/SU= -github.com/ansrivas/fiberprometheus/v2 v2.14.0/go.mod h1:sekqW4C04j0fWHXrimsTTX7ZUbPnX0d/8w+E5SxHTeg= -github.com/araddon/dateparse v0.0.0-20210429162001-6b43995a97de h1:FxWPpzIjnTlhPwqqXc4/vE0f7GvRjuAsbW+HOIe8KnA= -github.com/araddon/dateparse v0.0.0-20210429162001-6b43995a97de/go.mod h1:DCaWoUhZrYW9p1lxo/cm8EmUOOzAPSEZNGF2DK1dJgw= -github.com/bahlo/generic-list-go v0.2.0 h1:5sz/EEAK+ls5wF+NeqDpk5+iNdMDXrh3z3nPnH1Wvgk= -github.com/bahlo/generic-list-go v0.2.0/go.mod h1:2KvAjgMlE5NNynlg/5iLrrCCZ2+5xWbdbCW3pNTGyYg= -github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= -github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= -github.com/bsm/ginkgo/v2 v2.12.0 h1:Ny8MWAHyOepLGlLKYmXG4IEkioBysk6GpaRTLC8zwWs= -github.com/bsm/ginkgo/v2 v2.12.0/go.mod h1:SwYbGRRDovPVboqFv0tPTcG1sN61LM1Z4ARdbAV9g4c= -github.com/bsm/gomega v1.27.10 h1:yeMWxP2pV2fG3FgAODIY8EiRE3dy0aeFYt4l7wh6yKA= -github.com/bsm/gomega v1.27.10/go.mod h1:JyEr/xRbxbtgWNi8tIEVPUYZ5Dzef52k01W3YH0H+O0= -github.com/buger/jsonparser v1.1.1 h1:2PnMjfWD7wBILjqQbt530v576A/cAbQvEW9gGIpYMUs= -github.com/buger/jsonparser v1.1.1/go.mod h1:6RYKKt7H4d4+iWqouImQ9R2FZql3VbhNgx27UK13J/0= -github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs= -github.com/cespare/xxhash/v2 v2.3.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= -github.com/chromedp/cdproto v0.0.0-20250724212937-08a3db8b4327 h1:UQ4AU+BGti3Sy/aLU8KVseYKNALcX9UXY6DfpwQ6J8E= -github.com/chromedp/cdproto v0.0.0-20250724212937-08a3db8b4327/go.mod h1:NItd7aLkcfOA/dcMXvl8p1u+lQqioRMq/SqDp71Pb/k= -github.com/chromedp/chromedp v0.14.2 h1:r3b/WtwM50RsBZHMUm9fsNhhzRStTHrKdr2zmwbZSzM= -github.com/chromedp/chromedp v0.14.2/go.mod h1:rHzAv60xDE7VNy/MYtTUrYreSc0ujt2O1/C3bzctYBo= -github.com/chromedp/sysutil v1.1.0 h1:PUFNv5EcprjqXZD9nJb9b/c9ibAbxiYo4exNWZyipwM= -github.com/chromedp/sysutil v1.1.0/go.mod h1:WiThHUdltqCNKGc4gaU50XgYjwjYIhKWoHGPTUfWTJ8= -github.com/coreos/go-systemd/v22 v22.5.0/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc= -github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= -github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= -github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= -github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78= -github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc= -github.com/dodopayments/dodopayments-go v1.70.0 h1:ejsToCzKxWwPRp/uAmPIyulhoYxKjHZNB4CVudlZIno= -github.com/dodopayments/dodopayments-go v1.70.0/go.mod h1:8ZBB5JQSIA5r3jLLVZ+rbIYDaidg/+8oSzCj6FZuVTM= -github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY= -github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto= -github.com/elliotchance/pie/v2 v2.9.0 h1:BkEhh8b/avGCSpXpABSjNuytxlI/S2snkjT3vtVORjw= -github.com/elliotchance/pie/v2 v2.9.0/go.mod h1:18t0dgGFH006g4eVdDtWfgFZPQEgl10IoEO8YWEq3Og= -github.com/fasthttp/websocket v1.5.8 h1:k5DpirKkftIF/w1R8ZzjSgARJrs54Je9YJK37DL/Ah8= -github.com/fasthttp/websocket v1.5.8/go.mod h1:d08g8WaT6nnyvg9uMm8K9zMYyDjfKyj3170AtPRuVU0= -github.com/forPelevin/gomoji v1.2.0 h1:9k4WVSSkE1ARO/BWywxgEUBvR/jMnao6EZzrql5nxJ8= -github.com/forPelevin/gomoji v1.2.0/go.mod h1:8+Z3KNGkdslmeGZBC3tCrwMrcPy5GRzAD+gL9NAwMXg= -github.com/fsnotify/fsnotify v1.9.0 h1:2Ml+OJNzbYCTzsxtv8vKSFD9PbJjmhYF14k/jKC7S9k= -github.com/fsnotify/fsnotify v1.9.0/go.mod h1:8jBTzvmWwFyi3Pb8djgCCO5IBqzKJ/Jwo8TRcHyHii0= -github.com/go-co-op/gocron/v2 v2.14.0 h1:bWPJeIdd4ioqiEpLLD1BVSTrtae7WABhX/WaVJbKVqg= -github.com/go-co-op/gocron/v2 v2.14.0/go.mod h1:ZF70ZwEqz0OO4RBXE1sNxnANy/zvwLcattWEFsqpKig= -github.com/go-json-experiment/json v0.0.0-20250725192818-e39067aee2d2 h1:iizUGZ9pEquQS5jTGkh4AqeeHCMbfbjeb0zMt0aEFzs= -github.com/go-json-experiment/json v0.0.0-20250725192818-e39067aee2d2/go.mod h1:TiCD2a1pcmjd7YnhGH0f/zKNcCD06B029pHhzV23c2M= -github.com/go-logr/logr v1.4.3 h1:CjnDlHq8ikf6E492q6eKboGOC0T8CDaOvkHCIg8idEI= -github.com/go-logr/logr v1.4.3/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY= -github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag= -github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE= -github.com/go-shiori/dom v0.0.0-20230515143342-73569d674e1c h1:wpkoddUomPfHiOziHZixGO5ZBS73cKqVzZipfrLmO1w= -github.com/go-shiori/dom v0.0.0-20230515143342-73569d674e1c/go.mod h1:oVDCh3qjJMLVUSILBRwrm+Bc6RNXGZYtoh9xdvf1ffM= -github.com/go-shiori/go-readability v0.0.0-20241012063810-92284fa8a71f h1:cypj7SJh+47G9J3VCPdMzT3uWcXWAWDJA54ErTfOigI= -github.com/go-shiori/go-readability v0.0.0-20241012063810-92284fa8a71f/go.mod h1:YWa00ashoPZMAOElrSn4E1cJErhDVU6PWAll4Hxzn+w= -github.com/go-sql-driver/mysql v1.9.3 h1:U/N249h2WzJ3Ukj8SowVFjdtZKfu9vlLZxjPXV1aweo= -github.com/go-sql-driver/mysql v1.9.3/go.mod h1:qn46aNg1333BRMNU69Lq93t8du/dwxI64Gl8i5p1WMU= -github.com/gobwas/httphead v0.1.0 h1:exrUm0f4YX0L7EBwZHuCF4GDp8aJfVeBrlLQrs6NqWU= -github.com/gobwas/httphead v0.1.0/go.mod h1:O/RXo79gxV8G+RqlR/otEwx4Q36zl9rqC5u12GKvMCM= -github.com/gobwas/pool v0.2.1 h1:xfeeEhW7pwmX8nuLVlqbzVc7udMDrwetjEv+TZIz1og= -github.com/gobwas/pool v0.2.1/go.mod h1:q8bcK0KcYlCgd9e7WYLm9LpyS+YeLd8JVDW6WezmKEw= -github.com/gobwas/ws v1.4.0 h1:CTaoG1tojrh4ucGPcoJFiAQUAsEWekEWvLy7GsVNqGs= -github.com/gobwas/ws v1.4.0/go.mod h1:G3gNqMNtPppf5XUz7O4shetPpcZ1VJ7zt18dlUeakrc= -github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= -github.com/gofiber/contrib/websocket v1.3.4 h1:tWeBdbJ8q0WFQXariLN4dBIbGH9KBU75s0s7YXplOSg= -github.com/gofiber/contrib/websocket v1.3.4/go.mod h1:kTFBPC6YENCnKfKx0BoOFjgXxdz7E85/STdkmZPEmPs= -github.com/gofiber/fiber/v2 v2.52.9 h1:YjKl5DOiyP3j0mO61u3NTmK7or8GzzWzCFzkboyP5cw= -github.com/gofiber/fiber/v2 v2.52.9/go.mod h1:YEcBbO/FB+5M1IZNBP9FO3J9281zgPAreiI1oqg8nDw= -github.com/gogs/chardet v0.0.0-20211120154057-b7413eaefb8f h1:3BSP1Tbs2djlpprl7wCLuiqMaUh5SJkkzI2gDs+FgLs= -github.com/gogs/chardet v0.0.0-20211120154057-b7413eaefb8f/go.mod h1:Pcatq5tYkCW2Q6yrR2VRHlbHpZ/R4/7qyL1TCF7vl14= -github.com/golang-jwt/jwt/v5 v5.3.0 h1:pv4AsKCKKZuqlgs5sUmn4x8UlGa0kEVt/puTpKx9vvo= -github.com/golang-jwt/jwt/v5 v5.3.0/go.mod h1:fxCRLWMO43lRc8nhHWY6LGqRcf+1gQWArsqaEUEa5bE= -github.com/golang/snappy v0.0.4 h1:yAGX7huGHXlcLOEtBnF4w7FQwA26wojNCwOYAEhLjQM= -github.com/golang/snappy v0.0.4/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= -github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= -github.com/google/go-cmp v0.7.0 h1:wk8382ETsv4JYUZwIsn6YpYiWiBsYLSJiTsyBybVuN8= -github.com/google/go-cmp v0.7.0/go.mod h1:pXiqmnSA92OHEEa9HXL2W4E7lf9JzCmGVUdgjX3N/iU= -github.com/google/pprof v0.0.0-20250317173921-a4b03ec1a45e h1:ijClszYn+mADRFY17kjQEVQ1XRhq2/JR1M3sGqeJoxs= -github.com/google/pprof v0.0.0-20250317173921-a4b03ec1a45e/go.mod h1:boTsfXsheKC2y+lKOCMpSfarhxDeIzfZG1jqGcPl3cA= -github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= -github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= -github.com/hablullah/go-hijri v1.0.2 h1:drT/MZpSZJQXo7jftf5fthArShcaMtsal0Zf/dnmp6k= -github.com/hablullah/go-hijri v1.0.2/go.mod h1:OS5qyYLDjORXzK4O1adFw9Q5WfhOcMdAKglDkcTxgWQ= -github.com/hablullah/go-juliandays v1.0.0 h1:A8YM7wIj16SzlKT0SRJc9CD29iiaUzpBLzh5hr0/5p0= -github.com/hablullah/go-juliandays v1.0.0/go.mod h1:0JOYq4oFOuDja+oospuc61YoX+uNEn7Z6uHYTbBzdGc= -github.com/invopop/jsonschema v0.13.0 h1:KvpoAJWEjR3uD9Kbm2HWJmqsEaHt8lBUpd0qHcIi21E= -github.com/invopop/jsonschema v0.13.0/go.mod h1:ffZ5Km5SWWRAIN6wbDXItl95euhFz2uON45H2qjYt+0= -github.com/jalaali/go-jalaali v0.0.0-20210801064154-80525e88d958 h1:qxLoi6CAcXVzjfvu+KXIXJOAsQB62LXjsfbOaErsVzE= -github.com/jalaali/go-jalaali v0.0.0-20210801064154-80525e88d958/go.mod h1:Wqfu7mjUHj9WDzSSPI5KfBclTTEnLveRUFr/ujWnTgE= -github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0= -github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4= -github.com/jonboulle/clockwork v0.4.0 h1:p4Cf1aMWXnXAUh8lVfewRBx1zaTSYKrKMF2g3ST4RZ4= -github.com/jonboulle/clockwork v0.4.0/go.mod h1:xgRqUGwRcjKCO1vbZUEtSLrqKoPSsUpK7fnezOII0kc= -github.com/josharian/intern v1.0.0/go.mod h1:5DoeVV0s6jJacbCEi61lwdGj/aVlrQvzHFFd8Hwg//Y= -github.com/klauspost/compress v1.18.0 h1:c/Cqfb0r+Yi+JtIEq73FWXVkRonBlf0CRNYc8Zttxdo= -github.com/klauspost/compress v1.18.0/go.mod h1:2Pp+KzxcywXVXMr50+X0Q/Lsb43OQHYWRCY2AiWywWQ= -github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= -github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= -github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= -github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= -github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= -github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= -github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= -github.com/kylelemons/godebug v1.1.0 h1:RPNrshWIDI6G2gRW9EHilWtl7Z6Sb1BR0xunSBf0SNc= -github.com/kylelemons/godebug v1.1.0/go.mod h1:9/0rRGxNHcop5bhtWyNeEfOS8JIWk580+fNqagV/RAw= -github.com/ledongthuc/pdf v0.0.0-20250511090121-5959a4027728 h1:QwWKgMY28TAXaDl+ExRDqGQltzXqN/xypdKP86niVn8= -github.com/ledongthuc/pdf v0.0.0-20250511090121-5959a4027728/go.mod h1:1fEHWurg7pvf5SG6XNE5Q8UZmOwex51Mkx3SLhrW5B4= -github.com/magefile/mage v1.15.1-0.20230912152418-9f54e0f83e2a h1:tdPcGgyiH0K+SbsJBBm2oPyEIOTAvLBwD9TuUwVtZho= -github.com/magefile/mage v1.15.1-0.20230912152418-9f54e0f83e2a/go.mod h1:z5UZb/iS3GoOSn0JgWuiw7dxlurVYTu+/jHXqQg881A= -github.com/mailru/easyjson v0.7.7 h1:UGYAvKxe3sBsEDzO8ZeWOSlIQfWFlxbzLZe7hwFURr0= -github.com/mailru/easyjson v0.7.7/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc= -github.com/markusmobius/go-dateparser v1.2.3 h1:TvrsIvr5uk+3v6poDjaicnAFJ5IgtFHgLiuMY2Eb7Nw= -github.com/markusmobius/go-dateparser v1.2.3/go.mod h1:cMwQRrBUQlK1UI5TIFHEcvpsMbkWrQLXuaPNMFzuYLk= -github.com/markusmobius/go-domdistiller v0.0.0-20240926050704-25b8d046ffb4 h1:+7kfF1+dmSXV469sqjeNC+eKJF7xDuS5mvZA3DFVLLY= -github.com/markusmobius/go-domdistiller v0.0.0-20240926050704-25b8d046ffb4/go.mod h1:E7PoeC3nd4GqtxP1A64v7JDBxpAbpTSnhlq9/DHmQ28= -github.com/markusmobius/go-htmldate v1.9.1 h1:0kfVz0wdxGCBaotWNzdtIZKhy7+8ClBlzvANQ67Rlt8= -github.com/markusmobius/go-htmldate v1.9.1/go.mod h1:fLls4rjQDxYR+Pxhf0YR6Ht8dEeHd4SxK/NPaVqhMa8= -github.com/markusmobius/go-trafilatura v1.12.2 h1:JgEto0kDjwTuyXFl6TB+psrs1QGJqTdYJEbLhDy1vrw= -github.com/markusmobius/go-trafilatura v1.12.2/go.mod h1:2WnYLuvGBgJAarHaAQnsvofihEojt2xDDrtVJU5UXZI= -github.com/mattn/go-colorable v0.1.13 h1:fFA4WZxdEF4tXPZVKMLwD8oUnCTTo08duU7wxecdEvA= -github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg= -github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= -github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= -github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY= -github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= -github.com/mattn/go-runewidth v0.0.10/go.mod h1:RAqKPSqVFrSLVXbA8x7dzmKdmGzieGRCM46jaSJTDAk= -github.com/mattn/go-runewidth v0.0.16 h1:E5ScNMtiwvlvB5paMFdw9p4kSQzbXFikJ5SQO6TULQc= -github.com/mattn/go-runewidth v0.0.16/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w= -github.com/montanaflynn/stats v0.7.1 h1:etflOAAHORrCC44V+aR6Ftzort912ZU+YLiSTuV8eaE= -github.com/montanaflynn/stats v0.7.1/go.mod h1:etXPPgVO6n31NxCd9KQUMvCM+ve0ruNzt6R8Bnaayow= -github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 h1:C3w9PqII01/Oq1c1nUAm88MOHcQC9l5mIlSMApZMrHA= -github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ= -github.com/ncruces/go-strftime v1.0.0 h1:HMFp8mLCTPp341M/ZnA4qaf7ZlsbTc+miZjCLOFAw7w= -github.com/ncruces/go-strftime v1.0.0/go.mod h1:Fwc5htZGVVkseilnfgOVb9mKy6w1naJmn9CehxcKcls= -github.com/orisano/pixelmatch v0.0.0-20220722002657-fb0b55479cde h1:x0TT0RDC7UhAVbbWWBzr41ElhJx5tXPWkIHA2HWPRuw= -github.com/orisano/pixelmatch v0.0.0-20220722002657-fb0b55479cde/go.mod h1:nZgzbfBr3hhjoZnS66nKrHmduYNpc34ny7RK4z5/HM0= -github.com/patrickmn/go-cache v2.1.0+incompatible h1:HRMgzkcYKYpi3C8ajMPV8OFXaaRUnok+kx1WdO15EQc= -github.com/patrickmn/go-cache v2.1.0+incompatible/go.mod h1:3Qf8kWWT7OJRJbdiICTKqZju1ZixQ/KpMGzzAfe6+WQ= -github.com/philhofer/fwd v1.1.3-0.20240916144458-20a13a1f6b7c h1:dAMKvw0MlJT1GshSTtih8C2gDs04w8dReiOGXrGLNoY= -github.com/philhofer/fwd v1.1.3-0.20240916144458-20a13a1f6b7c/go.mod h1:RqIHx9QI14HlwKwm98g9Re5prTQ6LdeRQn+gXJFxsJM= -github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= -github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= -github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= -github.com/prometheus/client_golang v1.23.2 h1:Je96obch5RDVy3FDMndoUsjAhG5Edi49h0RJWRi/o0o= -github.com/prometheus/client_golang v1.23.2/go.mod h1:Tb1a6LWHB3/SPIzCoaDXI4I8UHKeFTEQ1YCr+0Gyqmg= -github.com/prometheus/client_model v0.6.2 h1:oBsgwpGs7iVziMvrGhE53c/GrLUsZdHnqNwqPLxwZyk= -github.com/prometheus/client_model v0.6.2/go.mod h1:y3m2F6Gdpfy6Ut/GBsUqTWZqCUvMVzSfMLjcu6wAwpE= -github.com/prometheus/common v0.66.1 h1:h5E0h5/Y8niHc5DlaLlWLArTQI7tMrsfQjHV+d9ZoGs= -github.com/prometheus/common v0.66.1/go.mod h1:gcaUsgf3KfRSwHY4dIMXLPV0K/Wg1oZ8+SbZk/HH/dA= -github.com/prometheus/procfs v0.16.1 h1:hZ15bTNuirocR6u0JZ6BAHHmwS1p8B4P6MRqxtzMyRg= -github.com/prometheus/procfs v0.16.1/go.mod h1:teAbpZRB1iIAJYREa1LsoWUXykVXA1KlTmWl8x/U+Is= -github.com/redis/go-redis/v9 v9.7.0 h1:HhLSs+B6O021gwzl+locl0zEDnyNkxMtf/Z3NNBMa9E= -github.com/redis/go-redis/v9 v9.7.0/go.mod h1:f6zhXITC7JUJIlPEiBOTXxJgPLdZcA93GewI7inzyWw= -github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE= -github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= -github.com/richardlehane/mscfb v1.0.4 h1:WULscsljNPConisD5hR0+OyZjwK46Pfyr6mPu5ZawpM= -github.com/richardlehane/mscfb v1.0.4/go.mod h1:YzVpcZg9czvAuhk9T+a3avCpcFPMUWm7gK3DypaEsUk= -github.com/richardlehane/msoleps v1.0.1/go.mod h1:BWev5JBpU9Ko2WAgmZEuiz4/u3ZYTKbjLycmwiWUfWg= -github.com/richardlehane/msoleps v1.0.4 h1:WuESlvhX3gH2IHcd8UqyCuFY5yiq/GR/yqaSM/9/g00= -github.com/richardlehane/msoleps v1.0.4/go.mod h1:BWev5JBpU9Ko2WAgmZEuiz4/u3ZYTKbjLycmwiWUfWg= -github.com/rivo/uniseg v0.1.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= -github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= -github.com/rivo/uniseg v0.4.7 h1:WUdvkW8uEhrYfLC4ZzdpI2ztxP1I582+49Oc5Mq64VQ= -github.com/rivo/uniseg v0.4.7/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88= -github.com/robfig/cron/v3 v3.0.1 h1:WdRxkvbJztn8LMz/QEvLN5sBU+xKpSqwwUO1Pjr4qDs= -github.com/robfig/cron/v3 v3.0.1/go.mod h1:eQICP3HwyT7UooqI/z+Ov+PtYAWygg1TEWWzGIFLtro= -github.com/rogpeppe/go-internal v1.10.0 h1:TMyTOH3F/DB16zRVcYyreMH6GnZZrwQVAoYjRBZyWFQ= -github.com/rogpeppe/go-internal v1.10.0/go.mod h1:UQnix2H7Ngw/k4C5ijL5+65zddjncjaFoBhdsK/akog= -github.com/rs/xid v1.5.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg= -github.com/rs/zerolog v1.33.0 h1:1cU2KZkvPxNyfgEmhHAz/1A9Bz+llsdYzklWFzgp0r8= -github.com/rs/zerolog v1.33.0/go.mod h1:/7mN4D5sKwJLZQ2b/znpjC3/GQWY/xaDXUM0kKWRHss= -github.com/savsgio/gotils v0.0.0-20240303185622-093b76447511 h1:KanIMPX0QdEdB4R3CiimCAbxFrhB3j7h0/OvpYGVQa8= -github.com/savsgio/gotils v0.0.0-20240303185622-093b76447511/go.mod h1:sM7Mt7uEoCeFSCBM+qBrqvEo+/9vdmj19wzp3yzUhmg= -github.com/scylladb/termtables v0.0.0-20191203121021-c4c0b6d42ff4/go.mod h1:C1a7PQSMz9NShzorzCiG2fk9+xuCgLkPeCvMHYR2OWg= -github.com/sergi/go-diff v1.4.0 h1:n/SP9D5ad1fORl+llWyN+D6qoUETXNZARKjyY2/KVCw= -github.com/sergi/go-diff v1.4.0/go.mod h1:A0bzQcvG0E7Rwjx0REVgAGH58e96+X0MeOfepqsbeW4= -github.com/sirupsen/logrus v1.9.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ= -github.com/sirupsen/logrus v1.9.3/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ= -github.com/standard-webhooks/standard-webhooks/libraries v0.0.0-20251210175704-b03a68fe8b19 h1:8rMUmsyom6y/10iTAgqkfv8zHVKxVQxFwlOb42V23cA= -github.com/standard-webhooks/standard-webhooks/libraries v0.0.0-20251210175704-b03a68fe8b19/go.mod h1:L1MQhA6x4dn9r007T033lsaZMv9EmBAdXyU/+EF40fo= -github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= -github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= -github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= -github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= -github.com/stretchr/testify v1.11.1 h1:7s2iGBzp5EwR7/aIZr8ao5+dra3wiQyKjjFuvgVKu7U= -github.com/stretchr/testify v1.11.1/go.mod h1:wZwfW3scLgRK+23gO65QZefKpKQRnfz6sD981Nm4B6U= -github.com/temoto/robotstxt v1.1.2 h1:W2pOjSJ6SWvldyEuiFXNxz3xZ8aiWX5LbfDiOFd7Fxg= -github.com/temoto/robotstxt v1.1.2/go.mod h1:+1AmkuG3IYkh1kv0d2qEB9Le88ehNO0zwOr3ujewlOo= -github.com/tetratelabs/wazero v1.8.1 h1:NrcgVbWfkWvVc4UtT4LRLDf91PsOzDzefMdwhLfA550= -github.com/tetratelabs/wazero v1.8.1/go.mod h1:yAI0XTsMBhREkM/YDAK/zNou3GoiAce1P6+rp/wQhjs= -github.com/tidwall/gjson v1.14.2/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk= -github.com/tidwall/gjson v1.14.4 h1:uo0p8EbA09J7RQaflQ1aBRffTR7xedD2bcIVSYxLnkM= -github.com/tidwall/gjson v1.14.4/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk= -github.com/tidwall/match v1.1.1 h1:+Ho715JplO36QYgwN9PGYNhgZvoUSc9X2c80KVTi+GA= -github.com/tidwall/match v1.1.1/go.mod h1:eRSPERbgtNPcGhD8UCthc6PmLEQXEWd3PRB5JTxsfmM= -github.com/tidwall/pretty v1.2.0/go.mod h1:ITEVvHYasfjBbM0u2Pg8T2nJnzm8xPwvNhhsoaGGjNU= -github.com/tidwall/pretty v1.2.1 h1:qjsOFOWWQl+N3RsoF5/ssm1pHmJJwhjlSbZ51I6wMl4= -github.com/tidwall/pretty v1.2.1/go.mod h1:ITEVvHYasfjBbM0u2Pg8T2nJnzm8xPwvNhhsoaGGjNU= -github.com/tidwall/sjson v1.2.5 h1:kLy8mja+1c9jlljvWTlSazM7cKDRfJuR/bOJhcY5NcY= -github.com/tidwall/sjson v1.2.5/go.mod h1:Fvgq9kS/6ociJEDnK0Fk1cpYF4FIW6ZF7LAe+6jwd28= -github.com/tiendc/go-deepcopy v1.7.1 h1:LnubftI6nYaaMOcaz0LphzwraqN8jiWTwm416sitff4= -github.com/tiendc/go-deepcopy v1.7.1/go.mod h1:4bKjNC2r7boYOkD2IOuZpYjmlDdzjbpTRyCx+goBCJQ= -github.com/tinylib/msgp v1.2.5 h1:WeQg1whrXRFiZusidTQqzETkRpGjFjcIhW6uqWH09po= -github.com/tinylib/msgp v1.2.5/go.mod h1:ykjzy2wzgrlvpDCRc4LA8UXy6D8bzMSuAF3WD57Gok0= -github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw= -github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc= -github.com/valyala/fasthttp v1.65.0 h1:j/u3uzFEGFfRxw79iYzJN+TteTJwbYkru9uDp3d0Yf8= -github.com/valyala/fasthttp v1.65.0/go.mod h1:P/93/YkKPMsKSnATEeELUCkG8a7Y+k99uxNHVbKINr4= -github.com/wasilibs/go-re2 v1.7.0 h1:bYhl8gn+a9h01dxwotNycxkiFPTiSgwUrIz8KZJ90Lc= -github.com/wasilibs/go-re2 v1.7.0/go.mod h1:sUsZMLflgl+LNivDE229omtmvjICmOseT9xOy199VDU= -github.com/wasilibs/nottinygc v0.4.0 h1:h1TJMihMC4neN6Zq+WKpLxgd9xCFMw7O9ETLwY2exJQ= -github.com/wasilibs/nottinygc v0.4.0/go.mod h1:oDcIotskuYNMpqMF23l7Z8uzD4TC0WXHK8jetlB3HIo= -github.com/wasilibs/wazero-helpers v0.0.0-20240620070341-3dff1577cd52 h1:OvLBa8SqJnZ6P+mjlzc2K7PM22rRUPE1x32G9DTPrC4= -github.com/wasilibs/wazero-helpers v0.0.0-20240620070341-3dff1577cd52/go.mod h1:jMeV4Vpbi8osrE/pKUxRZkVaA0EX7NZN0A9/oRzgpgY= -github.com/wk8/go-ordered-map/v2 v2.1.8 h1:5h/BUHu93oj4gIdvHHHGsScSTMijfx5PeYkE/fJgbpc= -github.com/wk8/go-ordered-map/v2 v2.1.8/go.mod h1:5nJHM5DyteebpVlHnWMV0rPz6Zp7+xBAnxjb1X5vnTw= -github.com/xdg-go/pbkdf2 v1.0.0 h1:Su7DPu48wXMwC3bs7MCNG+z4FhcyEuz5dlvchbq0B0c= -github.com/xdg-go/pbkdf2 v1.0.0/go.mod h1:jrpuAogTd400dnrH08LKmI/xc1MbPOebTwRqcT5RDeI= -github.com/xdg-go/scram v1.1.2 h1:FHX5I5B4i4hKRVRBCFRxq1iQRej7WO3hhBuJf+UUySY= -github.com/xdg-go/scram v1.1.2/go.mod h1:RT/sEzTbU5y00aCK8UOx6R7YryM0iF1N2MOmC3kKLN4= -github.com/xdg-go/stringprep v1.0.4 h1:XLI/Ng3O1Atzq0oBs3TWm+5ZVgkq2aqdlvP9JtoZ6c8= -github.com/xdg-go/stringprep v1.0.4/go.mod h1:mPGuuIYwz7CmR2bT9j4GbQqutWS1zV24gijq1dTyGkM= -github.com/xuri/efp v0.0.1 h1:fws5Rv3myXyYni8uwj2qKjVaRP30PdjeYe2Y6FDsCL8= -github.com/xuri/efp v0.0.1/go.mod h1:ybY/Jr0T0GTCnYjKqmdwxyxn2BQf2RcQIIvex5QldPI= -github.com/xuri/excelize/v2 v2.10.0 h1:8aKsP7JD39iKLc6dH5Tw3dgV3sPRh8uRVXu/fMstfW4= -github.com/xuri/excelize/v2 v2.10.0/go.mod h1:SC5TzhQkaOsTWpANfm+7bJCldzcnU/jrhqkTi/iBHBU= -github.com/xuri/nfp v0.0.2-0.20250530014748-2ddeb826f9a9 h1:+C0TIdyyYmzadGaL/HBLbf3WdLgC29pgyhTjAT/0nuE= -github.com/xuri/nfp v0.0.2-0.20250530014748-2ddeb826f9a9/go.mod h1:WwHg+CVyzlv/TX9xqBFXEZAuxOPxn2k1GNHwG41IIUQ= -github.com/xyproto/randomstring v1.0.5 h1:YtlWPoRdgMu3NZtP45drfy1GKoojuR7hmRcnhZqKjWU= -github.com/xyproto/randomstring v1.0.5/go.mod h1:rgmS5DeNXLivK7YprL0pY+lTuhNQW3iGxZ18UQApw/E= -github.com/yosssi/gohtml v0.0.0-20201013000340-ee4748c638f4 h1:0sw0nJM544SpsihWx1bkXdYLQDlzRflMgFJQ4Yih9ts= -github.com/yosssi/gohtml v0.0.0-20201013000340-ee4748c638f4/go.mod h1:+ccdNT0xMY1dtc5XBxumbYfOUhmduiGudqaDgD2rVRE= -github.com/youmark/pkcs8 v0.0.0-20240726163527-a2c0da244d78 h1:ilQV1hzziu+LLM3zUTJ0trRztfwgjqKnBWNtSRkbmwM= -github.com/youmark/pkcs8 v0.0.0-20240726163527-a2c0da244d78/go.mod h1:aL8wCCfTfSfmXjznFBSZNN13rSJjlIOI1fUNAtF7rmI= -github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= -github.com/yuin/goldmark v1.7.13 h1:GPddIs617DnBLFFVJFgpo1aBfe/4xcvMc3SB5t/D0pA= -github.com/yuin/goldmark v1.7.13/go.mod h1:ip/1k0VRfGynBgxOz0yCqHrbZXhcjxyuS66Brc7iBKg= -go.mongodb.org/mongo-driver v1.17.1 h1:Wic5cJIwJgSpBhe3lx3+/RybR5PiYRMpVFgO7cOHyIM= -go.mongodb.org/mongo-driver v1.17.1/go.mod h1:wwWm/+BuOddhcq3n68LKRmgk2wXzmF6s0SFOa0GINL4= -go.opentelemetry.io/auto/sdk v1.1.0 h1:cH53jehLUN6UFLY71z+NDOiNJqDdPRaXzTel0sJySYA= -go.opentelemetry.io/auto/sdk v1.1.0/go.mod h1:3wSPjt5PWp2RhlCcmmOial7AvC4DQqZb7a7wCow3W8A= -go.opentelemetry.io/otel v1.37.0 h1:9zhNfelUvx0KBfu/gb+ZgeAfAgtWrfHJZcAqFC228wQ= -go.opentelemetry.io/otel v1.37.0/go.mod h1:ehE/umFRLnuLa/vSccNq9oS1ErUlkkK71gMcN34UG8I= -go.opentelemetry.io/otel/exporters/stdout/stdouttrace v1.37.0 h1:SNhVp/9q4Go/XHBkQ1/d5u9P/U+L1yaGPoi0x+mStaI= -go.opentelemetry.io/otel/exporters/stdout/stdouttrace v1.37.0/go.mod h1:tx8OOlGH6R4kLV67YaYO44GFXloEjGPZuMjEkaaqIp4= -go.opentelemetry.io/otel/metric v1.37.0 h1:mvwbQS5m0tbmqML4NqK+e3aDiO02vsf/WgbsdpcPoZE= -go.opentelemetry.io/otel/metric v1.37.0/go.mod h1:04wGrZurHYKOc+RKeye86GwKiTb9FKm1WHtO+4EVr2E= -go.opentelemetry.io/otel/sdk v1.37.0 h1:ItB0QUqnjesGRvNcmAcU0LyvkVyGJ2xftD29bWdDvKI= -go.opentelemetry.io/otel/sdk v1.37.0/go.mod h1:VredYzxUvuo2q3WRcDnKDjbdvmO0sCzOvVAiY+yUkAg= -go.opentelemetry.io/otel/trace v1.37.0 h1:HLdcFNbRQBE2imdSEgm/kwqmQj1Or1l/7bW6mxVK7z4= -go.opentelemetry.io/otel/trace v1.37.0/go.mod h1:TlgrlQ+PtQO5XFerSPUYG0JSgGyryXewPGyayAWSBS0= -go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto= -go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE= -go.yaml.in/yaml/v2 v2.4.2 h1:DzmwEr2rDGHl7lsFgAHxmNz/1NlQ7xLIrlN2h5d1eGI= -go.yaml.in/yaml/v2 v2.4.2/go.mod h1:081UH+NErpNdqlCXm3TtEran0rJZGxAYx9hb/ELlsPU= -golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= -golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= -golang.org/x/crypto v0.13.0/go.mod h1:y6Z2r+Rw4iayiXXAIxJIDAJ1zMW4yaTpebo8fPOliYc= -golang.org/x/crypto v0.19.0/go.mod h1:Iy9bg/ha4yyC70EfRS8jz+B6ybOBKMaSxLj6P6oBDfU= -golang.org/x/crypto v0.23.0/go.mod h1:CKFgDieR+mRhux2Lsu27y0fO304Db0wZe70UKqHu0v8= -golang.org/x/crypto v0.31.0/go.mod h1:kDsLvtWBEx7MV9tJOj9bnXsPbxwJQ6csT/x4KIN4Ssk= -golang.org/x/crypto v0.45.0 h1:jMBrvKuj23MTlT0bQEOBcAE0mjg8mK9RXFhRH6nyF3Q= -golang.org/x/crypto v0.45.0/go.mod h1:XTGrrkGJve7CYK7J8PEww4aY7gM3qMCElcJQ8n8JdX4= -golang.org/x/crypto v0.47.0 h1:V6e3FRj+n4dbpw86FJ8Fv7XVOql7TEwpHapKoMJ/GO8= -golang.org/x/crypto v0.47.0/go.mod h1:ff3Y9VzzKbwSSEzWqJsJVBnWmRwRSHt/6Op5n9bQc4A= -golang.org/x/exp v0.0.0-20251023183803-a4bb9ffd2546 h1:mgKeJMpvi0yx/sU5GsxQ7p6s2wtOnGAHZWCHUM4KGzY= -golang.org/x/exp v0.0.0-20251023183803-a4bb9ffd2546/go.mod h1:j/pmGrbnkbPtQfxEe5D0VQhZC6qKbfKifgD0oM7sR70= -golang.org/x/image v0.25.0 h1:Y6uW6rH1y5y/LK1J8BPWZtr6yZ7hrsy6hFrXjgsc2fQ= -golang.org/x/image v0.25.0/go.mod h1:tCAmOEGthTtkalusGp1g3xa2gke8J6c2N565dTyl9Rs= -golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= -golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= -golang.org/x/mod v0.12.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= -golang.org/x/mod v0.15.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c= -golang.org/x/mod v0.17.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c= -golang.org/x/mod v0.29.0 h1:HV8lRxZC4l2cr3Zq1LvtOsi/ThTgWnUk/y64QSs8GwA= -golang.org/x/mod v0.29.0/go.mod h1:NyhrlYXJ2H4eJiRy/WDBO6HMqZQ6q9nk4JzS3NuCK+w= -golang.org/x/mod v0.31.0 h1:HaW9xtz0+kOcWKwli0ZXy79Ix+UW/vOfmWI5QVd2tgI= -golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= -golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= -golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= -golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg= -golang.org/x/net v0.15.0/go.mod h1:idbUs1IY1+zTqbi8yxTbhexhEEk5ur9LInksu6HrEpk= -golang.org/x/net v0.21.0/go.mod h1:bIjVDfnllIU7BJ2DNgfnXvpSvtn8VRwhlsaeUTyUS44= -golang.org/x/net v0.25.0/go.mod h1:JkAGAh7GEvH74S6FOH42FLoXpXbE/aqXSrIQjXgsiwM= -golang.org/x/net v0.33.0/go.mod h1:HXLR5J+9DxmrqMwG9qjGCxZ+zKXxBru04zlTvWlWuN4= -golang.org/x/net v0.47.0 h1:Mx+4dIFzqraBXUugkia1OOvlD6LemFo1ALMHjrXDOhY= -golang.org/x/net v0.47.0/go.mod h1:/jNxtkgq5yWUGYkaZGqo27cfGZ1c5Nen03aYrrKpVRU= -golang.org/x/net v0.48.0 h1:zyQRTTrjc33Lhh0fBgT/H3oZq9WuvRR5gPC70xpDiQU= -golang.org/x/net v0.48.0/go.mod h1:+ndRgGjkh8FGtu1w1FGbEC31if4VrNVMuKTgcAAnQRY= -golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.3.0/go.mod h1:FU7BRWz2tNW+3quACPkgCx/L+uEAv1htQ0V83Z9Rj+Y= -golang.org/x/sync v0.6.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= -golang.org/x/sync v0.7.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= -golang.org/x/sync v0.10.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= -golang.org/x/sync v0.18.0 h1:kr88TuHDroi+UVf+0hZnirlk8o8T+4MrK6mr60WkH/I= -golang.org/x/sync v0.18.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI= -golang.org/x/sync v0.19.0 h1:vV+1eWNmZ5geRlYjzm2adRgW2/mcpevXNg50YZtPCE4= -golang.org/x/sync v0.19.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI= -golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= -golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.17.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= -golang.org/x/sys v0.20.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= -golang.org/x/sys v0.28.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= -golang.org/x/sys v0.38.0 h1:3yZWxaJjBmCWXqhN1qh02AkOnCQ1poK6oF+a7xWL6Gc= -golang.org/x/sys v0.38.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks= -golang.org/x/sys v0.40.0 h1:DBZZqJ2Rkml6QMQsZywtnjnnGvHza6BTfYFWY9kjEWQ= -golang.org/x/sys v0.40.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks= -golang.org/x/telemetry v0.0.0-20240228155512-f48c80bd79b2/go.mod h1:TeRTkGYfJXctD9OcfyVLyj2J3IxLnKwHJR8f4D8a3YE= -golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= -golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= -golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= -golang.org/x/term v0.8.0/go.mod h1:xPskH00ivmX89bAKVGSKKtLOWNx2+17Eiy94tnKShWo= -golang.org/x/term v0.12.0/go.mod h1:owVbMEjm3cBLCHdkQu9b1opXd4ETQWc3BhuQGKgXgvU= -golang.org/x/term v0.17.0/go.mod h1:lLRBjIVuehSbZlaOtGMbcMncT+aqLLLmKrsjNrUguwk= -golang.org/x/term v0.20.0/go.mod h1:8UkIAJTvZgivsXaD6/pH6U9ecQzZ45awqEOzuCvwpFY= -golang.org/x/term v0.27.0/go.mod h1:iMsnZpn0cago0GOrHO2+Y7u7JPn5AylBrcoWkElMTSM= -golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= -golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= -golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= -golang.org/x/text v0.3.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ= -golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= -golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= -golang.org/x/text v0.13.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= -golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU= -golang.org/x/text v0.15.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU= -golang.org/x/text v0.21.0/go.mod h1:4IBbMaMmOPCJ8SecivzSH54+73PCFmPWxNTLm+vZkEQ= -golang.org/x/text v0.31.0 h1:aC8ghyu4JhP8VojJ2lEHBnochRno1sgL6nEi9WGFGMM= -golang.org/x/text v0.31.0/go.mod h1:tKRAlv61yKIjGGHX/4tP1LTbc13YSec1pxVEWXzfoeM= -golang.org/x/text v0.33.0 h1:B3njUFyqtHDUI5jMn1YIr5B0IE2U0qck04r6d4KPAxE= -golang.org/x/text v0.33.0/go.mod h1:LuMebE6+rBincTi9+xWTY8TztLzKHc/9C1uBCG27+q8= -golang.org/x/time v0.14.0 h1:MRx4UaLrDotUKUdCIqzPC48t1Y9hANFKIRpNx+Te8PI= -golang.org/x/time v0.14.0/go.mod h1:eL/Oa2bBBK0TkX57Fyni+NgnyQQN4LitPmob2Hjnqw4= -golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= -golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= -golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU= -golang.org/x/tools v0.13.0/go.mod h1:HvlwmtVNQAhOuCjW7xxvovg8wbNq7LwfXh/k7wXUl58= -golang.org/x/tools v0.21.1-0.20240508182429-e35e4ccd0d2d/go.mod h1:aiJjzUbINMkxbQROHiO6hDPo2LHcIPhhQsa9DLh0yGk= -golang.org/x/tools v0.38.0 h1:Hx2Xv8hISq8Lm16jvBZ2VQf+RLmbd7wVUsALibYI/IQ= -golang.org/x/tools v0.38.0/go.mod h1:yEsQ/d/YK8cjh0L6rZlY8tgtlKiBNTL14pGDJPJpYQs= -golang.org/x/tools v0.40.0 h1:yLkxfA+Qnul4cs9QA3KnlFu0lVmd8JJfoq+E41uSutA= -golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= -google.golang.org/protobuf v1.36.8 h1:xHScyCOEuuwZEc6UtSOvPbAT4zRh0xcNRYekJwfqyMc= -google.golang.org/protobuf v1.36.8/go.mod h1:fuxRtAxBytpl4zzqUh6/eyUujkJdNiuEkXntxiD/uRU= -gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= -gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= -gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= -gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= -gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= -gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= -gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= -gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= -gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= -modernc.org/cc/v4 v4.26.5 h1:xM3bX7Mve6G8K8b+T11ReenJOT+BmVqQj0FY5T4+5Y4= -modernc.org/cc/v4 v4.26.5/go.mod h1:uVtb5OGqUKpoLWhqwNQo/8LwvoiEBLvZXIQ/SmO6mL0= -modernc.org/ccgo/v4 v4.28.1 h1:wPKYn5EC/mYTqBO373jKjvX2n+3+aK7+sICCv4Fjy1A= -modernc.org/ccgo/v4 v4.28.1/go.mod h1:uD+4RnfrVgE6ec9NGguUNdhqzNIeeomeXf6CL0GTE5Q= -modernc.org/fileutil v1.3.40 h1:ZGMswMNc9JOCrcrakF1HrvmergNLAmxOPjizirpfqBA= -modernc.org/fileutil v1.3.40/go.mod h1:HxmghZSZVAz/LXcMNwZPA/DRrQZEVP9VX0V4LQGQFOc= -modernc.org/gc/v2 v2.6.5 h1:nyqdV8q46KvTpZlsw66kWqwXRHdjIlJOhG6kxiV/9xI= -modernc.org/gc/v2 v2.6.5/go.mod h1:YgIahr1ypgfe7chRuJi2gD7DBQiKSLMPgBQe9oIiito= -modernc.org/goabi0 v0.2.0 h1:HvEowk7LxcPd0eq6mVOAEMai46V+i7Jrj13t4AzuNks= -modernc.org/goabi0 v0.2.0/go.mod h1:CEFRnnJhKvWT1c1JTI3Avm+tgOWbkOu5oPA8eH8LnMI= -modernc.org/libc v1.66.10 h1:yZkb3YeLx4oynyR+iUsXsybsX4Ubx7MQlSYEw4yj59A= -modernc.org/libc v1.66.10/go.mod h1:8vGSEwvoUoltr4dlywvHqjtAqHBaw0j1jI7iFBTAr2I= -modernc.org/mathutil v1.7.1 h1:GCZVGXdaN8gTqB1Mf/usp1Y/hSqgI2vAGGP4jZMCxOU= -modernc.org/mathutil v1.7.1/go.mod h1:4p5IwJITfppl0G4sUEDtCr4DthTaT47/N3aT6MhfgJg= -modernc.org/memory v1.11.0 h1:o4QC8aMQzmcwCK3t3Ux/ZHmwFPzE6hf2Y5LbkRs+hbI= -modernc.org/memory v1.11.0/go.mod h1:/JP4VbVC+K5sU2wZi9bHoq2MAkCnrt2r98UGeSK7Mjw= -modernc.org/opt v0.1.4 h1:2kNGMRiUjrp4LcaPuLY2PzUfqM/w9N23quVwhKt5Qm8= -modernc.org/opt v0.1.4/go.mod h1:03fq9lsNfvkYSfxrfUhZCWPk1lm4cq4N+Bh//bEtgns= -modernc.org/sortutil v1.2.1 h1:+xyoGf15mM3NMlPDnFqrteY07klSFxLElE2PVuWIJ7w= -modernc.org/sortutil v1.2.1/go.mod h1:7ZI3a3REbai7gzCLcotuw9AC4VZVpYMjDzETGsSMqJE= -modernc.org/sqlite v1.40.1 h1:VfuXcxcUWWKRBuP8+BR9L7VnmusMgBNNnBYGEe9w/iY= -modernc.org/sqlite v1.40.1/go.mod h1:9fjQZ0mB1LLP0GYrp39oOJXx/I2sxEnZtzCmEQIKvGE= -modernc.org/strutil v1.2.1 h1:UneZBkQA+DX2Rp35KcM69cSsNES9ly8mQWD71HKlOA0= -modernc.org/strutil v1.2.1/go.mod h1:EHkiggD70koQxjVdSBM3JKM7k6L0FbGE5eymy9i3B9A= -modernc.org/token v1.1.0 h1:Xl7Ap9dKaEs5kLoOQeQmPWevfnk/DM5qcLcYlA8ys6Y= -modernc.org/token v1.1.0/go.mod h1:UGzOrNV1mAFSEB63lOFHIpNRUVMvYTc6yu1SMY/XTDM= diff --git a/backend/internal/audio/service.go b/backend/internal/audio/service.go deleted file mode 100644 index 30d0b045..00000000 --- a/backend/internal/audio/service.go +++ /dev/null @@ -1,277 +0,0 @@ -package audio - -import ( - "bytes" - "encoding/json" - "fmt" - "io" - "log" - "mime/multipart" - "net/http" - "os" - "path/filepath" - "sync" - "time" -) - -// Provider represents a minimal provider interface for audio -type Provider struct { - ID int - Name string - BaseURL string - APIKey string - Enabled bool -} - -// ProviderGetter is a function type to get a provider -type ProviderGetter func() (*Provider, error) - -// Service handles audio transcription using Whisper API (Groq or OpenAI) -type Service struct { - httpClient *http.Client - groqProviderGetter ProviderGetter - openaiProviderGetter ProviderGetter - mu sync.RWMutex -} - -var ( - instance *Service - once sync.Once -) - -// GetService returns the singleton audio service -func GetService() *Service { - return instance -} - -// InitService initializes the audio service with dependencies -// Priority: Groq (cheaper) -> OpenAI (fallback) -func InitService(groqProviderGetter, openaiProviderGetter ProviderGetter) *Service { - once.Do(func() { - instance = &Service{ - httpClient: &http.Client{ - Timeout: 120 * time.Second, // Whisper can take a while for long audio - }, - groqProviderGetter: groqProviderGetter, - openaiProviderGetter: openaiProviderGetter, - } - }) - return instance -} - -// TranscribeRequest contains parameters for audio transcription -type TranscribeRequest struct { - AudioPath string - Language string // Optional language code (e.g., "en", "es", "fr") - Prompt string // Optional prompt to guide transcription - TranslateToEnglish bool // If true, translates non-English audio to English -} - -// TranscribeResponse contains the result of transcription -type TranscribeResponse struct { - Text string `json:"text"` - Language string `json:"language,omitempty"` - Duration float64 `json:"duration,omitempty"` - Provider string `json:"provider,omitempty"` // Which provider was used -} - -// Transcribe transcribes audio to text using Whisper API -// Tries Groq first (cheaper), falls back to OpenAI -// If TranslateToEnglish is true, uses the translation endpoint to output English -func (s *Service) Transcribe(req *TranscribeRequest) (*TranscribeResponse, error) { - s.mu.RLock() - defer s.mu.RUnlock() - - action := "Transcribing" - if req.TranslateToEnglish { - action = "Translating to English" - } - log.Printf("🎵 [AUDIO] %s audio: %s", action, req.AudioPath) - - // Try Groq first (much cheaper: $0.04/hour vs OpenAI $0.36/hour) - // Note: Groq supports transcription but translation support may be limited - if s.groqProviderGetter != nil && !req.TranslateToEnglish { - provider, err := s.groqProviderGetter() - if err == nil && provider != nil && provider.APIKey != "" { - log.Printf("🚀 [AUDIO] Using Groq Whisper (whisper-large-v3)") - resp, err := s.transcribeWithGroq(req, provider) - if err == nil { - return resp, nil - } - log.Printf("⚠️ [AUDIO] Groq transcription failed, trying OpenAI: %v", err) - } - } - - // Use OpenAI for translation or as fallback for transcription - if s.openaiProviderGetter != nil { - provider, err := s.openaiProviderGetter() - if err == nil && provider != nil && provider.APIKey != "" { - if req.TranslateToEnglish { - log.Printf("🌐 [AUDIO] Using OpenAI Whisper Translation (whisper-1)") - return s.translateWithOpenAI(req, provider) - } - log.Printf("🔄 [AUDIO] Using OpenAI Whisper (whisper-1)") - return s.transcribeWithOpenAI(req, provider) - } - } - - return nil, fmt.Errorf("no audio provider configured. Please add Groq or OpenAI API key") -} - -// transcribeWithGroq uses Groq's Whisper API (whisper-large-v3) -func (s *Service) transcribeWithGroq(req *TranscribeRequest, provider *Provider) (*TranscribeResponse, error) { - return s.transcribeWithProvider(req, provider, "https://api.groq.com/openai/v1/audio/transcriptions", "whisper-large-v3", "Groq") -} - -// transcribeWithOpenAI uses OpenAI's Whisper API (whisper-1) -func (s *Service) transcribeWithOpenAI(req *TranscribeRequest, provider *Provider) (*TranscribeResponse, error) { - return s.transcribeWithProvider(req, provider, "https://api.openai.com/v1/audio/transcriptions", "whisper-1", "OpenAI") -} - -// translateWithOpenAI uses OpenAI's Whisper Translation API to translate audio to English -func (s *Service) translateWithOpenAI(req *TranscribeRequest, provider *Provider) (*TranscribeResponse, error) { - return s.transcribeWithProvider(req, provider, "https://api.openai.com/v1/audio/translations", "whisper-1", "OpenAI-Translation") -} - -// transcribeWithProvider is the common transcription logic for any Whisper-compatible API -func (s *Service) transcribeWithProvider(req *TranscribeRequest, provider *Provider, apiURL, model, providerName string) (*TranscribeResponse, error) { - // Open audio file - audioFile, err := os.Open(req.AudioPath) - if err != nil { - return nil, fmt.Errorf("failed to open audio file: %w", err) - } - defer audioFile.Close() - - // Get file info - fileInfo, err := audioFile.Stat() - if err != nil { - return nil, fmt.Errorf("failed to stat audio file: %w", err) - } - - log.Printf("🔄 [AUDIO] Sending audio to %s Whisper API (%d bytes, model: %s)", providerName, fileInfo.Size(), model) - - // Create multipart form - body := &bytes.Buffer{} - writer := multipart.NewWriter(body) - - // Add file field - filename := filepath.Base(req.AudioPath) - part, err := writer.CreateFormFile("file", filename) - if err != nil { - return nil, fmt.Errorf("failed to create form file: %w", err) - } - - if _, err := io.Copy(part, audioFile); err != nil { - return nil, fmt.Errorf("failed to copy audio data: %w", err) - } - - // Add model field - if err := writer.WriteField("model", model); err != nil { - return nil, fmt.Errorf("failed to write model field: %w", err) - } - - // Add optional language - if req.Language != "" { - if err := writer.WriteField("language", req.Language); err != nil { - return nil, fmt.Errorf("failed to write language field: %w", err) - } - } - - // Add optional prompt - if req.Prompt != "" { - if err := writer.WriteField("prompt", req.Prompt); err != nil { - return nil, fmt.Errorf("failed to write prompt field: %w", err) - } - } - - // Add response format - if err := writer.WriteField("response_format", "verbose_json"); err != nil { - return nil, fmt.Errorf("failed to write response_format field: %w", err) - } - - if err := writer.Close(); err != nil { - return nil, fmt.Errorf("failed to close multipart writer: %w", err) - } - - // Create request - httpReq, err := http.NewRequest("POST", apiURL, body) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - httpReq.Header.Set("Content-Type", writer.FormDataContentType()) - httpReq.Header.Set("Authorization", fmt.Sprintf("Bearer %s", provider.APIKey)) - - // Make request - resp, err := s.httpClient.Do(httpReq) - if err != nil { - return nil, fmt.Errorf("API request failed: %w", err) - } - defer resp.Body.Close() - - respBody, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - if resp.StatusCode != http.StatusOK { - log.Printf("❌ [AUDIO] %s Whisper API error: %d - %s", providerName, resp.StatusCode, string(respBody)) - - // Try to parse error message - var errorResp struct { - Error struct { - Message string `json:"message"` - Type string `json:"type"` - } `json:"error"` - } - if err := json.Unmarshal(respBody, &errorResp); err == nil && errorResp.Error.Message != "" { - return nil, fmt.Errorf("%s Whisper API error: %s", providerName, errorResp.Error.Message) - } - - return nil, fmt.Errorf("%s Whisper API error: %d", providerName, resp.StatusCode) - } - - // Parse response - var apiResp struct { - Text string `json:"text"` - Language string `json:"language"` - Duration float64 `json:"duration"` - } - - if err := json.Unmarshal(respBody, &apiResp); err != nil { - return nil, fmt.Errorf("failed to parse response: %w", err) - } - - log.Printf("✅ [AUDIO] %s transcription successful (%d chars, %.1fs duration)", providerName, len(apiResp.Text), apiResp.Duration) - - return &TranscribeResponse{ - Text: apiResp.Text, - Language: apiResp.Language, - Duration: apiResp.Duration, - Provider: providerName, - }, nil -} - -// GetSupportedFormats returns the list of supported audio formats -func GetSupportedFormats() []string { - return []string{ - "mp3", "mp4", "mpeg", "mpga", "m4a", "wav", "webm", "ogg", "flac", - } -} - -// IsSupportedFormat checks if a MIME type is supported for transcription -func IsSupportedFormat(mimeType string) bool { - supportedTypes := map[string]bool{ - "audio/mpeg": true, - "audio/mp3": true, - "audio/mp4": true, - "audio/x-m4a": true, - "audio/wav": true, - "audio/x-wav": true, - "audio/wave": true, - "audio/webm": true, - "audio/ogg": true, - "audio/flac": true, - } - return supportedTypes[mimeType] -} diff --git a/backend/internal/audio/service_test.go b/backend/internal/audio/service_test.go deleted file mode 100644 index 7a5c6956..00000000 --- a/backend/internal/audio/service_test.go +++ /dev/null @@ -1,153 +0,0 @@ -package audio - -import ( - "testing" -) - -// TestSupportedFormats verifies all expected audio formats are supported -func TestSupportedFormats(t *testing.T) { - supportedMimeTypes := []string{ - "audio/mpeg", - "audio/mp3", - "audio/wav", - "audio/x-wav", - "audio/wave", - "audio/mp4", - "audio/x-m4a", - "audio/webm", - "audio/ogg", - "audio/flac", - } - - for _, mimeType := range supportedMimeTypes { - if !IsSupportedFormat(mimeType) { - t.Errorf("MIME type %s should be supported", mimeType) - } - } -} - -// TestUnsupportedFormats verifies unsupported formats are rejected -func TestUnsupportedFormats(t *testing.T) { - unsupportedMimeTypes := []string{ - "video/mp4", - "image/jpeg", - "application/pdf", - "text/plain", - "audio/midi", - "audio/aiff", - } - - for _, mimeType := range unsupportedMimeTypes { - if IsSupportedFormat(mimeType) { - t.Errorf("MIME type %s should NOT be supported", mimeType) - } - } -} - -// TestGetSupportedFormats verifies the list of supported formats -func TestGetSupportedFormats(t *testing.T) { - formats := GetSupportedFormats() - - if len(formats) == 0 { - t.Error("GetSupportedFormats should return non-empty list") - } - - // Check some expected formats are in the list (file extensions, not MIME types) - expectedFormats := map[string]bool{ - "mp3": false, - "wav": false, - "mp4": false, - "ogg": false, - "webm": false, - "flac": false, - } - - for _, format := range formats { - if _, ok := expectedFormats[format]; ok { - expectedFormats[format] = true - } - } - - for format, found := range expectedFormats { - if !found { - t.Errorf("Expected format %s in supported formats list", format) - } - } -} - -// TestTranscribeRequestValidation tests request validation -func TestTranscribeRequestValidation(t *testing.T) { - // Service requires initialization, so we test the request structure - req := &TranscribeRequest{ - AudioPath: "/path/to/audio.mp3", - Language: "en", - Prompt: "Test transcription", - } - - if req.AudioPath == "" { - t.Error("AudioPath should be set") - } - if req.Language != "en" { - t.Errorf("Language should be 'en', got %s", req.Language) - } -} - -// TestTranscribeResponseStructure tests response structure -func TestTranscribeResponseStructure(t *testing.T) { - resp := &TranscribeResponse{ - Text: "Hello world", - Language: "en", - Duration: 5.5, - } - - if resp.Text == "" { - t.Error("Text should not be empty") - } - if resp.Duration <= 0 { - t.Error("Duration should be positive") - } -} - -// TestProviderStructure tests provider structure -func TestProviderStructure(t *testing.T) { - provider := &Provider{ - ID: 1, - Name: "openai", - BaseURL: "https://api.openai.com/v1", - APIKey: "test-key", - Enabled: true, - } - - if provider.Name != "openai" { - t.Errorf("Expected provider name 'openai', got %s", provider.Name) - } - if !provider.Enabled { - t.Error("Provider should be enabled") - } -} - -// TestGetServiceSingleton verifies singleton pattern -func TestGetServiceSingleton(t *testing.T) { - // GetService should return nil if not initialized - svc := GetService() - // Note: This may return non-nil if InitService was called elsewhere - // The test mainly verifies no panic occurs - _ = svc -} - -// Benchmark tests -func BenchmarkIsSupportedFormat(b *testing.B) { - testCases := []string{ - "audio/mpeg", - "audio/wav", - "video/mp4", - "image/jpeg", - } - - b.ResetTimer() - for i := 0; i < b.N; i++ { - for _, tc := range testCases { - IsSupportedFormat(tc) - } - } -} diff --git a/backend/internal/config/config.go b/backend/internal/config/config.go deleted file mode 100644 index 82f2a8e9..00000000 --- a/backend/internal/config/config.go +++ /dev/null @@ -1,127 +0,0 @@ -package config - -import ( - "encoding/json" - "fmt" - "os" - "strconv" - "strings" - "time" - - "claraverse/internal/models" -) - -// Config holds all application configuration -type Config struct { - Port string - DatabaseURL string // MySQL DSN: mysql://user:pass@host:port/dbname?parseTime=true - SupabaseURL string - SupabaseKey string - SearXNGURL string - RedisURL string - - // DodoPayments configuration - DodoAPIKey string - DodoWebhookSecret string - DodoBusinessID string - DodoEnvironment string // "live" or "test" - - // Promotional campaign configuration - PromoEnabled bool - PromoStartDate time.Time - PromoEndDate time.Time - PromoDuration int // days - - // Superadmin configuration - SuperadminUserIDs []string // List of Supabase user IDs with superadmin access -} - -// Load loads configuration from environment variables with defaults -func Load() *Config { - // Parse superadmin user IDs (comma-separated) - superadminEnv := getEnv("SUPERADMIN_USER_IDS", "") - var superadminUserIDs []string - if superadminEnv != "" { - superadminUserIDs = strings.Split(superadminEnv, ",") - // Trim whitespace from each ID - for i := range superadminUserIDs { - superadminUserIDs[i] = strings.TrimSpace(superadminUserIDs[i]) - } - } - - return &Config{ - Port: getEnv("PORT", "3001"), - DatabaseURL: getEnv("DATABASE_URL", ""), - SupabaseURL: getEnv("SUPABASE_URL", ""), - SupabaseKey: getEnv("SUPABASE_KEY", ""), - SearXNGURL: getEnv("SEARXNG_URL", "http://localhost:8080"), - RedisURL: getEnv("REDIS_URL", "redis://localhost:6379"), - - // DodoPayments configuration - DodoAPIKey: getEnv("DODO_API_KEY", ""), - DodoWebhookSecret: getEnv("DODO_WEBHOOK_SECRET", ""), - DodoBusinessID: getEnv("DODO_BUSINESS_ID", ""), - DodoEnvironment: getEnv("DODO_ENVIRONMENT", "test"), - - // Promotional campaign configuration - PromoEnabled: getBoolEnv("PROMO_ENABLED", true), - PromoStartDate: getTimeEnv("PROMO_START_DATE", "2026-01-01T00:00:00Z"), - PromoEndDate: getTimeEnv("PROMO_END_DATE", "2026-02-01T00:00:00Z"), - PromoDuration: getIntEnv("PROMO_DURATION_DAYS", 30), - - // Superadmin configuration - SuperadminUserIDs: superadminUserIDs, - } -} - -// LoadProviders loads providers configuration from JSON file -func LoadProviders(filePath string) (*models.ProvidersConfig, error) { - data, err := os.ReadFile(filePath) - if err != nil { - return nil, fmt.Errorf("failed to read providers file: %w", err) - } - - var config models.ProvidersConfig - if err := json.Unmarshal(data, &config); err != nil { - return nil, fmt.Errorf("failed to parse providers JSON: %w", err) - } - - return &config, nil -} - -func getEnv(key, defaultValue string) string { - if value := os.Getenv(key); value != "" { - return value - } - return defaultValue -} - -func getBoolEnv(key string, defaultValue bool) bool { - if value := os.Getenv(key); value != "" { - parsed, err := strconv.ParseBool(value) - if err == nil { - return parsed - } - } - return defaultValue -} - -func getIntEnv(key string, defaultValue int) int { - if value := os.Getenv(key); value != "" { - parsed, err := strconv.Atoi(value) - if err == nil { - return parsed - } - } - return defaultValue -} - -func getTimeEnv(key string, defaultValue string) time.Time { - value := getEnv(key, defaultValue) - parsed, err := time.Parse(time.RFC3339, value) - if err != nil { - // If parsing fails, return zero time - return time.Time{} - } - return parsed -} diff --git a/backend/internal/crypto/encryption.go b/backend/internal/crypto/encryption.go deleted file mode 100644 index e25a6f65..00000000 --- a/backend/internal/crypto/encryption.go +++ /dev/null @@ -1,177 +0,0 @@ -package crypto - -import ( - "crypto/aes" - "crypto/cipher" - "crypto/rand" - "encoding/base64" - "encoding/hex" - "errors" - "fmt" - "io" - - "golang.org/x/crypto/hkdf" - "crypto/sha256" -) - -// EncryptionService handles encryption/decryption of user data -type EncryptionService struct { - masterKey []byte -} - -// NewEncryptionService creates a new encryption service with the given master key -// masterKey should be a 32-byte hex-encoded string (64 characters) -func NewEncryptionService(masterKeyHex string) (*EncryptionService, error) { - if masterKeyHex == "" { - return nil, errors.New("encryption master key is required") - } - - masterKey, err := hex.DecodeString(masterKeyHex) - if err != nil { - return nil, fmt.Errorf("invalid master key format (must be hex): %w", err) - } - - if len(masterKey) != 32 { - return nil, fmt.Errorf("master key must be 32 bytes (64 hex characters), got %d bytes", len(masterKey)) - } - - return &EncryptionService{ - masterKey: masterKey, - }, nil -} - -// DeriveUserKey derives a unique encryption key for a specific user -// using HKDF (HMAC-based Key Derivation Function) -func (e *EncryptionService) DeriveUserKey(userID string) ([]byte, error) { - if userID == "" { - return nil, errors.New("user ID is required for key derivation") - } - - // Use HKDF to derive a user-specific key - hkdfReader := hkdf.New(sha256.New, e.masterKey, []byte(userID), []byte("claraverse-user-encryption")) - - userKey := make([]byte, 32) // AES-256 requires 32-byte key - if _, err := io.ReadFull(hkdfReader, userKey); err != nil { - return nil, fmt.Errorf("failed to derive user key: %w", err) - } - - return userKey, nil -} - -// Encrypt encrypts plaintext using AES-256-GCM with a user-specific key -// Returns base64-encoded ciphertext (nonce prepended) -func (e *EncryptionService) Encrypt(userID string, plaintext []byte) (string, error) { - if len(plaintext) == 0 { - return "", nil // Return empty string for empty input - } - - // Derive user-specific key - userKey, err := e.DeriveUserKey(userID) - if err != nil { - return "", err - } - - // Create AES cipher - block, err := aes.NewCipher(userKey) - if err != nil { - return "", fmt.Errorf("failed to create cipher: %w", err) - } - - // Create GCM mode - gcm, err := cipher.NewGCM(block) - if err != nil { - return "", fmt.Errorf("failed to create GCM: %w", err) - } - - // Generate random nonce - nonce := make([]byte, gcm.NonceSize()) - if _, err := io.ReadFull(rand.Reader, nonce); err != nil { - return "", fmt.Errorf("failed to generate nonce: %w", err) - } - - // Encrypt and prepend nonce - ciphertext := gcm.Seal(nonce, nonce, plaintext, nil) - - // Return as base64 - return base64.StdEncoding.EncodeToString(ciphertext), nil -} - -// Decrypt decrypts base64-encoded ciphertext using AES-256-GCM -func (e *EncryptionService) Decrypt(userID string, ciphertextB64 string) ([]byte, error) { - if ciphertextB64 == "" { - return nil, nil // Return nil for empty input - } - - // Decode base64 - ciphertext, err := base64.StdEncoding.DecodeString(ciphertextB64) - if err != nil { - return nil, fmt.Errorf("failed to decode ciphertext: %w", err) - } - - // Derive user-specific key - userKey, err := e.DeriveUserKey(userID) - if err != nil { - return nil, err - } - - // Create AES cipher - block, err := aes.NewCipher(userKey) - if err != nil { - return nil, fmt.Errorf("failed to create cipher: %w", err) - } - - // Create GCM mode - gcm, err := cipher.NewGCM(block) - if err != nil { - return nil, fmt.Errorf("failed to create GCM: %w", err) - } - - // Extract nonce - nonceSize := gcm.NonceSize() - if len(ciphertext) < nonceSize { - return nil, errors.New("ciphertext too short") - } - - nonce, ciphertext := ciphertext[:nonceSize], ciphertext[nonceSize:] - - // Decrypt - plaintext, err := gcm.Open(nil, nonce, ciphertext, nil) - if err != nil { - return nil, fmt.Errorf("failed to decrypt: %w", err) - } - - return plaintext, nil -} - -// EncryptString is a convenience method for encrypting strings -func (e *EncryptionService) EncryptString(userID string, plaintext string) (string, error) { - return e.Encrypt(userID, []byte(plaintext)) -} - -// DecryptString is a convenience method for decrypting to strings -func (e *EncryptionService) DecryptString(userID string, ciphertext string) (string, error) { - plaintext, err := e.Decrypt(userID, ciphertext) - if err != nil { - return "", err - } - return string(plaintext), nil -} - -// EncryptJSON encrypts a JSON byte slice -func (e *EncryptionService) EncryptJSON(userID string, jsonData []byte) (string, error) { - return e.Encrypt(userID, jsonData) -} - -// DecryptJSON decrypts to a JSON byte slice -func (e *EncryptionService) DecryptJSON(userID string, ciphertext string) ([]byte, error) { - return e.Decrypt(userID, ciphertext) -} - -// GenerateMasterKey generates a new random 32-byte master key (for setup) -func GenerateMasterKey() (string, error) { - key := make([]byte, 32) - if _, err := io.ReadFull(rand.Reader, key); err != nil { - return "", fmt.Errorf("failed to generate key: %w", err) - } - return hex.EncodeToString(key), nil -} diff --git a/backend/internal/database/database.go b/backend/internal/database/database.go deleted file mode 100644 index 6565a1c4..00000000 --- a/backend/internal/database/database.go +++ /dev/null @@ -1,229 +0,0 @@ -package database - -import ( - "database/sql" - "fmt" - "log" - "os" - "strings" - - _ "github.com/go-sql-driver/mysql" -) - -// DB wraps the SQL database connection -type DB struct { - *sql.DB -} - -// New creates a new database connection -// Supports both MySQL DSN (mysql://...) and legacy SQLite path for backwards compatibility -func New(dsn string) (*DB, error) { - var db *sql.DB - var err error - - // Detect database type from DSN - if strings.HasPrefix(dsn, "mysql://") { - // MySQL DSN format: mysql://user:pass@host:port/dbname?parseTime=true - // Convert to Go MySQL driver format: user:pass@tcp(host:port)/dbname?parseTime=true - dsn = strings.TrimPrefix(dsn, "mysql://") - - // Parse the DSN to add tcp() wrapper around host:port - // Format: user:pass@host:port/dbname -> user:pass@tcp(host:port)/dbname - parts := strings.SplitN(dsn, "@", 2) - if len(parts) == 2 { - hostAndRest := parts[1] - // Find the '/' that separates host:port from dbname - slashIdx := strings.Index(hostAndRest, "/") - if slashIdx > 0 { - host := hostAndRest[:slashIdx] - rest := hostAndRest[slashIdx:] - dsn = parts[0] + "@tcp(" + host + ")" + rest - } - } - - db, err = sql.Open("mysql", dsn) - } else { - // Legacy SQLite path (for backwards compatibility during migration) - return nil, fmt.Errorf("SQLite no longer supported - please use DATABASE_URL with MySQL DSN") - } - - if err != nil { - return nil, fmt.Errorf("failed to open database: %w", err) - } - - // Configure connection pool - db.SetMaxOpenConns(25) - db.SetMaxIdleConns(5) - - if err := db.Ping(); err != nil { - return nil, fmt.Errorf("failed to ping database: %w", err) - } - - log.Println("✅ MySQL database connected") - - return &DB{db}, nil -} - -// Initialize creates all required tables -// NOTE: MySQL schema is created via migrations/001_initial_schema.sql on first run -// This function only runs additional migrations for schema evolution -func (db *DB) Initialize() error { - log.Println("🔍 Checking database schema...") - - // Run migrations for existing databases - if err := db.runMigrations(); err != nil { - return fmt.Errorf("failed to run migrations: %w", err) - } - - log.Println("✅ Database initialized successfully") - return nil -} - -// runMigrations runs database migrations for schema updates -// Uses INFORMATION_SCHEMA to check for column existence (MySQL-compatible) -func (db *DB) runMigrations() error { - dbName := os.Getenv("MYSQL_DATABASE") - if dbName == "" { - dbName = "claraverse" // default - } - - // Helper function to check if column exists - columnExists := func(tableName, columnName string) (bool, error) { - var count int - query := ` - SELECT COUNT(*) - FROM INFORMATION_SCHEMA.COLUMNS - WHERE TABLE_SCHEMA = ? AND TABLE_NAME = ? AND COLUMN_NAME = ? - ` - err := db.QueryRow(query, dbName, tableName, columnName).Scan(&count) - if err != nil { - return false, err - } - return count > 0, nil - } - - // Helper function to check if table exists - tableExists := func(tableName string) (bool, error) { - var count int - query := ` - SELECT COUNT(*) - FROM INFORMATION_SCHEMA.TABLES - WHERE TABLE_SCHEMA = ? AND TABLE_NAME = ? - ` - err := db.QueryRow(query, dbName, tableName).Scan(&count) - if err != nil { - return false, err - } - return count > 0, nil - } - - // Migration: Add audio_only column to providers table (if missing) - if exists, _ := tableExists("providers"); exists { - if colExists, _ := columnExists("providers", "audio_only"); !colExists { - log.Println("📦 Running migration: Adding audio_only to providers table") - if _, err := db.Exec("ALTER TABLE providers ADD COLUMN audio_only BOOLEAN DEFAULT FALSE"); err != nil { - return fmt.Errorf("failed to add audio_only to providers: %w", err) - } - log.Println("✅ Migration completed: providers.audio_only added") - } - } - - // Migration: Add image_only column to providers table (if missing) - if exists, _ := tableExists("providers"); exists { - if colExists, _ := columnExists("providers", "image_only"); !colExists { - log.Println("📦 Running migration: Adding image_only to providers table") - if _, err := db.Exec("ALTER TABLE providers ADD COLUMN image_only BOOLEAN DEFAULT FALSE"); err != nil { - return fmt.Errorf("failed to add image_only to providers: %w", err) - } - log.Println("✅ Migration completed: providers.image_only added") - } - } - - // Migration: Add image_edit_only column to providers table (if missing) - if exists, _ := tableExists("providers"); exists { - if colExists, _ := columnExists("providers", "image_edit_only"); !colExists { - log.Println("📦 Running migration: Adding image_edit_only to providers table") - if _, err := db.Exec("ALTER TABLE providers ADD COLUMN image_edit_only BOOLEAN DEFAULT FALSE"); err != nil { - return fmt.Errorf("failed to add image_edit_only to providers: %w", err) - } - log.Println("✅ Migration completed: providers.image_edit_only added") - } - } - - // Migration: Add secure column to providers table (if missing) - if exists, _ := tableExists("providers"); exists { - if colExists, _ := columnExists("providers", "secure"); !colExists { - log.Println("📦 Running migration: Adding secure to providers table") - if _, err := db.Exec("ALTER TABLE providers ADD COLUMN secure BOOLEAN DEFAULT FALSE COMMENT 'Privacy-focused provider'"); err != nil { - return fmt.Errorf("failed to add secure to providers: %w", err) - } - log.Println("✅ Migration completed: providers.secure added") - } - } - - // Migration: Add default_model column to providers table (if missing) - if exists, _ := tableExists("providers"); exists { - if colExists, _ := columnExists("providers", "default_model"); !colExists { - log.Println("📦 Running migration: Adding default_model to providers table") - if _, err := db.Exec("ALTER TABLE providers ADD COLUMN default_model VARCHAR(255)"); err != nil { - return fmt.Errorf("failed to add default_model to providers: %w", err) - } - log.Println("✅ Migration completed: providers.default_model added") - } - } - - // Migration: Add smart_tool_router column to models table (if missing) - if exists, _ := tableExists("models"); exists { - if colExists, _ := columnExists("models", "smart_tool_router"); !colExists { - log.Println("📦 Running migration: Adding smart_tool_router to models table") - if _, err := db.Exec("ALTER TABLE models ADD COLUMN smart_tool_router BOOLEAN DEFAULT FALSE COMMENT 'Can predict tool usage'"); err != nil { - return fmt.Errorf("failed to add smart_tool_router to models: %w", err) - } - log.Println("✅ Migration completed: models.smart_tool_router added") - } - } - - // Migration: Add agents_enabled column to models table (if missing) - if exists, _ := tableExists("models"); exists { - if colExists, _ := columnExists("models", "agents_enabled"); !colExists { - log.Println("📦 Running migration: Adding agents_enabled to models table") - if _, err := db.Exec("ALTER TABLE models ADD COLUMN agents_enabled BOOLEAN DEFAULT FALSE COMMENT 'Available in agent builder'"); err != nil { - return fmt.Errorf("failed to add agents_enabled to models: %w", err) - } - log.Println("✅ Migration completed: models.agents_enabled added") - } - } - - // Migration: Add created_at and updated_at timestamps to models (if missing) - if exists, _ := tableExists("models"); exists { - if colExists, _ := columnExists("models", "created_at"); !colExists { - log.Println("📦 Running migration: Adding created_at to models table") - if _, err := db.Exec("ALTER TABLE models ADD COLUMN created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP"); err != nil { - return fmt.Errorf("failed to add created_at to models: %w", err) - } - log.Println("✅ Migration completed: models.created_at added") - } - - if colExists, _ := columnExists("models", "updated_at"); !colExists { - log.Println("📦 Running migration: Adding updated_at to models table") - if _, err := db.Exec("ALTER TABLE models ADD COLUMN updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP"); err != nil { - return fmt.Errorf("failed to add updated_at to models: %w", err) - } - log.Println("✅ Migration completed: models.updated_at added") - } - } - - // Migration: Add smart_tool_router column to model_aliases table (if missing) - if exists, _ := tableExists("model_aliases"); exists { - if colExists, _ := columnExists("model_aliases", "smart_tool_router"); !colExists { - log.Println("📦 Running migration: Adding smart_tool_router to model_aliases table") - if _, err := db.Exec("ALTER TABLE model_aliases ADD COLUMN smart_tool_router BOOLEAN DEFAULT FALSE"); err != nil { - return fmt.Errorf("failed to add smart_tool_router to model_aliases: %w", err) - } - log.Println("✅ Migration completed: model_aliases.smart_tool_router added") - } - } - - log.Println("✅ All migrations completed") - return nil -} diff --git a/backend/internal/database/mongodb.go b/backend/internal/database/mongodb.go deleted file mode 100644 index c53e432f..00000000 --- a/backend/internal/database/mongodb.go +++ /dev/null @@ -1,264 +0,0 @@ -package database - -import ( - "context" - "fmt" - "log" - "time" - - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/mongo" - "go.mongodb.org/mongo-driver/mongo/options" - "go.mongodb.org/mongo-driver/mongo/readpref" -) - -// MongoDB wraps the MongoDB client and database -type MongoDB struct { - client *mongo.Client - database *mongo.Database - dbName string -} - -// Collection names -const ( - CollectionUsers = "users" - CollectionAgents = "agents" - CollectionWorkflows = "workflows" - CollectionBuilderConversations = "builder_conversations" - CollectionExecutions = "executions" - CollectionProviders = "providers" - CollectionModels = "models" - CollectionMCPConnections = "mcp_connections" - CollectionMCPTools = "mcp_tools" - CollectionMCPAuditLog = "mcp_audit_log" - CollectionCredentials = "credentials" - CollectionChats = "chats" - - // Memory system collections - CollectionMemories = "memories" - CollectionMemoryExtractionJobs = "memory_extraction_jobs" - CollectionConversationEngagement = "conversation_engagement" -) - -// NewMongoDB creates a new MongoDB connection with connection pooling -func NewMongoDB(uri string) (*MongoDB, error) { - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - // Configure client options with connection pooling - clientOptions := options.Client(). - ApplyURI(uri). - SetMaxPoolSize(100). - SetMinPoolSize(10). - SetMaxConnIdleTime(30 * time.Second). - SetServerSelectionTimeout(5 * time.Second). - SetConnectTimeout(10 * time.Second) - - // Connect to MongoDB - client, err := mongo.Connect(ctx, clientOptions) - if err != nil { - return nil, fmt.Errorf("failed to connect to MongoDB: %w", err) - } - - // Ping to verify connection - if err := client.Ping(ctx, readpref.Primary()); err != nil { - return nil, fmt.Errorf("failed to ping MongoDB: %w", err) - } - - // Extract database name from URI or use default - dbName := extractDBName(uri) - if dbName == "" { - dbName = "claraverse" - } - - db := &MongoDB{ - client: client, - database: client.Database(dbName), - dbName: dbName, - } - - log.Printf("✅ Connected to MongoDB database: %s", dbName) - - return db, nil -} - -// extractDBName extracts the database name from MongoDB URI -func extractDBName(uri string) string { - // Simple extraction - works for standard MongoDB URIs - // mongodb://localhost:27017/claraverse -> claraverse - // mongodb+srv://user:pass@cluster/claraverse -> claraverse - opts := options.Client().ApplyURI(uri) - if opts.Auth != nil && opts.Auth.AuthSource != "" { - return opts.Auth.AuthSource - } - // Default fallback - return "claraverse" -} - -// Initialize creates indexes for all collections -func (m *MongoDB) Initialize(ctx context.Context) error { - log.Println("📦 Initializing MongoDB indexes...") - - // Users collection indexes - if err := m.createIndexes(ctx, CollectionUsers, []mongo.IndexModel{ - {Keys: bson.D{{Key: "supabaseUserId", Value: 1}}, Options: options.Index().SetUnique(true)}, - {Keys: bson.D{{Key: "email", Value: 1}}}, - }); err != nil { - return fmt.Errorf("failed to create users indexes: %w", err) - } - - // Agents collection indexes - if err := m.createIndexes(ctx, CollectionAgents, []mongo.IndexModel{ - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "updatedAt", Value: -1}}}, - {Keys: bson.D{{Key: "status", Value: 1}}}, - }); err != nil { - return fmt.Errorf("failed to create agents indexes: %w", err) - } - - // Workflows collection indexes - if err := m.createIndexes(ctx, CollectionWorkflows, []mongo.IndexModel{ - {Keys: bson.D{{Key: "agentId", Value: 1}, {Key: "version", Value: -1}}}, - }); err != nil { - return fmt.Errorf("failed to create workflows indexes: %w", err) - } - - // Builder conversations collection indexes - if err := m.createIndexes(ctx, CollectionBuilderConversations, []mongo.IndexModel{ - {Keys: bson.D{{Key: "agentId", Value: 1}}}, - {Keys: bson.D{{Key: "userId", Value: 1}}}, - {Keys: bson.D{{Key: "expiresAt", Value: 1}}, Options: options.Index().SetExpireAfterSeconds(0)}, - }); err != nil { - return fmt.Errorf("failed to create builder_conversations indexes: %w", err) - } - - // Executions collection indexes - if err := m.createIndexes(ctx, CollectionExecutions, []mongo.IndexModel{ - {Keys: bson.D{{Key: "agentId", Value: 1}, {Key: "startedAt", Value: -1}}}, - {Keys: bson.D{{Key: "status", Value: 1}}}, - }); err != nil { - return fmt.Errorf("failed to create executions indexes: %w", err) - } - - // Providers collection indexes - if err := m.createIndexes(ctx, CollectionProviders, []mongo.IndexModel{ - {Keys: bson.D{{Key: "name", Value: 1}}, Options: options.Index().SetUnique(true)}, - }); err != nil { - return fmt.Errorf("failed to create providers indexes: %w", err) - } - - // Models collection indexes - if err := m.createIndexes(ctx, CollectionModels, []mongo.IndexModel{ - {Keys: bson.D{{Key: "providerId", Value: 1}}}, - {Keys: bson.D{{Key: "isVisible", Value: 1}}}, - }); err != nil { - return fmt.Errorf("failed to create models indexes: %w", err) - } - - // MCP connections indexes - if err := m.createIndexes(ctx, CollectionMCPConnections, []mongo.IndexModel{ - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "isActive", Value: 1}}}, - {Keys: bson.D{{Key: "clientId", Value: 1}}, Options: options.Index().SetUnique(true)}, - }); err != nil { - return fmt.Errorf("failed to create mcp_connections indexes: %w", err) - } - - // MCP tools indexes - if err := m.createIndexes(ctx, CollectionMCPTools, []mongo.IndexModel{ - {Keys: bson.D{{Key: "userId", Value: 1}}}, - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "toolName", Value: 1}}, Options: options.Index().SetUnique(true)}, - }); err != nil { - return fmt.Errorf("failed to create mcp_tools indexes: %w", err) - } - - // MCP audit log indexes - if err := m.createIndexes(ctx, CollectionMCPAuditLog, []mongo.IndexModel{ - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "executedAt", Value: -1}}}, - }); err != nil { - return fmt.Errorf("failed to create mcp_audit_log indexes: %w", err) - } - - // Chats collection indexes (for cloud sync) - if err := m.createIndexes(ctx, CollectionChats, []mongo.IndexModel{ - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "updatedAt", Value: -1}}}, // List user's chats sorted by recent - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "chatId", Value: 1}}, Options: options.Index().SetUnique(true)}, // Unique chat per user - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "isStarred", Value: 1}}}, // Filter starred chats - }); err != nil { - return fmt.Errorf("failed to create chats indexes: %w", err) - } - - // Memories collection indexes - if err := m.createIndexes(ctx, CollectionMemories, []mongo.IndexModel{ - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "isArchived", Value: 1}, {Key: "score", Value: -1}}}, // Get top active memories - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "contentHash", Value: 1}}, Options: options.Index().SetUnique(true)}, // Deduplication - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "category", Value: 1}}}, // Filter by category - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "tags", Value: 1}}}, // Tag-based lookup - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "lastAccessedAt", Value: -1}}}, // Recency tracking - }); err != nil { - return fmt.Errorf("failed to create memories indexes: %w", err) - } - - // Memory extraction jobs collection indexes - if err := m.createIndexes(ctx, CollectionMemoryExtractionJobs, []mongo.IndexModel{ - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "status", Value: 1}}}, // Find pending jobs - {Keys: bson.D{{Key: "createdAt", Value: 1}}, Options: options.Index().SetExpireAfterSeconds(86400)}, // TTL: cleanup after 24h - }); err != nil { - return fmt.Errorf("failed to create memory_extraction_jobs indexes: %w", err) - } - - // Conversation engagement collection indexes - if err := m.createIndexes(ctx, CollectionConversationEngagement, []mongo.IndexModel{ - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "conversationId", Value: 1}}, Options: options.Index().SetUnique(true)}, // Unique per user+conversation - }); err != nil { - return fmt.Errorf("failed to create conversation_engagement indexes: %w", err) - } - - log.Println("✅ MongoDB indexes initialized successfully") - return nil -} - -// createIndexes creates indexes for a collection -func (m *MongoDB) createIndexes(ctx context.Context, collectionName string, indexes []mongo.IndexModel) error { - collection := m.database.Collection(collectionName) - _, err := collection.Indexes().CreateMany(ctx, indexes) - return err -} - -// Collection returns a collection handle -func (m *MongoDB) Collection(name string) *mongo.Collection { - return m.database.Collection(name) -} - -// Client returns the underlying MongoDB client -func (m *MongoDB) Client() *mongo.Client { - return m.client -} - -// Database returns the underlying MongoDB database -func (m *MongoDB) Database() *mongo.Database { - return m.database -} - -// Close closes the MongoDB connection -func (m *MongoDB) Close(ctx context.Context) error { - log.Println("🔌 Closing MongoDB connection...") - return m.client.Disconnect(ctx) -} - -// Ping checks if the database connection is alive -func (m *MongoDB) Ping(ctx context.Context) error { - return m.client.Ping(ctx, readpref.Primary()) -} - -// WithTransaction executes a function within a transaction -func (m *MongoDB) WithTransaction(ctx context.Context, fn func(sessCtx mongo.SessionContext) error) error { - session, err := m.client.StartSession() - if err != nil { - return fmt.Errorf("failed to start session: %w", err) - } - defer session.EndSession(ctx) - - _, err = session.WithTransaction(ctx, func(sessCtx mongo.SessionContext) (interface{}, error) { - return nil, fn(sessCtx) - }) - return err -} diff --git a/backend/internal/document/service.go b/backend/internal/document/service.go deleted file mode 100644 index 52ad4455..00000000 --- a/backend/internal/document/service.go +++ /dev/null @@ -1,475 +0,0 @@ -package document - -import ( - "bytes" - "context" - "fmt" - "log" - "os" - "path/filepath" - "sync" - "time" - - "github.com/chromedp/chromedp" - "github.com/chromedp/cdproto/page" - "github.com/google/uuid" - "github.com/yuin/goldmark" - "github.com/yuin/goldmark/extension" -) - -// GeneratedDocument represents a generated document -type GeneratedDocument struct { - DocumentID string - UserID string - ConversationID string - Filename string - FilePath string - Size int64 - DownloadURL string - ContentType string // MIME type for download (e.g., "application/pdf", "text/plain") - CreatedAt time.Time - Downloaded bool - DownloadedAt *time.Time -} - -// Service handles document generation and management -type Service struct { - outputDir string - documents map[string]*GeneratedDocument - mu sync.RWMutex -} - -var ( - serviceInstance *Service - serviceOnce sync.Once -) - -// GetService returns the singleton document service -func GetService() *Service { - serviceOnce.Do(func() { - outputDir := "./generated" - if err := os.MkdirAll(outputDir, 0700); err != nil { - log.Printf("⚠️ Warning: Could not create generated directory: %v", err) - } - serviceInstance = &Service{ - outputDir: outputDir, - documents: make(map[string]*GeneratedDocument), - } - }) - return serviceInstance -} - -// GenerateDocumentFromHTML creates a PDF document from custom HTML content -func (s *Service) GenerateDocumentFromHTML(htmlContent, filename, title, userID, conversationID string) (*GeneratedDocument, error) { - // Wrap HTML in complete document structure if not already present - fullHTML := htmlContent - - // Check if HTML is a complete document (has ) - hasDoctype := bytes.Contains([]byte(htmlContent), []byte(" - - - - - %s - - - -%s - -`, title, htmlContent) - } - - // Generate unique document ID and filename - documentID := uuid.New().String() - safeFilename := filename + ".pdf" - filePath := filepath.Join(s.outputDir, documentID+".pdf") - - // Convert HTML to PDF using chromedp - if err := s.generatePDF(fullHTML, filePath); err != nil { - return nil, fmt.Errorf("failed to generate PDF: %w", err) - } - - // Get file size - fileInfo, err := os.Stat(filePath) - if err != nil { - return nil, fmt.Errorf("failed to get file info: %w", err) - } - - // Create document record - doc := &GeneratedDocument{ - DocumentID: documentID, - UserID: userID, - ConversationID: conversationID, - Filename: safeFilename, - FilePath: filePath, - Size: fileInfo.Size(), - DownloadURL: fmt.Sprintf("/api/download/%s", documentID), - ContentType: "application/pdf", - CreatedAt: time.Now(), - Downloaded: false, - } - - // Store document - s.mu.Lock() - s.documents[documentID] = doc - s.mu.Unlock() - - log.Printf("📄 [DOCUMENT-SERVICE] Generated custom HTML PDF: %s (%d bytes)", safeFilename, fileInfo.Size()) - - return doc, nil -} - -// GenerateDocument creates a PDF document from markdown content (deprecated - use GenerateDocumentFromHTML) -// Kept for backward compatibility with existing code -func (s *Service) GenerateDocument(content, filename, title, userID, conversationID string) (*GeneratedDocument, error) { - // Convert markdown to HTML with GFM extensions - var htmlBuf bytes.Buffer - md := goldmark.New( - goldmark.WithExtensions( - extension.GFM, // GitHub Flavored Markdown (includes Table, Strikethrough, Linkify, TaskList) - ), - ) - if err := md.Convert([]byte(content), &htmlBuf); err != nil { - return nil, fmt.Errorf("failed to convert markdown: %w", err) - } - - // Wrap in HTML template with basic styling - fullHTML := fmt.Sprintf(` - - - - %s - - - - %s - -`, title, htmlBuf.String()) - - // Generate unique document ID and filename - documentID := uuid.New().String() - safeFilename := filename + ".pdf" - filePath := filepath.Join(s.outputDir, documentID+".pdf") - - // Convert HTML to PDF using chromedp - if err := s.generatePDF(fullHTML, filePath); err != nil { - return nil, fmt.Errorf("failed to generate PDF: %w", err) - } - - // Get file size - fileInfo, err := os.Stat(filePath) - if err != nil { - return nil, fmt.Errorf("failed to get file info: %w", err) - } - - // Create document record - doc := &GeneratedDocument{ - DocumentID: documentID, - UserID: userID, - ConversationID: conversationID, - Filename: safeFilename, - FilePath: filePath, - Size: fileInfo.Size(), - DownloadURL: fmt.Sprintf("/api/download/%s", documentID), - ContentType: "application/pdf", - CreatedAt: time.Now(), - Downloaded: false, - } - - // Store document - s.mu.Lock() - s.documents[documentID] = doc - s.mu.Unlock() - - log.Printf("📄 [DOCUMENT-SERVICE] Generated PDF from markdown: %s (%d bytes)", safeFilename, fileInfo.Size()) - - return doc, nil -} - -// generatePDF converts HTML to PDF using chromedp -func (s *Service) generatePDF(htmlContent, outputPath string) error { - // Create allocator options for headless Chrome - opts := append(chromedp.DefaultExecAllocatorOptions[:], - chromedp.ExecPath("/usr/bin/chromium-browser"), - chromedp.NoSandbox, - chromedp.DisableGPU, - chromedp.Flag("disable-dev-shm-usage", true), - chromedp.Flag("no-first-run", true), - chromedp.Flag("no-default-browser-check", true), - ) - - // Create allocator context - allocCtx, allocCancel := chromedp.NewExecAllocator(context.Background(), opts...) - defer allocCancel() - - // Create context - ctx, cancel := chromedp.NewContext(allocCtx) - defer cancel() - - // Set timeout - ctx, cancel = context.WithTimeout(ctx, 30*time.Second) - defer cancel() - - // Generate PDF - var pdfBuffer []byte - if err := chromedp.Run(ctx, - chromedp.Navigate("about:blank"), - chromedp.ActionFunc(func(ctx context.Context) error { - frameTree, err := page.GetFrameTree().Do(ctx) - if err != nil { - return err - } - return page.SetDocumentContent(frameTree.Frame.ID, htmlContent).Do(ctx) - }), - chromedp.ActionFunc(func(ctx context.Context) error { - var err error - pdfBuffer, _, err = page.PrintToPDF(). - WithPrintBackground(true). - WithDisplayHeaderFooter(false). - WithMarginTop(0). - WithMarginBottom(0). - WithMarginLeft(0). - WithMarginRight(0). - WithPaperWidth(8.27). // A4 width in inches - WithPaperHeight(11.69). // A4 height in inches - WithScale(1.0). // 100% scale, no shrinking - Do(ctx) - return err - }), - ); err != nil { - return err - } - - // Write PDF to file with restricted permissions (owner read/write only for security) - if err := os.WriteFile(outputPath, pdfBuffer, 0600); err != nil { - return err - } - - return nil -} - -// GetDocument retrieves a document by ID -func (s *Service) GetDocument(documentID string) (*GeneratedDocument, bool) { - s.mu.RLock() - defer s.mu.RUnlock() - doc, exists := s.documents[documentID] - return doc, exists -} - -// MarkDownloaded marks a document as downloaded -func (s *Service) MarkDownloaded(documentID string) { - s.mu.Lock() - defer s.mu.Unlock() - - if doc, exists := s.documents[documentID]; exists { - now := time.Now() - doc.Downloaded = true - doc.DownloadedAt = &now - log.Printf("✅ [DOCUMENT-SERVICE] Document downloaded: %s", doc.Filename) - } -} - -// CleanupDownloadedDocuments deletes documents that have been downloaded -func (s *Service) CleanupDownloadedDocuments() { - s.mu.Lock() - defer s.mu.Unlock() - - now := time.Now() - cleanedCount := 0 - - for docID, doc := range s.documents { - shouldDelete := false - - // Delete if downloaded AND 5 minutes passed (fast path) - if doc.Downloaded && doc.DownloadedAt != nil { - if now.Sub(*doc.DownloadedAt) > 5*time.Minute { - shouldDelete = true - log.Printf("🗑️ [DOCUMENT-SERVICE] Deleting downloaded document: %s (downloaded %v ago)", - doc.Filename, now.Sub(*doc.DownloadedAt)) - } - } - - // Delete if created over 10 minutes ago (main TTL - privacy-first) - if now.Sub(doc.CreatedAt) > 10*time.Minute { - shouldDelete = true - log.Printf("🗑️ [DOCUMENT-SERVICE] Deleting expired document: %s (created %v ago)", - doc.Filename, now.Sub(doc.CreatedAt)) - } - - if shouldDelete { - // Delete file from disk - if err := os.Remove(doc.FilePath); err != nil && !os.IsNotExist(err) { - log.Printf("⚠️ Failed to delete document file %s: %v", doc.FilePath, err) - } - - // Remove from map - delete(s.documents, docID) - cleanedCount++ - } - } - - if cleanedCount > 0 { - log.Printf("✅ [DOCUMENT-SERVICE] Cleaned up %d documents", cleanedCount) - } -} - -// GenerateTextFile creates a text-based file with the given content and extension -func (s *Service) GenerateTextFile(content, filename, extension, userID, conversationID string) (*GeneratedDocument, error) { - // Sanitize extension (remove leading dot if present) - if len(extension) > 0 && extension[0] == '.' { - extension = extension[1:] - } - - // Generate unique document ID and filename - documentID := uuid.New().String() - safeFilename := filename + "." + extension - filePath := filepath.Join(s.outputDir, documentID+"."+extension) - - // Write content to file with restricted permissions (owner read/write only for security) - if err := os.WriteFile(filePath, []byte(content), 0600); err != nil { - return nil, fmt.Errorf("failed to write text file: %w", err) - } - - // Get file size - fileInfo, err := os.Stat(filePath) - if err != nil { - return nil, fmt.Errorf("failed to get file info: %w", err) - } - - // Get appropriate content type - contentType := getContentTypeForExtension(extension) - - // Create document record - doc := &GeneratedDocument{ - DocumentID: documentID, - UserID: userID, - ConversationID: conversationID, - Filename: safeFilename, - FilePath: filePath, - Size: fileInfo.Size(), - DownloadURL: fmt.Sprintf("/api/download/%s", documentID), - ContentType: contentType, - CreatedAt: time.Now(), - Downloaded: false, - } - - // Store document - s.mu.Lock() - s.documents[documentID] = doc - s.mu.Unlock() - - log.Printf("📝 [DOCUMENT-SERVICE] Generated text file: %s (%d bytes)", safeFilename, fileInfo.Size()) - - return doc, nil -} - -// getContentTypeForExtension returns the MIME type for a given file extension -func getContentTypeForExtension(ext string) string { - contentTypes := map[string]string{ - // Text formats - "txt": "text/plain", - "text": "text/plain", - "log": "text/plain", - - // Data formats - "json": "application/json", - "yaml": "application/x-yaml", - "yml": "application/x-yaml", - "xml": "application/xml", - "csv": "text/csv", - "tsv": "text/tab-separated-values", - - // Config formats - "ini": "text/plain", - "toml": "application/toml", - "env": "text/plain", - "conf": "text/plain", - "cfg": "text/plain", - - // Web formats - "html": "text/html", - "htm": "text/html", - "css": "text/css", - "js": "application/javascript", - "mjs": "application/javascript", - "ts": "application/typescript", - "tsx": "application/typescript", - "jsx": "text/jsx", - - // Markdown - "md": "text/markdown", - "markdown": "text/markdown", - - // Programming languages - "py": "text/x-python", - "go": "text/x-go", - "rs": "text/x-rust", - "java": "text/x-java", - "c": "text/x-c", - "cpp": "text/x-c++", - "h": "text/x-c", - "hpp": "text/x-c++", - "cs": "text/x-csharp", - "rb": "text/x-ruby", - "php": "text/x-php", - "swift": "text/x-swift", - "kt": "text/x-kotlin", - "scala": "text/x-scala", - "r": "text/x-r", - - // Shell scripts - "sh": "application/x-sh", - "bash": "application/x-sh", - "zsh": "application/x-sh", - "ps1": "application/x-powershell", - "bat": "application/x-msdos-program", - "cmd": "application/x-msdos-program", - - // Database - "sql": "application/sql", - - // Other - "diff": "text/x-diff", - "patch": "text/x-diff", - } - - if contentType, ok := contentTypes[ext]; ok { - return contentType - } - - // Default to text/plain for unknown extensions - return "text/plain" -} diff --git a/backend/internal/e2b/executor.go b/backend/internal/e2b/executor.go deleted file mode 100644 index 740fa9aa..00000000 --- a/backend/internal/e2b/executor.go +++ /dev/null @@ -1,334 +0,0 @@ -package e2b - -import ( - "bytes" - "context" - "encoding/json" - "fmt" - "io" - "mime/multipart" - "net/http" - "os" - "time" - - "github.com/sirupsen/logrus" -) - -// E2BExecutorService handles communication with the E2B Python microservice -type E2BExecutorService struct { - baseURL string - httpClient *http.Client - logger *logrus.Logger -} - -// ExecuteRequest represents a code execution request -type ExecuteRequest struct { - Code string `json:"code"` - Timeout int `json:"timeout,omitempty"` // seconds -} - -// PlotResult represents a generated plot -type PlotResult struct { - Format string `json:"format"` // "png", "svg", etc. - Data string `json:"data"` // base64 encoded -} - -// ExecuteResponse represents the response from code execution -type ExecuteResponse struct { - Success bool `json:"success"` - Stdout string `json:"stdout"` - Stderr string `json:"stderr"` - Error *string `json:"error"` - Plots []PlotResult `json:"plots"` - ExecutionTime *float64 `json:"execution_time"` -} - -// AdvancedExecuteRequest represents a request with dependencies and output files -type AdvancedExecuteRequest struct { - Code string `json:"code"` - Timeout int `json:"timeout,omitempty"` - Dependencies []string `json:"dependencies,omitempty"` - OutputFiles []string `json:"output_files,omitempty"` -} - -// FileResult represents a file retrieved from the sandbox -type FileResult struct { - Filename string `json:"filename"` - Data string `json:"data"` // base64 encoded - Size int `json:"size"` -} - -// AdvancedExecuteResponse represents the response with files -type AdvancedExecuteResponse struct { - Success bool `json:"success"` - Stdout string `json:"stdout"` - Stderr string `json:"stderr"` - Error *string `json:"error"` - Plots []PlotResult `json:"plots"` - Files []FileResult `json:"files"` - ExecutionTime *float64 `json:"execution_time"` - InstallOutput string `json:"install_output"` -} - -var ( - e2bExecutorServiceInstance *E2BExecutorService -) - -// GetE2BExecutorService returns the singleton instance of E2BExecutorService -func GetE2BExecutorService() *E2BExecutorService { - if e2bExecutorServiceInstance == nil { - logger := logrus.New() - logger.SetFormatter(&logrus.JSONFormatter{}) - - // Get E2B service URL from environment - baseURL := os.Getenv("E2B_SERVICE_URL") - if baseURL == "" { - baseURL = "http://e2b-service:8001" // Default for Docker Compose - } - - e2bExecutorServiceInstance = &E2BExecutorService{ - baseURL: baseURL, - httpClient: &http.Client{ - Timeout: 330 * time.Second, // 5.5 minutes to allow 5 min execution + overhead - }, - logger: logger, - } - - e2bExecutorServiceInstance.logger.WithField("baseURL", baseURL).Info("E2B Executor Service initialized") - } - return e2bExecutorServiceInstance -} - -// HealthCheck checks if the E2B service is healthy -func (s *E2BExecutorService) HealthCheck(ctx context.Context) error { - url := fmt.Sprintf("%s/health", s.baseURL) - - req, err := http.NewRequestWithContext(ctx, "GET", url, nil) - if err != nil { - return fmt.Errorf("failed to create health check request: %w", err) - } - - resp, err := s.httpClient.Do(req) - if err != nil { - return fmt.Errorf("health check request failed: %w", err) - } - defer resp.Body.Close() - - if resp.StatusCode != http.StatusOK { - body, _ := io.ReadAll(resp.Body) - return fmt.Errorf("health check failed: status=%d, body=%s", resp.StatusCode, string(body)) - } - - s.logger.Info("E2B service health check passed") - return nil -} - -// Execute runs Python code in an E2B sandbox -func (s *E2BExecutorService) Execute(ctx context.Context, code string, timeout int) (*ExecuteResponse, error) { - url := fmt.Sprintf("%s/execute", s.baseURL) - - // Prepare request - reqBody := ExecuteRequest{ - Code: code, - Timeout: timeout, - } - - jsonData, err := json.Marshal(reqBody) - if err != nil { - return nil, fmt.Errorf("failed to marshal request: %w", err) - } - - s.logger.WithFields(logrus.Fields{ - "code_length": len(code), - "timeout": timeout, - }).Info("Executing code in E2B sandbox") - - // Create HTTP request - req, err := http.NewRequestWithContext(ctx, "POST", url, bytes.NewBuffer(jsonData)) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - req.Header.Set("Content-Type", "application/json") - - // Execute request - resp, err := s.httpClient.Do(req) - if err != nil { - return nil, fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - // Read response - body, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - // Check status code - if resp.StatusCode != http.StatusOK { - return nil, fmt.Errorf("execution failed: status=%d, body=%s", resp.StatusCode, string(body)) - } - - // Parse response - var result ExecuteResponse - if err := json.Unmarshal(body, &result); err != nil { - return nil, fmt.Errorf("failed to parse response: %w", err) - } - - s.logger.WithFields(logrus.Fields{ - "success": result.Success, - "plot_count": len(result.Plots), - "has_stdout": len(result.Stdout) > 0, - "has_stderr": len(result.Stderr) > 0, - }).Info("Code execution completed") - - return &result, nil -} - -// ExecuteWithFiles runs Python code with uploaded files -func (s *E2BExecutorService) ExecuteWithFiles(ctx context.Context, code string, files map[string][]byte, timeout int) (*ExecuteResponse, error) { - url := fmt.Sprintf("%s/execute-with-files", s.baseURL) - - // Create multipart form - body := &bytes.Buffer{} - writer := multipart.NewWriter(body) - - // Add code field - if err := writer.WriteField("code", code); err != nil { - return nil, fmt.Errorf("failed to write code field: %w", err) - } - - // Add timeout field - if err := writer.WriteField("timeout", fmt.Sprintf("%d", timeout)); err != nil { - return nil, fmt.Errorf("failed to write timeout field: %w", err) - } - - // Add files - for filename, content := range files { - part, err := writer.CreateFormFile("files", filename) - if err != nil { - return nil, fmt.Errorf("failed to create form file %s: %w", filename, err) - } - - if _, err := part.Write(content); err != nil { - return nil, fmt.Errorf("failed to write file %s: %w", filename, err) - } - - s.logger.WithFields(logrus.Fields{ - "filename": filename, - "size": len(content), - }).Info("Added file to request") - } - - if err := writer.Close(); err != nil { - return nil, fmt.Errorf("failed to close multipart writer: %w", err) - } - - s.logger.WithFields(logrus.Fields{ - "code_length": len(code), - "file_count": len(files), - "timeout": timeout, - }).Info("Executing code with files in E2B sandbox") - - // Create HTTP request - req, err := http.NewRequestWithContext(ctx, "POST", url, body) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - req.Header.Set("Content-Type", writer.FormDataContentType()) - - // Execute request - resp, err := s.httpClient.Do(req) - if err != nil { - return nil, fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - // Read response - respBody, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - // Check status code - if resp.StatusCode != http.StatusOK { - return nil, fmt.Errorf("execution failed: status=%d, body=%s", resp.StatusCode, string(respBody)) - } - - // Parse response - var result ExecuteResponse - if err := json.Unmarshal(respBody, &result); err != nil { - return nil, fmt.Errorf("failed to parse response: %w", err) - } - - s.logger.WithFields(logrus.Fields{ - "success": result.Success, - "plot_count": len(result.Plots), - "has_stdout": len(result.Stdout) > 0, - "has_stderr": len(result.Stderr) > 0, - }).Info("Code execution with files completed") - - return &result, nil -} - -// ExecuteAdvanced runs Python code with dependencies and retrieves output files -func (s *E2BExecutorService) ExecuteAdvanced(ctx context.Context, req AdvancedExecuteRequest) (*AdvancedExecuteResponse, error) { - url := fmt.Sprintf("%s/execute-advanced", s.baseURL) - - // Set default timeout - if req.Timeout == 0 { - req.Timeout = 30 - } - - jsonData, err := json.Marshal(req) - if err != nil { - return nil, fmt.Errorf("failed to marshal request: %w", err) - } - - s.logger.WithFields(logrus.Fields{ - "code_length": len(req.Code), - "timeout": req.Timeout, - "dependencies": req.Dependencies, - "output_files": req.OutputFiles, - }).Info("Executing advanced code in E2B sandbox") - - // Create HTTP request - httpReq, err := http.NewRequestWithContext(ctx, "POST", url, bytes.NewBuffer(jsonData)) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - httpReq.Header.Set("Content-Type", "application/json") - - // Execute request - resp, err := s.httpClient.Do(httpReq) - if err != nil { - return nil, fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - // Read response - body, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - // Check status code - if resp.StatusCode != http.StatusOK { - return nil, fmt.Errorf("execution failed: status=%d, body=%s", resp.StatusCode, string(body)) - } - - // Parse response - var result AdvancedExecuteResponse - if err := json.Unmarshal(body, &result); err != nil { - return nil, fmt.Errorf("failed to parse response: %w", err) - } - - s.logger.WithFields(logrus.Fields{ - "success": result.Success, - "plot_count": len(result.Plots), - "file_count": len(result.Files), - "has_stdout": len(result.Stdout) > 0, - "has_stderr": len(result.Stderr) > 0, - }).Info("Advanced code execution completed") - - return &result, nil -} diff --git a/backend/internal/examples/interactive_prompt_example.go b/backend/internal/examples/interactive_prompt_example.go deleted file mode 100644 index 7c6418c7..00000000 --- a/backend/internal/examples/interactive_prompt_example.go +++ /dev/null @@ -1,180 +0,0 @@ -package examples - -import ( - "github.com/google/uuid" - - "claraverse/internal/models" -) - -// ExampleSimplePrompt shows how to create a simple interactive prompt -// with basic question types (text, number) -func ExampleSimplePrompt(conversationID string) models.ServerMessage { - return models.ServerMessage{ - Type: "interactive_prompt", - PromptID: uuid.New().String(), - ConversationID: conversationID, - Title: "Need More Information", - Description: "To help you better, I need a few more details.", - Questions: []models.InteractiveQuestion{ - { - ID: "name", - Type: "text", - Label: "What's your name?", - Placeholder: "Enter your name...", - Required: true, - }, - { - ID: "age", - Type: "number", - Label: "How old are you?", - Required: false, - Validation: &models.QuestionValidation{ - Min: floatPtr(0), - Max: floatPtr(150), - }, - }, - }, - AllowSkip: boolPtr(true), - } -} - -// ExampleComplexPrompt shows how to create a complex prompt -// with all question types and validation -func ExampleComplexPrompt(conversationID string) models.ServerMessage { - return models.ServerMessage{ - Type: "interactive_prompt", - PromptID: uuid.New().String(), - ConversationID: conversationID, - Title: "Create a New Project", - Description: "To create your project, I need some information about your requirements.", - Questions: []models.InteractiveQuestion{ - { - ID: "language", - Type: "select", - Label: "What programming language do you want to use?", - Required: true, - Options: []string{"Python", "JavaScript", "TypeScript", "Java", "Go"}, - AllowOther: true, - }, - { - ID: "features", - Type: "multi-select", - Label: "Which features do you need?", - Required: true, - Options: []string{"Authentication", "Database", "API", "Testing"}, - AllowOther: true, - }, - { - ID: "complexity", - Type: "number", - Label: "Complexity level (1-10)", - Required: true, - Validation: &models.QuestionValidation{ - Min: floatPtr(1), - Max: floatPtr(10), - }, - }, - { - ID: "async", - Type: "checkbox", - Label: "Use async/await?", - Required: false, - }, - { - ID: "description", - Type: "text", - Label: "Project description", - Placeholder: "Describe your project...", - Required: false, - Validation: &models.QuestionValidation{ - MinLength: intPtr(10), - MaxLength: intPtr(200), - }, - }, - }, - AllowSkip: boolPtr(false), // User must answer - } -} - -// ExampleEmailValidation shows how to create a prompt with email validation -func ExampleEmailValidation(conversationID string) models.ServerMessage { - return models.ServerMessage{ - Type: "interactive_prompt", - PromptID: uuid.New().String(), - ConversationID: conversationID, - Title: "Email Verification", - Description: "Please verify your email address to continue.", - Questions: []models.InteractiveQuestion{ - { - ID: "email", - Type: "text", - Label: "Email address", - Placeholder: "your@email.com", - Required: true, - Validation: &models.QuestionValidation{ - Pattern: `^[^\s@]+@[^\s@]+\.[^\s@]+$`, // Email regex - }, - }, - { - ID: "agree", - Type: "checkbox", - Label: "I agree to the terms and conditions", - Required: true, - }, - }, - AllowSkip: boolPtr(false), - } -} - -// Example of how to use SendInteractivePrompt in a tool or handler -// -// func (h *WebSocketHandler) SomeToolOrHandler(userConn *models.UserConnection) { -// // Create a prompt -// prompt := ExampleSimplePrompt(userConn.ConversationID) -// -// // Send it to the user -// success := h.SendInteractivePrompt(userConn, prompt) -// if !success { -// log.Printf("Failed to send prompt to user") -// return -// } -// -// // The response will be received in handleInteractivePromptResponse -// // You can store the promptID and wait for the response before continuing -// } - -// Example of sending a validation error -func ExampleValidationError(conversationID, promptID string) models.ServerMessage { - return models.ServerMessage{ - Type: "prompt_validation_error", - PromptID: promptID, - ConversationID: conversationID, - Errors: map[string]string{ - "email": "Please enter a valid email address", - "age": "Age must be between 0 and 150", - }, - } -} - -// Example of sending a timeout message -func ExampleTimeout(conversationID, promptID string) models.ServerMessage { - return models.ServerMessage{ - Type: "prompt_timeout", - PromptID: promptID, - ConversationID: conversationID, - ErrorMessage: "Prompt timed out. Please try again.", - } -} - -// Helper functions -func boolPtr(b bool) *bool { - return &b -} - -func floatPtr(f float64) *float64 { - return &f -} - -func intPtr(i int) *int { - return &i -} diff --git a/backend/internal/execution/agent_block_executor.go b/backend/internal/execution/agent_block_executor.go deleted file mode 100644 index 39311ba6..00000000 --- a/backend/internal/execution/agent_block_executor.go +++ /dev/null @@ -1,2887 +0,0 @@ -package execution - -import ( - "bufio" - "bytes" - "claraverse/internal/filecache" - "claraverse/internal/models" - "claraverse/internal/services" - "claraverse/internal/tools" - "context" - "encoding/base64" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "os" - "path/filepath" - "regexp" - "strings" - "time" - - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// ExecutionModePrefix is injected before all system prompts during workflow execution -// This forces the LLM into "action mode" rather than "conversational mode" -const ExecutionModePrefix = `## WORKFLOW EXECUTION MODE - MANDATORY INSTRUCTIONS - -You are operating in WORKFLOW EXECUTION MODE. This is NOT a conversation. - -CRITICAL RULES: -1. DO NOT ask questions - all required data is provided below -2. DO NOT explain what you're about to do - JUST DO IT immediately -3. DO NOT hesitate or offer alternatives - execute the primary task NOW -4. MUST use the available tools to complete your task - tool usage is MANDATORY, not optional -5. DO NOT generate placeholder/example data - use the ACTUAL data provided in the input -6. After completing tool calls, provide a brief confirmation and STOP -7. DO NOT ask for webhook URLs, credentials, or configuration - these are auto-injected - -EXECUTION PATTERN: -1. Read the input data provided -2. Immediately call the required tool(s) with the actual data -3. Return a brief confirmation of what was done -4. STOP - do not continue iterating or ask follow-up questions - -` - -// ToolUsageError represents a validation error when required tools were not called -type ToolUsageError struct { - Type string `json:"type"` // "no_tool_called" or "required_tool_missing" - Message string `json:"message"` - EnabledTools []string `json:"enabledTools,omitempty"` - MissingTools []string `json:"missingTools,omitempty"` -} - -// ToolUsageValidator validates that required tools were called during block execution -type ToolUsageValidator struct { - requiredTools []string - enabledTools []string - requireAnyTool bool -} - -// NewToolUsageValidator creates a new validator based on block configuration -func NewToolUsageValidator(config models.AgentBlockConfig) *ToolUsageValidator { - return &ToolUsageValidator{ - requiredTools: config.RequiredTools, - enabledTools: config.EnabledTools, - requireAnyTool: len(config.EnabledTools) > 0 && config.RequireToolUsage, - } -} - -// Validate checks if the required tools were called -// Returns nil if validation passes, or a ToolUsageError if it fails -// IMPORTANT: Tools that were called but failed (e.g., API rate limits) still count as "called" -// because the LLM correctly attempted to use the tool - we don't want to retry in that case -func (v *ToolUsageValidator) Validate(toolCalls []models.ToolCallRecord) *ToolUsageError { - if !v.requireAnyTool && len(v.requiredTools) == 0 { - return nil // No validation needed - } - - // Build sets of attempted tools (all calls) and successful tools - attemptedTools := make(map[string]bool) - successfulTools := make(map[string]bool) - var failedToolErrors []string - - for _, tc := range toolCalls { - attemptedTools[tc.Name] = true - if tc.Error == "" { - successfulTools[tc.Name] = true - } else { - // Track tool errors for better error messages - failedToolErrors = append(failedToolErrors, fmt.Sprintf("%s: %s", tc.Name, tc.Error)) - } - } - - // Check if any tool was ATTEMPTED (when tools are enabled and required) - // If a tool was called but failed externally (API error), we don't retry - the LLM did its job - if v.requireAnyTool && len(attemptedTools) == 0 { - return &ToolUsageError{ - Type: "no_tool_called", - Message: "Block has tools enabled but none were called. The LLM responded with text only instead of using the available tools.", - EnabledTools: v.enabledTools, - } - } - - // If tools were attempted but ALL failed, check if it's a parameter error or external failure - // Parameter errors (wrong enum, invalid action, etc.) should trigger retry - // External errors (API down, rate limit, auth failure) should not retry - if len(attemptedTools) > 0 && len(successfulTools) == 0 && len(failedToolErrors) > 0 { - // Check if errors are parameter/validation issues that the LLM can fix - allParameterErrors := true - for _, errMsg := range failedToolErrors { - // Parameter errors contain hints like "Did you mean", "is not valid", "unsupported action" - isParameterError := strings.Contains(errMsg, "Did you mean") || - strings.Contains(errMsg, "is not valid") || - strings.Contains(errMsg, "unsupported action") || - strings.Contains(errMsg, "is required") || - strings.Contains(errMsg, "invalid action") - - if !isParameterError { - allParameterErrors = false - break - } - } - - // If all errors are parameter errors, treat as validation failure (will retry) - // If any error is external (API, network, etc.), don't retry - if !allParameterErrors { - log.Printf("⚠️ [TOOL-VALIDATOR] Tools were called but failed externally: %v", failedToolErrors) - return nil // Don't retry - external failure - } - // If allParameterErrors is true, fall through to validation logic below - // This will trigger a retry with the error message feedback - } - - // Check required specific tools - must be ATTEMPTED (not necessarily successful) - // If a required tool was called but failed, that's an external issue, not a retry case - var missingTools []string - for _, required := range v.requiredTools { - if !attemptedTools[required] { - missingTools = append(missingTools, required) - } - } - - if len(missingTools) > 0 { - return &ToolUsageError{ - Type: "required_tool_missing", - Message: fmt.Sprintf("Required tools not called: %v. These tools must be used to complete the task.", missingTools), - MissingTools: missingTools, - } - } - - return nil -} - -// AgentBlockExecutor executes LLM blocks as mini-agents with tool support -type AgentBlockExecutor struct { - chatService *services.ChatService - providerService *services.ProviderService - toolRegistry *tools.Registry - credentialService *services.CredentialService - httpClient *http.Client -} - -// NewAgentBlockExecutor creates a new agent block executor -func NewAgentBlockExecutor( - chatService *services.ChatService, - providerService *services.ProviderService, - toolRegistry *tools.Registry, - credentialService *services.CredentialService, -) *AgentBlockExecutor { - return &AgentBlockExecutor{ - chatService: chatService, - providerService: providerService, - toolRegistry: toolRegistry, - credentialService: credentialService, - httpClient: &http.Client{ - Timeout: 120 * time.Second, - }, - } -} - -// Execute runs an LLM block as a mini-agent with tool support -// Uses two-phase approach: -// - Phase 1: Task execution with tools (no schema concerns) -// - Phase 2: Schema formatting (dedicated formatting step) -// Retry logic only handles tool usage validation, not schema errors -func (e *AgentBlockExecutor) Execute(ctx context.Context, block models.Block, inputs map[string]any) (map[string]any, error) { - // Parse config with defaults - config := e.parseConfig(block.Config) - - // Create tool usage validator - validator := NewToolUsageValidator(config) - - // Retry loop for tool usage validation ONLY - // Schema formatting is now handled as a separate phase in executeOnce - var lastResult map[string]any - var lastValidationError *ToolUsageError - - for attempt := 0; attempt <= config.MaxRetries; attempt++ { - if attempt > 0 { - retryReason := fmt.Sprintf("Tool validation failed: %s", lastValidationError.Message) - log.Printf("🔄 [AGENT-BLOCK] Retry attempt %d for block '%s' - %s", - attempt, block.Name, retryReason) - // Inject retry context for stronger prompting - inputs["_retryAttempt"] = attempt - inputs["_retryReason"] = retryReason - } - - // Execute the block (includes Phase 1 + Phase 2) - result, err := e.executeOnce(ctx, block, inputs, config) - if err != nil { - return nil, err // Execution error, not validation error - don't retry - } - - // Validate tool usage - toolCalls, _ := result["toolCalls"].([]models.ToolCallRecord) - validationError := validator.Validate(toolCalls) - - if validationError != nil { - // Tool validation failed - lastResult = result - lastValidationError = validationError - log.Printf("⚠️ [AGENT-BLOCK] Block '%s' tool validation failed (attempt %d/%d): %s", - block.Name, attempt+1, config.MaxRetries+1, validationError.Message) - - // If this was the last attempt, return with warning - if attempt == config.MaxRetries { - log.Printf("⚠️ [AGENT-BLOCK] Block '%s' exhausted all %d retry attempts, returning with tool validation warning", - block.Name, config.MaxRetries+1) - result["_toolValidationWarning"] = validationError.Message - result["_toolValidationType"] = validationError.Type - delete(result, "_retryAttempt") - delete(result, "_retryReason") - return result, nil - } - - // Clear retry-specific state for next attempt - delete(inputs, "_retryAttempt") - delete(inputs, "_retryReason") - continue - } - - // Tool validation passed - schema formatting was already handled in executeOnce Phase 2 - // Clean up and return - delete(result, "_retryAttempt") - delete(result, "_retryReason") - if attempt > 0 { - log.Printf("✅ [AGENT-BLOCK] Block '%s' succeeded on retry attempt %d", block.Name, attempt) - } - return result, nil - } - - // Should never reach here, but return last result as fallback - return lastResult, nil -} - -// executeOnce performs a single execution attempt of the LLM block -func (e *AgentBlockExecutor) executeOnce(ctx context.Context, block models.Block, inputs map[string]any, config models.AgentBlockConfig) (map[string]any, error) { - - // Check for workflow-level model override (set in Start block) - if workflowModelID, ok := inputs["_workflowModelId"].(string); ok && workflowModelID != "" { - log.Printf("🎯 [AGENT-BLOCK] Block '%s': Using workflow model override: %s", block.Name, workflowModelID) - config.Model = workflowModelID - } - - log.Printf("🤖 [AGENT-BLOCK] Block '%s': model=%s, enabledTools=%v, maxToolCalls=%d", - block.Name, config.Model, config.EnabledTools, config.MaxToolCalls) - - // Resolve model (alias -> direct -> fallback) - provider, modelID, err := e.resolveModel(config.Model) - if err != nil { - return nil, fmt.Errorf("failed to resolve model: %w", err) - } - - log.Printf("✅ [AGENT-BLOCK] Resolved model '%s' -> '%s' (provider: %s)", - config.Model, modelID, provider.Name) - - // Extract data files for context injection (but don't auto-enable tools) - dataFiles := e.extractDataFileAttachments(inputs) - if len(dataFiles) > 0 { - log.Printf("📊 [AGENT-BLOCK] Found %d data file(s) - tools must be explicitly configured", len(dataFiles)) - } - - // NOTE: Auto-detection removed - blocks only use explicitly configured tools - // Users must configure enabledTools in the block settings - - // Build initial messages with interpolated prompts - messages := e.buildMessages(config, inputs) - - // Filter tools to only those enabled for this block - enabledTools := e.filterTools(config.EnabledTools) - - log.Printf("🔧 [AGENT-BLOCK] Enabled %d tools for block '%s'", len(enabledTools), block.Name) - - // Track tool calls and token usage - var allToolCalls []models.ToolCallRecord - var totalTokens models.TokenUsage - iterations := 0 - var lastContent string // Track last content for timeout handling - - // Track generated chart images for auto-injection into Discord/Slack messages - // The LLM sees sanitized placeholders like [CHART_IMAGE_SAVED] but we need the real base64 - var generatedCharts []string - - // PRE-POPULATE generatedCharts with artifacts from previous blocks - // This allows Discord/Slack blocks to access charts generated by upstream blocks - generatedCharts = e.extractChartsFromInputs(inputs) - if len(generatedCharts) > 0 { - log.Printf("🖼️ [AGENT-BLOCK] Pre-loaded %d chart(s) from previous block artifacts for auto-injection", len(generatedCharts)) - } - - // Run agent loop - continues until LLM stops calling tools or timeout - for { - iterations++ - log.Printf("🔄 [AGENT-BLOCK] Iteration %d for block '%s'", iterations, block.Name) - - // Enforce iteration limit to prevent infinite loops - maxIterations := 10 // Default safety limit - if config.MaxToolCalls > 0 { - maxIterations = config.MaxToolCalls - } - if iterations > maxIterations { - log.Printf("🛑 [AGENT-BLOCK] Block '%s' reached max iterations (%d)", block.Name, maxIterations) - return e.buildTimeoutResult(inputs, modelID, totalTokens, allToolCalls, lastContent, iterations) - } - - // Track tool calls to detect repetition within this iteration only - // Reset per iteration to allow same tool across different iterations (legitimate refinement) - executedToolCalls := make(map[string]bool) - - // Check if context is cancelled (timeout) - select { - case <-ctx.Done(): - log.Printf("⏱️ [AGENT-BLOCK] Block '%s' timed out after %d iterations", block.Name, iterations) - return e.buildTimeoutResult(inputs, modelID, totalTokens, allToolCalls, lastContent, iterations) - default: - // Continue execution - } - - // Call LLM with retry for transient errors (timeout, rate limit, server errors) - response, retryAttempts, err := e.callLLMWithRetry(ctx, provider, modelID, messages, enabledTools, config.Temperature, config.RetryPolicy) - if err != nil { - // Include retry info in error for debugging - if len(retryAttempts) > 0 { - return nil, fmt.Errorf("LLM call failed in iteration %d after %d attempt(s): %w", iterations, len(retryAttempts)+1, err) - } - return nil, fmt.Errorf("LLM call failed in iteration %d: %w", iterations, err) - } - - // Track retry attempts for surfacing in output (first iteration only to avoid duplicates) - if iterations == 1 && len(retryAttempts) > 0 { - inputs["_llmRetryAttempts"] = retryAttempts - } - - // Accumulate tokens - totalTokens.Input += response.InputTokens - totalTokens.Output += response.OutputTokens - - // Check finish_reason for explicit stop signal from LLM - // This is how chat mode knows to stop - agent mode should do the same - if response.FinishReason == "stop" || response.FinishReason == "end_turn" { - log.Printf("✅ [AGENT-BLOCK] LLM signaled stop (finish_reason=%s), completing block '%s'", - response.FinishReason, block.Name) - // Continue to completion logic below (same as no tool calls) - } - - // Check if there are tool calls - if len(response.ToolCalls) == 0 || response.FinishReason == "stop" || response.FinishReason == "end_turn" { - // No more tools - agent is done (Phase 1 complete) - log.Printf("✅ [AGENT-BLOCK] Block '%s' Phase 1 (task execution) completed after %d iteration(s)", block.Name, iterations) - - // Build result starting with all inputs (pass through workflow variables) - result := make(map[string]any) - for k, v := range inputs { - result[k] = v - } - - // Store raw LLM response - result["rawResponse"] = response.Content - result["response"] = response.Content // Default, may be overwritten by schema formatting - - // CRITICAL: Extract and surface tool results for downstream blocks - // This solves the data passing problem where tool output was buried in toolCalls - toolResults := e.extractToolResultsForDownstream(allToolCalls) - toolResultsMaps := make([]map[string]any, 0) - if len(toolResults) > 0 { - // Store parsed tool results for easy access - result["toolResults"] = toolResults - - // Surface key data fields at top level for template access - // e.g., {{block-name.text}} instead of {{block-name.toolResults.transcribe_audio.text}} - for _, tr := range toolResults { - if trMap, ok := tr.(map[string]any); ok { - toolResultsMaps = append(toolResultsMaps, trMap) - // Surface common data fields - for _, key := range []string{"text", "data", "content", "result", "output", "transcription"} { - if val, exists := trMap[key]; exists && val != nil { - // Only surface if not already set by LLM response - if _, alreadySet := result[key]; !alreadySet { - result[key] = val - log.Printf("📤 [AGENT-BLOCK] Surfaced tool result field '%s' to top level", key) - } - } - } - } - } - } - - // Phase 2: Schema formatting (if schema is defined) - // This is the key change - we use a dedicated formatting step for structured output - if config.OutputSchema != nil { - log.Printf("📐 [AGENT-BLOCK] Block '%s' starting Phase 2 (schema formatting)", block.Name) - - // Prepare input for schema formatter - formatInput := FormatInput{ - RawData: response.Content, - ToolResults: toolResultsMaps, - LLMResponse: response.Content, - Context: config.SystemPrompt, - } - - // Call the dedicated schema formatter - formatOutput, err := e.FormatToSchema(ctx, formatInput, config.OutputSchema, modelID) - if err != nil { - log.Printf("⚠️ [AGENT-BLOCK] Phase 2 schema formatting error: %v", err) - // Continue with raw output if formatting fails - result["_formatError"] = err.Error() - } else if formatOutput != nil { - if formatOutput.Success { - log.Printf("✅ [AGENT-BLOCK] Phase 2 schema formatting succeeded") - // Use the formatted data as the response - result["response"] = formatOutput.Data - result["data"] = formatOutput.Data - result["output"] = formatOutput.Data - // Also spread the fields at top level for easy access - for k, v := range formatOutput.Data { - result[k] = v - } - // Track the formatting tokens - totalTokens.Input += formatOutput.Tokens.Input - totalTokens.Output += formatOutput.Tokens.Output - } else { - log.Printf("⚠️ [AGENT-BLOCK] Phase 2 schema formatting failed: %s", formatOutput.Error) - result["_formatError"] = formatOutput.Error - // Fall back to basic parsing without validation - if parsedOutput, err := e.parseAndValidateOutput(response.Content, nil, false); err == nil { - for k, v := range parsedOutput { - result[k] = v - } - } - } - } - } else { - // No schema defined - just parse the response as-is - output, err := e.parseAndValidateOutput(response.Content, nil, false) - if err != nil { - log.Printf("⚠️ [AGENT-BLOCK] Output parsing error (no schema): %v", err) - } else { - for k, v := range output { - result[k] = v - } - result["output"] = output - } - } - - // If LLM response is just a summary but we have tool data, use tool data as response - if len(allToolCalls) > 0 { - llmResponse, _ := result["response"].(string) - // Check if LLM response is a short summary (likely not the actual data) - if len(llmResponse) < 500 && len(toolResults) > 0 { - // Check if we have a text field from tools that's more substantial - if textResult, ok := result["text"].(string); ok && len(textResult) > len(llmResponse) { - // Keep LLM response as summary, but ensure text is available - result["summary"] = llmResponse - log.Printf("📤 [AGENT-BLOCK] Tool 'text' field (%d chars) surfaced; LLM response (%d chars) kept as 'summary'", - len(textResult), len(llmResponse)) - } - } - } - - result["model"] = modelID - result["tokens"] = map[string]int{ - "input": totalTokens.Input, - "output": totalTokens.Output, - "total": totalTokens.Input + totalTokens.Output, - } - result["toolCalls"] = allToolCalls - result["iterations"] = iterations - - // Check for tool errors and surface them for block checker - // This helps distinguish between "LLM didn't use tools" vs "LLM used tools but they failed externally" - var toolErrors []string - for _, tc := range allToolCalls { - if tc.Error != "" { - toolErrors = append(toolErrors, fmt.Sprintf("%s: %s", tc.Name, tc.Error)) - } - } - if len(toolErrors) > 0 { - result["_toolError"] = strings.Join(toolErrors, "; ") - log.Printf("⚠️ [AGENT-BLOCK] Block has %d tool error(s): %v", len(toolErrors), toolErrors) - } - - // Extract artifacts (charts, images) from tool calls for consistent API access - artifacts := e.extractArtifactsFromToolCalls(allToolCalls) - result["artifacts"] = artifacts - - // Extract generated files (PDFs, documents) from tool calls - generatedFiles := e.extractGeneratedFilesFromToolCalls(allToolCalls) - result["generatedFiles"] = generatedFiles - // Also expose the first file's download URL directly for easy access - if len(generatedFiles) > 0 { - result["file_url"] = generatedFiles[0].DownloadURL - result["file_name"] = generatedFiles[0].Filename - } - - // Surface retry information for debugging/monitoring - if retryAttempts, ok := inputs["_llmRetryAttempts"].([]models.RetryAttempt); ok && len(retryAttempts) > 0 { - result["_retryInfo"] = map[string]any{ - "totalAttempts": len(retryAttempts) + 1, - "retriedCount": len(retryAttempts), - "history": retryAttempts, - } - log.Printf("📊 [AGENT-BLOCK] Block completed with %d retry attempt(s)", len(retryAttempts)) - } - - log.Printf("🔍 [AGENT-BLOCK] Output keys: %v, artifacts: %d, files: %d, toolResults: %d", - getMapKeys(result), len(artifacts), len(generatedFiles), len(toolResults)) - return result, nil - } - - // Execute tools and add results to messages - log.Printf("🔧 [AGENT-BLOCK] Executing %d tool call(s) in iteration %d", len(response.ToolCalls), iterations) - - // Add assistant message with tool calls - assistantMsg := map[string]any{ - "role": "assistant", - "tool_calls": response.ToolCalls, - } - if response.Content != "" { - assistantMsg["content"] = response.Content - } - messages = append(messages, assistantMsg) - - // Execute each tool call - repeatDetected := false - for _, toolCall := range response.ToolCalls { - toolName := e.getToolName(toolCall) - - // Check for repetition - if same tool called twice, it's likely looping - if executedToolCalls[toolName] { - log.Printf("⚠️ [AGENT-BLOCK] Detected repeated call to '%s', stopping to prevent loop", toolName) - repeatDetected = true - break - } - executedToolCalls[toolName] = true - - // Extract userID from inputs for credential resolution (uses __user_id__ convention) - userID, _ := inputs["__user_id__"].(string) - toolRecord := e.executeToolCall(toolCall, inputs, dataFiles, generatedCharts, userID, config.Credentials) - allToolCalls = append(allToolCalls, toolRecord) - - // Extract any chart images from successful tool results for later injection - if toolRecord.Error == "" && toolRecord.Result != "" { - charts := e.extractChartsFromResult(toolRecord.Result) - if len(charts) > 0 { - generatedCharts = append(generatedCharts, charts...) - log.Printf("📊 [AGENT-BLOCK] Extracted %d chart(s) from tool '%s' (total: %d)", - len(charts), toolName, len(generatedCharts)) - } - } - - // Sanitize tool result for LLM - remove base64 images which are useless as text - sanitizedResult := e.sanitizeToolResultForLLM(toolRecord.Result) - - // Add tool result to messages - toolResultMsg := map[string]any{ - "role": "tool", - "tool_call_id": toolCall["id"], - "name": toolName, - "content": sanitizedResult, - } - if toolRecord.Error != "" { - toolResultMsg["content"] = fmt.Sprintf("Error: %s", toolRecord.Error) - } - messages = append(messages, toolResultMsg) - } - - // If repetition detected, exit loop and return current results - if repeatDetected { - log.Printf("🛑 [AGENT-BLOCK] Exiting loop due to repeated tool call") - return e.buildTimeoutResult(inputs, modelID, totalTokens, allToolCalls, lastContent, iterations) - } - - // Track last content for timeout fallback - if response.Content != "" { - lastContent = response.Content - } - } - // Note: Loop only exits via return statements (success or timeout) -} - -// buildTimeoutResult creates a result when the block times out -// Instead of returning an error, it returns the collected tool call data -// so downstream blocks can still use the information gathered -func (e *AgentBlockExecutor) buildTimeoutResult( - inputs map[string]any, - modelID string, - totalTokens models.TokenUsage, - allToolCalls []models.ToolCallRecord, - lastContent string, - iterations int, -) (map[string]any, error) { - // Build result starting with all inputs (pass through workflow variables) - result := make(map[string]any) - for k, v := range inputs { - result[k] = v - } - - // Build a summary from tool call results if no meaningful content was generated - var outputContent string - trimmedContent := strings.TrimSpace(lastContent) - if trimmedContent != "" { - outputContent = lastContent - } else if len(allToolCalls) > 0 { - // Compile tool results as the output - var summaryParts []string - for _, tc := range allToolCalls { - if tc.Result != "" && tc.Error == "" { - summaryParts = append(summaryParts, tc.Result) - } - } - if len(summaryParts) > 0 { - outputContent = strings.Join(summaryParts, "\n\n") - } - } - - // Flatten output fields directly into result for consistent access - // This makes {{block-name.response}} work the same as simple LLM executor - result["response"] = outputContent - result["timedOut"] = true - - // CRITICAL: Extract and surface tool results for downstream blocks (same as normal completion) - toolResults := e.extractToolResultsForDownstream(allToolCalls) - if len(toolResults) > 0 { - result["toolResults"] = toolResults - - // Surface key data fields at top level for template access - for _, tr := range toolResults { - if trMap, ok := tr.(map[string]any); ok { - for _, key := range []string{"text", "data", "content", "result", "output", "transcription"} { - if val, exists := trMap[key]; exists && val != nil { - if _, alreadySet := result[key]; !alreadySet { - result[key] = val - log.Printf("📤 [AGENT-BLOCK] Surfaced tool result field '%s' to top level (timeout)", key) - } - } - } - } - } - } - - // Also keep "output" for backward compatibility - result["output"] = map[string]any{ - "response": outputContent, - "timedOut": true, - "iterations": iterations, - "toolResults": len(allToolCalls), - } - result["rawResponse"] = outputContent - result["model"] = modelID - result["tokens"] = map[string]int{ - "input": totalTokens.Input, - "output": totalTokens.Output, - "total": totalTokens.Input + totalTokens.Output, - } - result["toolCalls"] = allToolCalls - result["iterations"] = iterations - - // Extract artifacts (charts, images) from tool calls for consistent API access - artifacts := e.extractArtifactsFromToolCalls(allToolCalls) - result["artifacts"] = artifacts - - // Extract generated files (PDFs, documents) from tool calls - generatedFiles := e.extractGeneratedFilesFromToolCalls(allToolCalls) - result["generatedFiles"] = generatedFiles - // Also expose the first file's download URL directly for easy access - if len(generatedFiles) > 0 { - result["file_url"] = generatedFiles[0].DownloadURL - result["file_name"] = generatedFiles[0].Filename - } - - log.Printf("⏱️ [AGENT-BLOCK] Timeout result built with %d tool calls, content length: %d, artifacts: %d, files: %d, toolResults: %d", - len(allToolCalls), len(outputContent), len(artifacts), len(generatedFiles), len(toolResults)) - log.Printf("🔍 [AGENT-BLOCK] Output keys: %v", getMapKeys(result)) - - return result, nil -} - -// parseToolsList converts various array types to []string for tool names -func parseToolsList(raw interface{}) []string { - var tools []string - - switch v := raw.(type) { - case []interface{}: - for _, t := range v { - if toolName, ok := t.(string); ok { - tools = append(tools, toolName) - } - } - case []string: - tools = v - case primitive.A: // BSON array type - for _, t := range v { - if toolName, ok := t.(string); ok { - tools = append(tools, toolName) - } - } - default: - log.Printf("⚠️ [CONFIG] Unknown enabledTools type: %T", raw) - } - - return tools -} - -// getDefaultModel returns the first available model from database -func (e *AgentBlockExecutor) getDefaultModel() string { - // Use chatService to get optimal text model - provider, modelID, err := e.chatService.GetTextProviderWithModel() - if err == nil && modelID != "" { - log.Printf("🎯 [AGENT-EXEC] Using dynamic default model: %s (provider: %s)", modelID, provider.Name) - return modelID - } - - // If that fails, try to get default provider with model - provider, modelID, err = e.chatService.GetDefaultProviderWithModel() - if err == nil && modelID != "" { - log.Printf("🎯 [AGENT-EXEC] Using fallback default model: %s", modelID) - return modelID - } - - // Last resort: return empty string (will cause error later if no model specified) - log.Printf("⚠️ [AGENT-EXEC] No default model available - agent execution will require explicit model") - return "" -} - -// parseConfig parses block config into AgentBlockConfig with defaults -func (e *AgentBlockExecutor) parseConfig(config map[string]any) models.AgentBlockConfig { - // Get dynamic default model from available models - defaultModel := e.getDefaultModel() - - result := models.AgentBlockConfig{ - Model: defaultModel, - Temperature: 0.7, - MaxToolCalls: 15, // Increased to allow agents with multiple search iterations - } - - // Model - if v, ok := config["model"].(string); ok && v != "" { - result.Model = v - } - if v, ok := config["modelId"].(string); ok && v != "" { - result.Model = v - } - - // Temperature - if v, ok := config["temperature"].(float64); ok { - result.Temperature = v - } - - // System prompt - if v, ok := config["systemPrompt"].(string); ok { - result.SystemPrompt = v - } - if v, ok := config["system_prompt"].(string); ok && result.SystemPrompt == "" { - result.SystemPrompt = v - } - - // User prompt - if v, ok := config["userPrompt"].(string); ok { - result.UserPrompt = v - } - if v, ok := config["userPromptTemplate"].(string); ok && result.UserPrompt == "" { - result.UserPrompt = v - } - if v, ok := config["user_prompt"].(string); ok && result.UserPrompt == "" { - result.UserPrompt = v - } - - // Enabled tools - handle multiple possible types from BSON/JSON - if enabledToolsRaw, exists := config["enabledTools"]; exists && enabledToolsRaw != nil { - result.EnabledTools = parseToolsList(enabledToolsRaw) - log.Printf("🔧 [CONFIG] Parsed enabledTools from config: %v (type was %T)", result.EnabledTools, enabledToolsRaw) - } - if len(result.EnabledTools) == 0 { - if enabledToolsRaw, exists := config["enabled_tools"]; exists && enabledToolsRaw != nil { - result.EnabledTools = parseToolsList(enabledToolsRaw) - log.Printf("🔧 [CONFIG] Parsed enabled_tools from config: %v (type was %T)", result.EnabledTools, enabledToolsRaw) - } - } - - // Max tool calls - if v, ok := config["maxToolCalls"].(float64); ok { - result.MaxToolCalls = int(v) - } - if v, ok := config["max_tool_calls"].(float64); ok && result.MaxToolCalls == 15 { - result.MaxToolCalls = int(v) - } - - // Credentials - array of credential IDs configured by user for tool authentication - if credentialsRaw, exists := config["credentials"]; exists && credentialsRaw != nil { - result.Credentials = parseToolsList(credentialsRaw) // Reuse the same parser ([]string) - log.Printf("🔐 [CONFIG] Parsed credentials from config: %v", result.Credentials) - } - - // Output schema - if v, ok := config["outputSchema"].(map[string]any); ok { - result.OutputSchema = e.parseJSONSchema(v) - log.Printf("📋 [CONFIG] Parsed outputSchema with %d required fields: %v", len(result.OutputSchema.Required), result.OutputSchema.Required) - } else { - log.Printf("📋 [CONFIG] No outputSchema found in config (outputSchema key: %v)", config["outputSchema"] != nil) - } - - // Strict output - if v, ok := config["strictOutput"].(bool); ok { - result.StrictOutput = v - } - - // Execution Mode Configuration (NEW) - // Parse requireToolUsage - if explicitly set, use that value - if v, ok := config["requireToolUsage"].(bool); ok { - result.RequireToolUsage = v - } else { - // Default: Auto-enable when tools are present for deterministic execution - result.RequireToolUsage = len(result.EnabledTools) > 0 - } - - // Parse maxRetries - default to 2 for resilience - result.MaxRetries = 2 // Default - if v, ok := config["maxRetries"].(float64); ok { - result.MaxRetries = int(v) - } - if v, ok := config["max_retries"].(float64); ok { - result.MaxRetries = int(v) - } - - // Parse requiredTools - specific tools that MUST be called - if requiredToolsRaw, exists := config["requiredTools"]; exists && requiredToolsRaw != nil { - result.RequiredTools = parseToolsList(requiredToolsRaw) - log.Printf("🔧 [CONFIG] Parsed requiredTools from config: %v", result.RequiredTools) - } - if len(result.RequiredTools) == 0 { - if requiredToolsRaw, exists := config["required_tools"]; exists && requiredToolsRaw != nil { - result.RequiredTools = parseToolsList(requiredToolsRaw) - } - } - - // Auto-lower temperature for execution mode when tools are enabled - // Lower temp = more deterministic tool calling - if len(result.EnabledTools) > 0 { - if _, explicitTemp := config["temperature"]; !explicitTemp { - result.Temperature = 0.3 - log.Printf("🔧 [CONFIG] Auto-lowered temperature to 0.3 for execution mode") - } - } - - // Parse RetryPolicy for LLM API call retries (transient error handling) - if retryPolicyRaw, exists := config["retryPolicy"]; exists && retryPolicyRaw != nil { - if retryMap, ok := retryPolicyRaw.(map[string]any); ok { - result.RetryPolicy = &models.RetryPolicy{} - - if v, ok := retryMap["maxRetries"].(float64); ok { - result.RetryPolicy.MaxRetries = int(v) - } - if v, ok := retryMap["initialDelay"].(float64); ok { - result.RetryPolicy.InitialDelay = int(v) - } - if v, ok := retryMap["maxDelay"].(float64); ok { - result.RetryPolicy.MaxDelay = int(v) - } - if v, ok := retryMap["backoffMultiplier"].(float64); ok { - result.RetryPolicy.BackoffMultiplier = v - } - if v, ok := retryMap["jitterPercent"].(float64); ok { - result.RetryPolicy.JitterPercent = int(v) - } - if retryOn, ok := retryMap["retryOn"].([]interface{}); ok { - for _, r := range retryOn { - if s, ok := r.(string); ok { - result.RetryPolicy.RetryOn = append(result.RetryPolicy.RetryOn, s) - } - } - } - log.Printf("🔧 [CONFIG] Parsed retryPolicy: maxRetries=%d, initialDelay=%dms", - result.RetryPolicy.MaxRetries, result.RetryPolicy.InitialDelay) - } - } - - // Apply default retry policy if not specified (for production resilience) - if result.RetryPolicy == nil { - result.RetryPolicy = models.DefaultRetryPolicy() - } - - log.Printf("🔧 [CONFIG] Execution mode: requireToolUsage=%v, maxRetries=%d, requiredTools=%v", - result.RequireToolUsage, result.MaxRetries, result.RequiredTools) - - return result -} - -// parseJSONSchema converts a map to JSONSchema -func (e *AgentBlockExecutor) parseJSONSchema(schema map[string]any) *models.JSONSchema { - result := &models.JSONSchema{} - - if v, ok := schema["type"].(string); ok { - result.Type = v - } - - if v, ok := schema["properties"].(map[string]any); ok { - result.Properties = make(map[string]*models.JSONSchema) - for key, prop := range v { - if propMap, ok := prop.(map[string]any); ok { - result.Properties[key] = e.parseJSONSchema(propMap) - } - } - } - - if v, ok := schema["items"].(map[string]any); ok { - result.Items = e.parseJSONSchema(v) - } - - // Handle required field - support multiple Go types including MongoDB's primitive.A - // Note: []interface{} and []any are the same type in Go, so only use one - switch v := schema["required"].(type) { - case []interface{}: - log.Printf("📋 [SCHEMA] Found required field ([]interface{}): %v", v) - for _, r := range v { - if req, ok := r.(string); ok { - result.Required = append(result.Required, req) - } - } - case []string: - log.Printf("📋 [SCHEMA] Found required field ([]string): %v", v) - result.Required = v - case primitive.A: // MongoDB BSON array type - log.Printf("📋 [SCHEMA] Found required field (primitive.A): %v", v) - for _, r := range v { - if req, ok := r.(string); ok { - result.Required = append(result.Required, req) - } - } - default: - if schema["required"] != nil { - log.Printf("📋 [SCHEMA] Unhandled required field type: %T", schema["required"]) - } - } - - if v, ok := schema["description"].(string); ok { - result.Description = v - } - - return result -} - -// jsonSchemaToMap converts a JSONSchema struct to a map for API requests -// This is used for native structured output (response_format with json_schema) -func (e *AgentBlockExecutor) jsonSchemaToMap(schema *models.JSONSchema) map[string]interface{} { - if schema == nil { - return nil - } - - result := map[string]interface{}{ - "type": schema.Type, - } - - // Convert properties - if len(schema.Properties) > 0 { - props := make(map[string]interface{}) - for key, prop := range schema.Properties { - props[key] = e.jsonSchemaToMap(prop) - } - result["properties"] = props - } - - // Convert items (for arrays) - if schema.Items != nil { - result["items"] = e.jsonSchemaToMap(schema.Items) - } - - // Add required fields - if len(schema.Required) > 0 { - result["required"] = schema.Required - } - - // Add description if present - if schema.Description != "" { - result["description"] = schema.Description - } - - // Add enum if present - if len(schema.Enum) > 0 { - result["enum"] = schema.Enum - } - - // Strict mode requires additionalProperties: false for objects - if schema.Type == "object" { - result["additionalProperties"] = false - } - - return result -} - -// resolveModel resolves model alias to actual model ID and provider -func (e *AgentBlockExecutor) resolveModel(modelID string) (*models.Provider, string, error) { - // Step 1: Try direct lookup - provider, err := e.providerService.GetByModelID(modelID) - if err == nil { - return provider, modelID, nil - } - - // Step 2: Try model alias resolution - log.Printf("🔄 [AGENT-BLOCK] Model '%s' not found directly, trying alias resolution...", modelID) - if aliasProvider, aliasModel, found := e.chatService.ResolveModelAlias(modelID); found { - return aliasProvider, aliasModel, nil - } - - // Step 3: Fallback to default provider with model - log.Printf("⚠️ [AGENT-BLOCK] Model '%s' not found, using default provider", modelID) - defaultProvider, defaultModel, err := e.chatService.GetDefaultProviderWithModel() - if err != nil { - return nil, "", fmt.Errorf("failed to find provider for model %s: %w", modelID, err) - } - - return defaultProvider, defaultModel, nil -} - -// buildMessages creates the initial messages with interpolated prompts -func (e *AgentBlockExecutor) buildMessages(config models.AgentBlockConfig, inputs map[string]any) []map[string]any { - messages := []map[string]any{} - - log.Printf("🔍 [AGENT-BLOCK] Building messages with inputs: %+v", inputs) - - // Check for data file attachments first (needed for system prompt enhancement) - dataAttachments := e.extractDataFileAttachments(inputs) - - // Build the enhanced system prompt with execution mode prefix - var systemPromptBuilder strings.Builder - - // ALWAYS inject execution mode preamble for deterministic behavior - systemPromptBuilder.WriteString(ExecutionModePrefix) - - // Add tool-specific mandatory instructions when tools are enabled - if len(config.EnabledTools) > 0 { - systemPromptBuilder.WriteString("## REQUIRED TOOLS FOR THIS TASK\n") - systemPromptBuilder.WriteString("You MUST use one or more of these tools to complete your task:\n\n") - - // Get tool descriptions to help the LLM understand how to use them - toolDescriptions := e.getToolDescriptions(config.EnabledTools) - for _, toolDesc := range toolDescriptions { - systemPromptBuilder.WriteString(toolDesc) - systemPromptBuilder.WriteString("\n") - } - - systemPromptBuilder.WriteString("\nIMPORTANT: DO NOT respond with text only. You MUST call at least one of the above tools.\n\n") - } - - // Check for retry context and add stronger instructions - if retryAttempt, ok := inputs["_retryAttempt"].(int); ok && retryAttempt > 0 { - retryReason, _ := inputs["_retryReason"].(string) - - // Determine if this is a schema error or tool error - if strings.Contains(retryReason, "Schema validation failed") || strings.Contains(retryReason, "schema") { - // Schema validation retry - guide LLM to fix JSON output format - systemPromptBuilder.WriteString(fmt.Sprintf(`## ⚠️ SCHEMA VALIDATION RETRY (Attempt %d) -Your previous response did NOT match the required JSON schema. -Error: %s - -CRITICAL REQUIREMENTS: -1. Your response MUST be valid JSON that matches the exact schema structure -2. Include ALL required fields - missing fields cause validation failure -3. Use the correct data types (strings vs numbers) as defined in the schema -4. If the schema expects an object with an array property, wrap your array in an object -5. Do NOT add extra fields not defined in the schema - -Fix your response NOW to match the required schema exactly. - -`, retryAttempt+1, retryReason)) - } else { - // Tool usage retry - systemPromptBuilder.WriteString(fmt.Sprintf(`## RETRY NOTICE (Attempt %d) -Your previous response did not use the required tools. -Reason: %s - -YOU MUST call the appropriate tool(s) NOW. Do not respond with text only. -This is your last chance - call the tool immediately. - -`, retryAttempt+1, retryReason)) - } - log.Printf("🔄 [AGENT-BLOCK] Added retry notice for attempt %d (reason: %s)", retryAttempt+1, retryReason) - } - - // Add the user's system prompt - if config.SystemPrompt != "" { - systemPromptBuilder.WriteString("## YOUR SPECIFIC TASK\n") - systemPromptBuilder.WriteString(interpolateTemplate(config.SystemPrompt, inputs)) - systemPromptBuilder.WriteString("\n\n") - } - - // If data files present, add analysis guidelines to system prompt - if len(dataAttachments) > 0 { - systemPromptBuilder.WriteString(` -## Data Analysis Guidelines -- The data file content is provided in the user message - you can see the structure -- Use the 'analyze_data' tool to run Python code - data is pre-loaded as pandas DataFrame 'df' -- Generate all charts/visualizations in ONE comprehensive tool call -- After receiving results with charts, provide your insights and STOP - do not repeat the analysis -- Charts are automatically captured - you will see [CHART_IMAGE_SAVED] in the result -`) - log.Printf("📊 [AGENT-BLOCK] Added data analysis guidelines to system prompt") - } - - // Add structured data context so LLM knows exactly what data is available - dataContextSection := e.buildDataContext(inputs) - if dataContextSection != "" { - systemPromptBuilder.WriteString(dataContextSection) - } - - // Build the final system prompt - finalSystemPrompt := systemPromptBuilder.String() - log.Printf("🔍 [AGENT-BLOCK] System prompt built (%d chars, %d tools enabled)", len(finalSystemPrompt), len(config.EnabledTools)) - - messages = append(messages, map[string]any{ - "role": "system", - "content": finalSystemPrompt, - }) - - // Add user prompt with potential image attachments - userPrompt := interpolateTemplate(config.UserPrompt, inputs) - log.Printf("🔍 [AGENT-BLOCK] User prompt (interpolated): %s", userPrompt) - - // Inject data file content into user prompt (dataAttachments already extracted above) - if len(dataAttachments) > 0 { - var dataContext strings.Builder - dataContext.WriteString("\n\n--- Data Files ---\n") - - for _, att := range dataAttachments { - dataContext.WriteString(fmt.Sprintf("\nFile: %s\n", att.Filename)) - dataContext.WriteString(fmt.Sprintf("Type: %s\n", att.MimeType)) - dataContext.WriteString("Content preview (first 100 lines):\n```\n") - dataContext.WriteString(att.Content) - dataContext.WriteString("\n```\n") - } - - userPrompt = userPrompt + dataContext.String() - log.Printf("📊 [AGENT-BLOCK] Injected %d data file(s) into prompt (%d chars added)", - len(dataAttachments), dataContext.Len()) - } - - // Check for image attachments and vision model support - log.Printf("🔍 [AGENT-BLOCK] Checking for image attachments in %d inputs...", len(inputs)) - imageAttachments := e.extractImageAttachments(inputs) - isVisionModel := e.isOpenAIVisionModel(config.Model) - log.Printf("🔍 [AGENT-BLOCK] Found %d image attachments, isVisionModel=%v (model=%s)", len(imageAttachments), isVisionModel, config.Model) - - if len(imageAttachments) > 0 && isVisionModel { - // Build multipart content with text and images - contentParts := []map[string]any{ - { - "type": "text", - "text": userPrompt, - }, - } - - for _, att := range imageAttachments { - imageURL := e.getImageAsBase64DataURL(att.FileID) - if imageURL != "" { - contentParts = append(contentParts, map[string]any{ - "type": "image_url", - "image_url": map[string]any{ - "url": imageURL, - "detail": "auto", - }, - }) - log.Printf("🖼️ [AGENT-BLOCK] Added image attachment: %s", att.Filename) - } - } - - messages = append(messages, map[string]any{ - "role": "user", - "content": contentParts, - }) - } else { - // Standard text message - messages = append(messages, map[string]any{ - "role": "user", - "content": userPrompt, - }) - } - - return messages -} - -// buildDataContext creates a structured data context section for the system prompt -// This helps the LLM understand exactly what data is available from previous blocks -func (e *AgentBlockExecutor) buildDataContext(inputs map[string]any) string { - var builder strings.Builder - var hasContent bool - - // Categorize inputs - var workflowInputs []string - blockOutputs := make(map[string]string) - - for key, value := range inputs { - // Skip internal keys - if strings.HasPrefix(key, "_") || strings.HasPrefix(key, "__") { - continue - } - - // Skip common passthrough keys - if key == "input" || key == "value" || key == "start" { - // Format the main input nicely - if valueStr := formatValueForContext(value); valueStr != "" { - workflowInputs = append(workflowInputs, fmt.Sprintf("- **%s**: %s", key, valueStr)) - hasContent = true - } - continue - } - - // Check if it's a block output (nested map with response key) - if m, ok := value.(map[string]any); ok { - if response, hasResponse := m["response"]; hasResponse { - if responseStr, ok := response.(string); ok && responseStr != "" { - // Truncate very long responses for context - if len(responseStr) > 1500 { - responseStr = responseStr[:1500] + "... [TRUNCATED - full data in user message]" - } - blockOutputs[key] = responseStr - hasContent = true - } - } - } - } - - // Always include current datetime for time-sensitive queries (safety net) - builder.WriteString("\n## CURRENT DATE AND TIME\n") - now := time.Now() - builder.WriteString(fmt.Sprintf("**Today's Date:** %s\n", now.Format("Monday, January 2, 2006"))) - builder.WriteString(fmt.Sprintf("**Current Time:** %s\n", now.Format("3:04 PM MST"))) - builder.WriteString(fmt.Sprintf("**ISO Format:** %s\n\n", now.Format(time.RFC3339))) - builder.WriteString("Use this date when searching for 'today', 'recent', 'latest', or 'current' information.\n\n") - - if !hasContent { - // Still return the datetime even if no other content - return builder.String() - } - - builder.WriteString("## AVAILABLE DATA (Already Resolved)\n") - builder.WriteString("The following data has been collected from previous steps and is ready for use:\n\n") - - // Present workflow inputs - if len(workflowInputs) > 0 { - builder.WriteString("### Direct Inputs\n") - for _, input := range workflowInputs { - builder.WriteString(input + "\n") - } - builder.WriteString("\n") - } - - // Present block outputs - if len(blockOutputs) > 0 { - builder.WriteString("### Data from Previous Blocks\n") - for blockID, response := range blockOutputs { - builder.WriteString(fmt.Sprintf("**From `%s`:**\n", blockID)) - builder.WriteString("```\n") - builder.WriteString(response) - builder.WriteString("\n```\n\n") - } - } - - builder.WriteString("Use this data directly - DO NOT ask for it or claim you don't have it.\n\n") - - return builder.String() -} - -// formatValueForContext formats a value for display in the data context -func formatValueForContext(value any) string { - switch v := value.(type) { - case string: - if len(v) > 500 { - return fmt.Sprintf("%q... [%d chars total]", v[:500], len(v)) - } - if len(v) > 100 { - return fmt.Sprintf("%q", v[:100]+"...") - } - return fmt.Sprintf("%q", v) - case float64: - if v == float64(int(v)) { - return fmt.Sprintf("%d", int(v)) - } - return fmt.Sprintf("%g", v) - case int: - return fmt.Sprintf("%d", v) - case bool: - return fmt.Sprintf("%t", v) - case map[string]any: - // Check for file reference - if fileID, ok := v["file_id"].(string); ok && fileID != "" { - filename, _ := v["filename"].(string) - return fmt.Sprintf("[File: %s]", filename) - } - // For other maps, JSON encode briefly - jsonBytes, err := json.Marshal(v) - if err != nil { - return "[complex object]" - } - if len(jsonBytes) > 200 { - return string(jsonBytes[:200]) + "..." - } - return string(jsonBytes) - default: - return "" - } -} - -// FileAttachment represents an image or file attachment -type FileAttachment struct { - FileID string `json:"file_id"` - Filename string `json:"filename"` - MimeType string `json:"mime_type"` - Type string `json:"type"` // "image", "document", "audio", "data" -} - -// extractImageAttachments extracts image attachments from inputs -func (e *AgentBlockExecutor) extractImageAttachments(inputs map[string]any) []FileAttachment { - var attachments []FileAttachment - - // Helper to extract attachment from a map - extractFromMap := func(attMap map[string]any) *FileAttachment { - att := FileAttachment{} - if v, ok := attMap["file_id"].(string); ok { - att.FileID = v - } else if v, ok := attMap["fileId"].(string); ok { - att.FileID = v - } - if v, ok := attMap["filename"].(string); ok { - att.Filename = v - } - if v, ok := attMap["mime_type"].(string); ok { - att.MimeType = v - } else if v, ok := attMap["mimeType"].(string); ok { - att.MimeType = v - } - if v, ok := attMap["type"].(string); ok { - att.Type = v - } - - // Only include images - if att.FileID != "" && (att.Type == "image" || strings.HasPrefix(att.MimeType, "image/")) { - return &att - } - return nil - } - - // Check for "_attachments" or "attachments" in inputs - var rawAttachments []interface{} - if att, ok := inputs["_attachments"].([]interface{}); ok { - rawAttachments = att - } else if att, ok := inputs["attachments"].([]interface{}); ok { - rawAttachments = att - } else if att, ok := inputs["images"].([]interface{}); ok { - rawAttachments = att - } - - for _, raw := range rawAttachments { - if attMap, ok := raw.(map[string]interface{}); ok { - if att := extractFromMap(attMap); att != nil { - attachments = append(attachments, *att) - } - } - } - - // Also check for single image file_id - if fileID, ok := inputs["image_file_id"].(string); ok && fileID != "" { - attachments = append(attachments, FileAttachment{ - FileID: fileID, - Type: "image", - }) - } - - // Check all inputs for file references that are images (e.g., from Start block) - for key, value := range inputs { - // Skip internal keys - if strings.HasPrefix(key, "_") || key == "attachments" || key == "images" { - continue - } - - // Try map[string]any first - if attMap, ok := value.(map[string]any); ok { - log.Printf("🔍 [AGENT-BLOCK] Input '%s' is map[string]any: %+v", key, attMap) - if att := extractFromMap(attMap); att != nil { - log.Printf("🖼️ [AGENT-BLOCK] Found image file reference in input '%s': %s", key, att.Filename) - attachments = append(attachments, *att) - } - } else if attMap, ok := value.(map[string]interface{}); ok { - // Try map[string]interface{} (JSON unmarshaling often produces this) - log.Printf("🔍 [AGENT-BLOCK] Input '%s' is map[string]interface{}: %+v", key, attMap) - // Convert to map[string]any - converted := make(map[string]any) - for k, v := range attMap { - converted[k] = v - } - if att := extractFromMap(converted); att != nil { - log.Printf("🖼️ [AGENT-BLOCK] Found image file reference in input '%s': %s", key, att.Filename) - attachments = append(attachments, *att) - } - } else if value != nil { - log.Printf("🔍 [AGENT-BLOCK] Input '%s' has type %T (not a map)", key, value) - } - } - - return attachments -} - -// DataFileAttachment represents a data file attachment (CSV, JSON, Excel, etc.) -type DataFileAttachment struct { - FileID string `json:"file_id"` - Filename string `json:"filename"` - MimeType string `json:"mime_type"` - Content string `json:"content"` // Preview content (first ~100 lines) -} - -// extractDataFileAttachments extracts data file attachments from inputs -func (e *AgentBlockExecutor) extractDataFileAttachments(inputs map[string]any) []DataFileAttachment { - var attachments []DataFileAttachment - - // Data file MIME types - dataTypes := map[string]bool{ - "text/csv": true, - "application/json": true, - "application/vnd.ms-excel": true, - "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet": true, - "text/plain": true, - "text/tab-separated-values": true, - } - - // Helper to check if a file is a data file - isDataFile := func(mimeType, filename string) bool { - if dataTypes[mimeType] { - return true - } - // Check by extension - ext := strings.ToLower(filepath.Ext(filename)) - return ext == ".csv" || ext == ".json" || ext == ".xlsx" || - ext == ".xls" || ext == ".tsv" || ext == ".txt" - } - - // Check all inputs for file references - for key, value := range inputs { - if strings.HasPrefix(key, "_") { - continue - } - - var attMap map[string]any - if m, ok := value.(map[string]any); ok { - attMap = m - } else if m, ok := value.(map[string]interface{}); ok { - attMap = make(map[string]any) - for k, v := range m { - attMap[k] = v - } - } - - if attMap == nil { - continue - } - - fileID, _ := attMap["file_id"].(string) - if fileID == "" { - fileID, _ = attMap["fileId"].(string) - } - filename, _ := attMap["filename"].(string) - mimeType, _ := attMap["mime_type"].(string) - if mimeType == "" { - mimeType, _ = attMap["mimeType"].(string) - } - - if fileID != "" && isDataFile(mimeType, filename) { - // Read file content from cache - content := e.readDataFileContent(fileID, filename) - if content != "" { - attachments = append(attachments, DataFileAttachment{ - FileID: fileID, - Filename: filename, - MimeType: mimeType, - Content: content, - }) - log.Printf("📊 [AGENT-BLOCK] Found data file in input '%s': %s (%d chars)", key, filename, len(content)) - } - } - } - - return attachments -} - -// readDataFileContent reads content from a data file (CSV, JSON, etc.) -// Returns a preview (first ~100 lines) suitable for LLM context -func (e *AgentBlockExecutor) readDataFileContent(fileID, filename string) string { - fileCacheService := filecache.GetService() - file, found := fileCacheService.Get(fileID) - if !found { - log.Printf("⚠️ [AGENT-BLOCK] Data file not found in cache: %s", fileID) - return "" - } - - if file.FilePath == "" { - log.Printf("⚠️ [AGENT-BLOCK] Data file path not available: %s", fileID) - return "" - } - - // Read file content - content, err := os.ReadFile(file.FilePath) - if err != nil { - log.Printf("❌ [AGENT-BLOCK] Failed to read data file: %v", err) - return "" - } - - // Convert to string and limit to first 100 lines for context - lines := strings.Split(string(content), "\n") - maxLines := 100 - if len(lines) > maxLines { - lines = lines[:maxLines] - } - - preview := strings.Join(lines, "\n") - log.Printf("✅ [AGENT-BLOCK] Read data file %s (%d lines, %d bytes)", - filename, len(lines), len(preview)) - - return preview -} - -// Artifact represents a generated artifact (chart, image, etc.) from tool execution -type Artifact struct { - Type string `json:"type"` // "chart", "image", "file" - Format string `json:"format"` // "png", "jpeg", "svg", etc. - Data string `json:"data"` // Base64 encoded data - Title string `json:"title"` // Optional title/description -} - -// extractArtifactsFromToolCalls extracts all artifacts (charts, images) from tool call results -// This provides a consistent format for API consumers to access generated visualizations -func (e *AgentBlockExecutor) extractArtifactsFromToolCalls(toolCalls []models.ToolCallRecord) []Artifact { - var artifacts []Artifact - - for _, tc := range toolCalls { - if tc.Error != "" || tc.Result == "" { - continue - } - - // Parse tool result as JSON - var resultData map[string]any - if err := json.Unmarshal([]byte(tc.Result), &resultData); err != nil { - continue - } - - // Look for charts/images in common E2B response formats - // E2B analyze_data returns: {"plots": [{"data": "base64...", "type": "png"}], ...} - if plots, ok := resultData["plots"].([]interface{}); ok { - for i, p := range plots { - if plot, ok := p.(map[string]interface{}); ok { - data, _ := plot["data"].(string) - format, _ := plot["type"].(string) - if format == "" { - format = "png" - } - if data != "" && len(data) > 100 { - artifacts = append(artifacts, Artifact{ - Type: "chart", - Format: format, - Data: data, - Title: fmt.Sprintf("Chart %d from %s", i+1, tc.Name), - }) - } - } - } - } - - // Also check for single image/plot fields - for _, key := range []string{"image", "plot", "chart", "figure", "png", "jpeg"} { - if data, ok := resultData[key].(string); ok && len(data) > 100 { - // Determine format from key or data URI - format := "png" - if key == "jpeg" { - format = "jpeg" - } - if strings.HasPrefix(data, "data:image/") { - // Extract format from data URI - if strings.Contains(data, "jpeg") || strings.Contains(data, "jpg") { - format = "jpeg" - } else if strings.Contains(data, "svg") { - format = "svg" - } - } - - artifacts = append(artifacts, Artifact{ - Type: "chart", - Format: format, - Data: data, - Title: fmt.Sprintf("Generated %s from %s", key, tc.Name), - }) - } - } - - // Check for base64_images array (another common E2B format) - if images, ok := resultData["base64_images"].([]interface{}); ok { - for i, img := range images { - if imgData, ok := img.(string); ok && len(imgData) > 100 { - artifacts = append(artifacts, Artifact{ - Type: "chart", - Format: "png", - Data: imgData, - Title: fmt.Sprintf("Image %d from %s", i+1, tc.Name), - }) - } - } - } - } - - log.Printf("📊 [AGENT-BLOCK] Extracted %d artifacts from tool calls", len(artifacts)) - return artifacts -} - -// GeneratedFile represents a file generated by a tool (PDF, document, etc.) -type GeneratedFile struct { - FileID string `json:"file_id"` - Filename string `json:"filename"` - DownloadURL string `json:"download_url"` - AccessCode string `json:"access_code,omitempty"` - Size int64 `json:"size,omitempty"` - MimeType string `json:"mime_type,omitempty"` -} - -// extractGeneratedFilesFromToolCalls extracts file references (PDFs, documents) from tool call results -// This makes download URLs available to subsequent blocks -func (e *AgentBlockExecutor) extractGeneratedFilesFromToolCalls(toolCalls []models.ToolCallRecord) []GeneratedFile { - var files []GeneratedFile - - for _, tc := range toolCalls { - if tc.Error != "" || tc.Result == "" { - continue - } - - // Parse tool result as JSON - var resultData map[string]any - if err := json.Unmarshal([]byte(tc.Result), &resultData); err != nil { - continue - } - - // Look for file reference fields - fileRef := GeneratedFile{} - - if v, ok := resultData["file_id"].(string); ok && v != "" { - fileRef.FileID = v - } - if v, ok := resultData["filename"].(string); ok && v != "" { - fileRef.Filename = v - } - if v, ok := resultData["download_url"].(string); ok && v != "" { - fileRef.DownloadURL = v - } - if v, ok := resultData["access_code"].(string); ok && v != "" { - fileRef.AccessCode = v - } - if v, ok := resultData["size"].(float64); ok { - fileRef.Size = int64(v) - } - if v, ok := resultData["mime_type"].(string); ok && v != "" { - fileRef.MimeType = v - } - - // Only add if we have meaningful file reference data - if fileRef.FileID != "" || fileRef.DownloadURL != "" { - files = append(files, fileRef) - log.Printf("📄 [AGENT-BLOCK] Extracted file reference: %s (url: %s)", fileRef.Filename, fileRef.DownloadURL) - } - } - - return files -} - -// sanitizeToolResultForLLM removes base64 image data from tool results -// Base64 images are huge and useless to the LLM as text - it can't "see" them -// Instead, we replace them with a placeholder indicating a chart was generated -func (e *AgentBlockExecutor) sanitizeToolResultForLLM(result string) string { - if len(result) < 1000 { - return result // Small results don't need sanitization - } - - chartsGenerated := false - - // Pattern to match base64 image data (PNG, JPEG, etc.) - // Matches: "data:image/png;base64,..." or just long base64 strings - base64Pattern := regexp.MustCompile(`"data:image/[^;]+;base64,[A-Za-z0-9+/=]{100,}"`) - if base64Pattern.MatchString(result) { - chartsGenerated = true - } - sanitized := base64Pattern.ReplaceAllString(result, `"[CHART_IMAGE_SAVED]"`) - - // Also match standalone base64 blocks that might not have data URI prefix - // Look for very long strings of base64 characters (>500 chars) - longBase64Pattern := regexp.MustCompile(`"[A-Za-z0-9+/=]{500,}"`) - if longBase64Pattern.MatchString(sanitized) { - chartsGenerated = true - } - sanitized = longBase64Pattern.ReplaceAllString(sanitized, `"[CHART_IMAGE_SAVED]"`) - - // Also handle base64 in "image" or "plot" fields common in E2B responses - imageFieldPattern := regexp.MustCompile(`"(image|plot|chart|png|jpeg|figure)":\s*"[A-Za-z0-9+/=]{100,}"`) - sanitized = imageFieldPattern.ReplaceAllString(sanitized, `"$1": "[CHART_IMAGE_SAVED]"`) - - // Truncate if still too long (max 20KB for tool results) - maxLen := 20000 - if len(sanitized) > maxLen { - sanitized = sanitized[:maxLen] + "\n... [TRUNCATED - Full result too large for LLM context]" - } - - // Add clear instruction when charts were generated - if chartsGenerated { - sanitized = sanitized + "\n\n[CHARTS SUCCESSFULLY GENERATED AND SAVED. Do NOT call analyze_data again. Provide your final summary/insights based on the analysis output above.]" - } - - originalLen := len(result) - newLen := len(sanitized) - if originalLen != newLen { - log.Printf("🧹 [AGENT-BLOCK] Sanitized tool result: %d -> %d chars (removed base64/large data, charts=%v)", - originalLen, newLen, chartsGenerated) - } - - return sanitized -} - -// extractChartsFromResult extracts base64 chart images from tool results -// This is used to collect charts for auto-injection into Discord/Slack messages -func (e *AgentBlockExecutor) extractChartsFromResult(result string) []string { - var charts []string - - // Try to parse as JSON - var resultData map[string]any - if err := json.Unmarshal([]byte(result), &resultData); err != nil { - return charts - } - - // Look for plots array (E2B analyze_data format) - if plots, ok := resultData["plots"].([]interface{}); ok { - for _, p := range plots { - if plot, ok := p.(map[string]interface{}); ok { - // Check for "data" field containing base64 - if data, ok := plot["data"].(string); ok && len(data) > 100 { - charts = append(charts, data) - } - } - } - } - - // Also check for single "image" or "chart" field - for _, key := range []string{"image", "chart", "plot", "figure"} { - if data, ok := resultData[key].(string); ok && len(data) > 100 { - charts = append(charts, data) - } - } - - return charts -} - -// extractChartsFromInputs extracts chart images from previous block artifacts -// This allows downstream blocks (like Discord Publisher) to access charts generated upstream -func (e *AgentBlockExecutor) extractChartsFromInputs(inputs map[string]any) []string { - var charts []string - - // Helper function to extract charts from artifacts slice - extractFromArtifacts := func(artifacts []Artifact) { - for _, artifact := range artifacts { - if artifact.Type == "chart" && artifact.Data != "" && len(artifact.Data) > 100 { - charts = append(charts, artifact.Data) - log.Printf("🖼️ [AGENT-BLOCK] Found chart artifact: %s (format: %s, %d bytes)", - artifact.Title, artifact.Format, len(artifact.Data)) - } - } - } - - // Helper to try converting interface to []Artifact - tryConvertArtifacts := func(v interface{}) bool { - // Direct type assertion for []Artifact - if artifacts, ok := v.([]Artifact); ok { - extractFromArtifacts(artifacts) - return true - } - // Try []execution.Artifact (same type, different reference) - if artifacts, ok := v.([]interface{}); ok { - for _, a := range artifacts { - if artifact, ok := a.(Artifact); ok { - if artifact.Type == "chart" && artifact.Data != "" && len(artifact.Data) > 100 { - charts = append(charts, artifact.Data) - } - } else if artifactMap, ok := a.(map[string]interface{}); ok { - // Handle map representation of artifact - artifactType, _ := artifactMap["type"].(string) - artifactData, _ := artifactMap["data"].(string) - if artifactType == "chart" && artifactData != "" && len(artifactData) > 100 { - charts = append(charts, artifactData) - } - } - } - return true - } - return false - } - - // 1. Check direct "artifacts" key in inputs - if artifacts, ok := inputs["artifacts"]; ok { - tryConvertArtifacts(artifacts) - } - - // 2. Check previous block outputs (e.g., inputs["data-analyzer"]["artifacts"]) - // These are stored as map[string]any with block IDs as keys - for key, value := range inputs { - // Skip non-block keys - if key == "artifacts" || key == "input" || key == "value" || key == "start" || - strings.HasPrefix(key, "_") || strings.HasPrefix(key, "__") { - continue - } - - // Check if value is a map (previous block output) - if blockOutput, ok := value.(map[string]any); ok { - // Look for artifacts in this block's output - if artifacts, ok := blockOutput["artifacts"]; ok { - tryConvertArtifacts(artifacts) - } - - // Also check for nested output.artifacts - if output, ok := blockOutput["output"].(map[string]any); ok { - if artifacts, ok := output["artifacts"]; ok { - tryConvertArtifacts(artifacts) - } - } - - // Check toolCalls for chart results (some tools return charts in result) - if toolCalls, ok := blockOutput["toolCalls"].([]models.ToolCallRecord); ok { - for _, tc := range toolCalls { - if tc.Result != "" { - extractedCharts := e.extractChartsFromResult(tc.Result) - charts = append(charts, extractedCharts...) - } - } - } - // Also try interface{} slice for toolCalls - if toolCalls, ok := blockOutput["toolCalls"].([]interface{}); ok { - for _, tc := range toolCalls { - if tcMap, ok := tc.(map[string]interface{}); ok { - if result, ok := tcMap["Result"].(string); ok && result != "" { - extractedCharts := e.extractChartsFromResult(result) - charts = append(charts, extractedCharts...) - } - } - } - } - } - } - - return charts -} - -// NOTE: detectToolsFromContext was removed - blocks now only use explicitly configured tools -// This ensures predictable behavior where users must configure enabledTools in the block settings - -// extractToolResultsForDownstream parses tool call results and extracts data for downstream blocks -// This solves the problem where tool output was buried in toolCalls and not accessible to next blocks -func (e *AgentBlockExecutor) extractToolResultsForDownstream(toolCalls []models.ToolCallRecord) map[string]any { - results := make(map[string]any) - - for _, tc := range toolCalls { - if tc.Error != "" || tc.Result == "" { - continue - } - - // Try to parse result as JSON - var parsed map[string]any - if err := json.Unmarshal([]byte(tc.Result), &parsed); err != nil { - // Not JSON - store as raw string - results[tc.Name] = tc.Result - continue - } - - // Store parsed result under tool name - results[tc.Name] = parsed - - log.Printf("📦 [AGENT-BLOCK] Extracted tool result for '%s': %d fields", tc.Name, len(parsed)) - } - - return results -} - -// isOpenAIVisionModel checks if the model is an OpenAI vision-capable model -func (e *AgentBlockExecutor) isOpenAIVisionModel(modelID string) bool { - // OpenAI vision-capable models - visionModels := []string{ - "gpt-4o", - "gpt-4o-mini", - "gpt-4-turbo", - "gpt-4-vision-preview", - "gpt-4-turbo-preview", - "gpt-5", // GPT-5 series (all variants support vision) - "gpt-5.1", // GPT-5.1 series - "o1", // o1 models - "o1-preview", - "o1-mini", - "o3", // o3 models - "o3-mini", - "o4-mini", // o4-mini (if released) - } - - modelLower := strings.ToLower(modelID) - for _, vm := range visionModels { - if strings.Contains(modelLower, vm) { - return true - } - } - - // Also check model aliases that might map to vision models - // Common patterns: any 4o variant or 5.x variant - if strings.Contains(modelLower, "gpt-4o") || strings.Contains(modelLower, "4o") || - strings.Contains(modelLower, "gpt-5") || strings.Contains(modelLower, "5.1") { - return true - } - - return false -} - -// getImageAsBase64DataURL converts an image file to a base64 data URL -func (e *AgentBlockExecutor) getImageAsBase64DataURL(fileID string) string { - // Get file from cache - fileCacheService := filecache.GetService() - file, found := fileCacheService.Get(fileID) - if !found { - log.Printf("⚠️ [AGENT-BLOCK] Image file not found: %s", fileID) - return "" - } - - // Verify it's an image - if !strings.HasPrefix(file.MimeType, "image/") { - log.Printf("⚠️ [AGENT-BLOCK] File is not an image: %s (%s)", fileID, file.MimeType) - return "" - } - - // Read image from disk - if file.FilePath == "" { - log.Printf("⚠️ [AGENT-BLOCK] Image file path not available: %s", fileID) - return "" - } - - imageData, err := os.ReadFile(file.FilePath) - if err != nil { - log.Printf("❌ [AGENT-BLOCK] Failed to read image file: %v", err) - return "" - } - - // Convert to base64 data URL - base64Image := base64.StdEncoding.EncodeToString(imageData) - dataURL := fmt.Sprintf("data:%s;base64,%s", file.MimeType, base64Image) - - log.Printf("✅ [AGENT-BLOCK] Converted image to base64 (%d bytes)", len(imageData)) - return dataURL -} - -// getToolDescriptions returns human-readable descriptions of enabled tools -// This helps the LLM understand how to use the tools, including key parameters -func (e *AgentBlockExecutor) getToolDescriptions(enabledTools []string) []string { - if len(enabledTools) == 0 { - return nil - } - - enabledSet := make(map[string]bool) - for _, name := range enabledTools { - enabledSet[name] = true - } - - var descriptions []string - allTools := e.toolRegistry.List() - - for _, tool := range allTools { - if fn, ok := tool["function"].(map[string]interface{}); ok { - name, _ := fn["name"].(string) - if !enabledSet[name] { - continue - } - - desc, _ := fn["description"].(string) - // Truncate long descriptions but keep key info - if len(desc) > 500 { - desc = desc[:500] + "..." - } - - // Build a concise tool summary - var sb strings.Builder - sb.WriteString(fmt.Sprintf("### %s\n", name)) - if desc != "" { - sb.WriteString(fmt.Sprintf("%s\n", desc)) - } - - // Extract key parameters to highlight - if params, ok := fn["parameters"].(map[string]interface{}); ok { - if props, ok := params["properties"].(map[string]interface{}); ok { - var keyParams []string - for paramName, paramDef := range props { - // Skip internal parameters - if strings.HasPrefix(paramName, "_") || paramName == "credential_id" || paramName == "api_key" { - continue - } - if paramMap, ok := paramDef.(map[string]interface{}); ok { - paramDesc, _ := paramMap["description"].(string) - // Highlight important params like file_url - if paramName == "file_url" || paramName == "download_url" { - keyParams = append(keyParams, fmt.Sprintf(" - **%s**: %s", paramName, paramDesc)) - } else if len(keyParams) < 5 { // Limit to 5 params - shortDesc := paramDesc - if len(shortDesc) > 100 { - shortDesc = shortDesc[:100] + "..." - } - keyParams = append(keyParams, fmt.Sprintf(" - %s: %s", paramName, shortDesc)) - } - } - } - if len(keyParams) > 0 { - sb.WriteString("Key parameters:\n") - for _, p := range keyParams { - sb.WriteString(p + "\n") - } - } - } - } - - descriptions = append(descriptions, sb.String()) - } - } - - return descriptions -} - -// filterTools returns only the tools that are enabled for this block -func (e *AgentBlockExecutor) filterTools(enabledTools []string) []map[string]interface{} { - if len(enabledTools) == 0 { - // No tools enabled for this block - return nil - } - - // Get all available tools - allTools := e.toolRegistry.List() - - // Filter to only enabled tools - var filtered []map[string]interface{} - enabledSet := make(map[string]bool) - for _, name := range enabledTools { - enabledSet[name] = true - } - - for _, tool := range allTools { - if fn, ok := tool["function"].(map[string]interface{}); ok { - if name, ok := fn["name"].(string); ok { - if enabledSet[name] { - filtered = append(filtered, tool) - } - } - } - } - - return filtered -} - -// LLMResponse represents the response from the LLM -type LLMResponse struct { - Content string - ToolCalls []map[string]any - FinishReason string // "stop", "tool_calls", "end_turn", etc. - InputTokens int - OutputTokens int -} - -// callLLM makes a streaming call to the LLM (required for ClaraVerse API compatibility) -func (e *AgentBlockExecutor) callLLM( - ctx context.Context, - provider *models.Provider, - modelID string, - messages []map[string]any, - tools []map[string]interface{}, - temperature float64, -) (*LLMResponse, error) { - return e.callLLMWithSchema(ctx, provider, modelID, messages, tools, temperature, nil) -} - -// callLLMWithSchema calls the LLM with optional native structured output support -func (e *AgentBlockExecutor) callLLMWithSchema( - ctx context.Context, - provider *models.Provider, - modelID string, - messages []map[string]any, - tools []map[string]interface{}, - temperature float64, - outputSchema *models.JSONSchema, -) (*LLMResponse, error) { - - // Detect provider type by base URL to avoid sending incompatible parameters - // OpenAI's API is strict and rejects unknown parameters with 400 errors - isOpenAI := strings.Contains(strings.ToLower(provider.BaseURL), "openai.com") - isOpenRouter := strings.Contains(strings.ToLower(provider.BaseURL), "openrouter.ai") - isGLM := strings.Contains(strings.ToLower(provider.BaseURL), "bigmodel.cn") || - strings.Contains(strings.ToLower(provider.Name), "glm") || - strings.Contains(strings.ToLower(modelID), "glm") - - // Check if the model supports native structured output - // OpenAI GPT-4o models and newer support json_schema response_format - supportsStructuredOutput := (isOpenAI || isOpenRouter) && outputSchema != nil && len(tools) == 0 - - // Build request body - use streaming for better compatibility with ClaraVerse API - requestBody := map[string]interface{}{ - "model": modelID, - "messages": messages, - "temperature": temperature, - "stream": true, // Use streaming - ClaraVerse API works better with streaming - } - - // Use correct token limit parameter based on provider - // OpenAI newer models (GPT-4o, o1, etc.) require max_completion_tokens instead of max_tokens - // Most models support 65K+ output tokens, so we use 32768 as a safe high limit - if isOpenAI { - requestBody["max_completion_tokens"] = 32768 - } else { - requestBody["max_tokens"] = 32768 - } - - // Add native structured output if supported and no tools are being used - // Note: Can't use response_format with tools - they're mutually exclusive - if supportsStructuredOutput { - // Convert JSONSchema to OpenAI's response_format structure - schemaMap := e.jsonSchemaToMap(outputSchema) - requestBody["response_format"] = map[string]interface{}{ - "type": "json_schema", - "json_schema": map[string]interface{}{ - "name": "structured_output", - "strict": true, - "schema": schemaMap, - }, - } - log.Printf("📋 [AGENT-BLOCK] Using native structured output (json_schema response_format)") - } - - // Add provider-specific parameters only where supported - if isGLM { - // GLM-specific parameters to disable reasoning mode - requestBody["think"] = false - requestBody["do_sample"] = true - requestBody["top_p"] = 0.95 - } else if !isOpenAI && !isOpenRouter { - // For other non-OpenAI providers, try common parameters that might be supported - // These providers typically ignore unknown parameters - requestBody["enable_thinking"] = false - } - // Note: OpenAI gets no extra parameters - their API is strict about unknown params - - // Only include tools if non-empty - if len(tools) > 0 { - requestBody["tools"] = tools - } - - bodyBytes, err := json.Marshal(requestBody) - if err != nil { - return nil, fmt.Errorf("failed to marshal request: %w", err) - } - - // Create request - endpoint := strings.TrimSuffix(provider.BaseURL, "/") + "/chat/completions" - log.Printf("🌐 [AGENT-BLOCK] Calling LLM: %s (model: %s, streaming: true)", endpoint, modelID) - - req, err := http.NewRequestWithContext(ctx, "POST", endpoint, bytes.NewReader(bodyBytes)) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Authorization", "Bearer "+provider.APIKey) - - // Execute request - resp, err := e.httpClient.Do(req) - if err != nil { - // Classify network/connection errors for retry logic - return nil, ClassifyError(err) - } - defer resp.Body.Close() - - if resp.StatusCode != http.StatusOK { - body, _ := io.ReadAll(resp.Body) - // Classify HTTP errors for retry logic (429, 5xx are retryable) - return nil, ClassifyHTTPError(resp.StatusCode, string(body)) - } - - // Process SSE stream and accumulate response - return e.processStreamResponse(resp.Body) -} - -// callLLMWithRetry wraps callLLM with retry logic for transient errors -// Returns the response, retry attempts history, and any final error -func (e *AgentBlockExecutor) callLLMWithRetry( - ctx context.Context, - provider *models.Provider, - modelID string, - messages []map[string]any, - tools []map[string]interface{}, - temperature float64, - retryPolicy *models.RetryPolicy, -) (*LLMResponse, []models.RetryAttempt, error) { - return e.callLLMWithRetryAndSchema(ctx, provider, modelID, messages, tools, temperature, retryPolicy, nil) -} - -// callLLMWithRetryAndSchema wraps callLLMWithSchema with retry logic for transient errors -// Returns the response, retry attempts history, and any final error -func (e *AgentBlockExecutor) callLLMWithRetryAndSchema( - ctx context.Context, - provider *models.Provider, - modelID string, - messages []map[string]any, - tools []map[string]interface{}, - temperature float64, - retryPolicy *models.RetryPolicy, - outputSchema *models.JSONSchema, -) (*LLMResponse, []models.RetryAttempt, error) { - - // Use default policy if not specified - if retryPolicy == nil { - retryPolicy = models.DefaultRetryPolicy() - } - - // Create backoff calculator - backoff := NewBackoffCalculator( - retryPolicy.InitialDelay, - retryPolicy.MaxDelay, - retryPolicy.BackoffMultiplier, - retryPolicy.JitterPercent, - ) - - var attempts []models.RetryAttempt - - for attempt := 0; attempt <= retryPolicy.MaxRetries; attempt++ { - attemptStart := time.Now() - - // Make the LLM call with optional schema - response, err := e.callLLMWithSchema(ctx, provider, modelID, messages, tools, temperature, outputSchema) - attemptDuration := time.Since(attemptStart).Milliseconds() - - if err == nil { - // Success! - if attempt > 0 { - log.Printf("✅ [AGENT-BLOCK] LLM call succeeded on retry attempt %d", attempt) - } - return response, attempts, nil - } - - // Classify the error (may already be classified from callLLM) - var execErr *ExecutionError - if e, ok := err.(*ExecutionError); ok { - execErr = e - } else { - execErr = ClassifyError(err) - } - - // Determine error type string for logging and tracking - errorType := "unknown" - if execErr.StatusCode == 429 { - errorType = "rate_limit" - } else if execErr.StatusCode >= 500 { - errorType = "server_error" - } else if strings.Contains(strings.ToLower(execErr.Message), "timeout") || - strings.Contains(strings.ToLower(execErr.Message), "deadline") { - errorType = "timeout" - } else if strings.Contains(strings.ToLower(execErr.Message), "network") || - strings.Contains(strings.ToLower(execErr.Message), "connection") { - errorType = "network_error" - } - - // Record this attempt - attempts = append(attempts, models.RetryAttempt{ - Attempt: attempt, - Error: execErr.Message, - ErrorType: errorType, - Timestamp: attemptStart, - Duration: attemptDuration, - }) - - // Check if we should retry - if attempt < retryPolicy.MaxRetries && ShouldRetry(execErr, retryPolicy.RetryOn) { - delay := backoff.NextDelay(attempt) - - // Use RetryAfter if available and longer (e.g., from 429 response) - if execErr.RetryAfter > 0 { - retryAfterDelay := time.Duration(execErr.RetryAfter) * time.Second - if retryAfterDelay > delay { - delay = retryAfterDelay - } - } - - log.Printf("🔄 [AGENT-BLOCK] LLM call failed (attempt %d/%d): %s [%s]. Retrying in %v", - attempt+1, retryPolicy.MaxRetries+1, execErr.Message, errorType, delay) - - // Wait before retry (respecting context cancellation) - select { - case <-time.After(delay): - // Continue to next attempt - case <-ctx.Done(): - return nil, attempts, &ExecutionError{ - Category: ErrorCategoryTransient, - Message: "Context cancelled during retry wait", - Retryable: false, - Cause: ctx.Err(), - } - } - } else { - // Not retryable or max retries exceeded - if attempt >= retryPolicy.MaxRetries { - log.Printf("❌ [AGENT-BLOCK] LLM call failed after %d attempt(s): %s [%s] (max retries exceeded)", - attempt+1, execErr.Message, errorType) - } else { - log.Printf("❌ [AGENT-BLOCK] LLM call failed: %s [%s] (not retryable)", - execErr.Message, errorType) - } - return nil, attempts, execErr - } - } - - // Should not reach here, but safety fallback - return nil, attempts, &ExecutionError{ - Category: ErrorCategoryUnknown, - Message: "Max retries exceeded", - Retryable: false, - } -} - -// processStreamResponse processes SSE stream and returns accumulated response -func (e *AgentBlockExecutor) processStreamResponse(reader io.Reader) (*LLMResponse, error) { - response := &LLMResponse{} - var contentBuilder strings.Builder - - // Track tool calls by index to accumulate streaming arguments - toolCallsMap := make(map[int]*toolCallAccumulator) - - scanner := bufio.NewScanner(reader) - for scanner.Scan() { - line := scanner.Text() - if !strings.HasPrefix(line, "data: ") { - continue - } - - data := strings.TrimPrefix(line, "data: ") - if data == "[DONE]" { - break - } - - var chunk map[string]interface{} - if err := json.Unmarshal([]byte(data), &chunk); err != nil { - continue - } - - choices, ok := chunk["choices"].([]interface{}) - if !ok || len(choices) == 0 { - continue - } - - choice := choices[0].(map[string]interface{}) - - // Capture finish_reason when available (usually in the final chunk) - if finishReason, ok := choice["finish_reason"].(string); ok && finishReason != "" { - response.FinishReason = finishReason - } - - delta, ok := choice["delta"].(map[string]interface{}) - if !ok { - continue - } - - // Accumulate content chunks - if content, ok := delta["content"].(string); ok { - contentBuilder.WriteString(content) - } - - // Accumulate tool calls - if toolCallsData, ok := delta["tool_calls"].([]interface{}); ok { - for _, tc := range toolCallsData { - toolCallChunk := tc.(map[string]interface{}) - - // Get tool call index - var index int - if idx, ok := toolCallChunk["index"].(float64); ok { - index = int(idx) - } - - // Initialize accumulator if needed - if _, exists := toolCallsMap[index]; !exists { - toolCallsMap[index] = &toolCallAccumulator{} - } - - acc := toolCallsMap[index] - - // Accumulate fields - if id, ok := toolCallChunk["id"].(string); ok { - acc.ID = id - } - if typ, ok := toolCallChunk["type"].(string); ok { - acc.Type = typ - } - if function, ok := toolCallChunk["function"].(map[string]interface{}); ok { - if name, ok := function["name"].(string); ok { - acc.Name = name - } - if args, ok := function["arguments"].(string); ok { - acc.Arguments.WriteString(args) - } - } - } - } - - // Extract token usage from chunk (some APIs include it in each chunk) - if usage, ok := chunk["usage"].(map[string]interface{}); ok { - if pt, ok := usage["prompt_tokens"].(float64); ok { - response.InputTokens = int(pt) - } - if ct, ok := usage["completion_tokens"].(float64); ok { - response.OutputTokens = int(ct) - } - } - } - - if err := scanner.Err(); err != nil { - return nil, fmt.Errorf("error reading stream: %w", err) - } - - // Set accumulated content - response.Content = contentBuilder.String() - - // Convert accumulated tool calls to response format - for _, acc := range toolCallsMap { - if acc.Name != "" { - toolCall := map[string]any{ - "id": acc.ID, - "type": acc.Type, - "function": map[string]any{ - "name": acc.Name, - "arguments": acc.Arguments.String(), - }, - } - response.ToolCalls = append(response.ToolCalls, toolCall) - } - } - - log.Printf("✅ [AGENT-BLOCK] Stream processed: content=%d chars, toolCalls=%d, finishReason=%s", - len(response.Content), len(response.ToolCalls), response.FinishReason) - - return response, nil -} - -// toolCallAccumulator accumulates streaming tool call data -type toolCallAccumulator struct { - ID string - Type string - Name string - Arguments strings.Builder -} - -// executeToolCall executes a single tool call and returns the record -func (e *AgentBlockExecutor) executeToolCall(toolCall map[string]any, blockInputs map[string]any, dataFiles []DataFileAttachment, generatedCharts []string, userID string, credentials []string) models.ToolCallRecord { - startTime := time.Now() - - record := models.ToolCallRecord{ - Arguments: make(map[string]any), - } - - // Extract tool name and arguments - if fn, ok := toolCall["function"].(map[string]any); ok { - if name, ok := fn["name"].(string); ok { - record.Name = name - } - if argsStr, ok := fn["arguments"].(string); ok { - if err := json.Unmarshal([]byte(argsStr), &record.Arguments); err != nil { - record.Error = fmt.Sprintf("failed to parse arguments: %v", err) - record.Duration = time.Since(startTime).Milliseconds() - return record - } - } - } - - if record.Name == "" { - record.Error = "missing tool name" - record.Duration = time.Since(startTime).Milliseconds() - return record - } - - // Interpolate template variables in tool arguments - // This allows tool calls to use {{input}} or other block outputs - record.Arguments = interpolateMapValues(record.Arguments, blockInputs) - - // AUTO-INJECT CSV DATA for analyze_data tool - // This fixes the issue where LLM uses filename as file_id (which doesn't exist in cache) - // Instead of relying on file lookup, we inject the already-extracted data directly - if record.Name == "analyze_data" && len(dataFiles) > 0 { - // Check if csv_data is already provided and non-empty - existingCSV, hasCSV := record.Arguments["csv_data"].(string) - if !hasCSV || existingCSV == "" { - // No csv_data provided - inject from our extracted data files - dataFile := dataFiles[0] // Use first data file - if dataFile.Content != "" { - record.Arguments["csv_data"] = dataFile.Content - // Clear file_id to prevent lookup attempts with invalid IDs - delete(record.Arguments, "file_id") - log.Printf("📊 [AGENT-BLOCK] Auto-injected csv_data from '%s' (%d chars) - bypassing file cache lookup", - dataFile.Filename, len(dataFile.Content)) - } - } - } - - // AUTO-INJECT CHART IMAGES for Discord/Slack messages - // The LLM may not include image_data at all, or use placeholders - we need the real base64 - if (record.Name == "send_discord_message" || record.Name == "send_slack_message") && len(generatedCharts) > 0 { - chartToInject := generatedCharts[len(generatedCharts)-1] // Use most recent chart - - // Check if image_data exists and is valid - imageData, hasImageData := record.Arguments["image_data"].(string) - - shouldInject := false - if !hasImageData || imageData == "" { - // No image_data provided - inject the chart - shouldInject = true - log.Printf("🖼️ [AGENT-BLOCK] No image_data in tool call, will auto-inject chart") - } else if len(imageData) < 100 { - // Placeholder or short text - not real base64 - shouldInject = true - log.Printf("🖼️ [AGENT-BLOCK] image_data is placeholder (%d chars), will auto-inject chart", len(imageData)) - } else if strings.Contains(imageData, "[CHART_IMAGE_SAVED]") || strings.Contains(imageData, "[BASE64") { - // Explicit placeholder text - shouldInject = true - log.Printf("🖼️ [AGENT-BLOCK] image_data contains placeholder text, will auto-inject chart") - } - - if shouldInject { - record.Arguments["image_data"] = chartToInject - // Also set a filename if not already set - if _, hasFilename := record.Arguments["image_filename"].(string); !hasFilename { - record.Arguments["image_filename"] = "chart.png" - } - log.Printf("📊 [AGENT-BLOCK] Auto-injected chart image into %s (%d bytes)", - record.Name, len(chartToInject)) - } - } - - // Inject credential resolver for tools that need authentication - // The resolver is user-scoped for security - only credentials owned by userID can be accessed - var resolver tools.CredentialResolver - if e.credentialService != nil && userID != "" { - resolver = e.credentialService.CreateCredentialResolver(userID) - record.Arguments[tools.CredentialResolverKey] = resolver - record.Arguments[tools.UserIDKey] = userID - } - - // Auto-inject credential_id for tools that need it - // This allows LLM to NOT know about credentials - we handle it automatically - toolIntegrationType := tools.GetIntegrationTypeForTool(record.Name) - if toolIntegrationType != "" && e.credentialService != nil && userID != "" { - var credentialID string - - // First, try to find from explicitly configured credentials - if len(credentials) > 0 && resolver != nil { - credentialID = findCredentialForIntegrationType(credentials, toolIntegrationType, resolver) - if credentialID != "" { - log.Printf("🔐 [AGENT-BLOCK] Found credential_id=%s from block config for tool=%s", - credentialID, record.Name) - } - } - - // If no credential found in block config, try runtime auto-discovery from user's credentials - if credentialID == "" { - log.Printf("🔍 [AGENT-BLOCK] No credentials in block config for tool=%s, trying runtime auto-discovery...", record.Name) - ctx := context.Background() - userCreds, err := e.credentialService.ListByUserAndType(ctx, userID, toolIntegrationType) - if err != nil { - log.Printf("⚠️ [AGENT-BLOCK] Failed to fetch user credentials: %v", err) - } else if len(userCreds) == 1 { - // Exactly one credential of this type - auto-use it - credentialID = userCreds[0].ID - log.Printf("🔐 [AGENT-BLOCK] Runtime auto-discovered single credential: %s (%s) for tool=%s", - userCreds[0].Name, credentialID, record.Name) - } else if len(userCreds) > 1 { - log.Printf("⚠️ [AGENT-BLOCK] Multiple credentials (%d) found for %s - cannot auto-select. User should configure in Block Settings.", - len(userCreds), toolIntegrationType) - } else { - log.Printf("⚠️ [AGENT-BLOCK] No %s credentials found for user. Please add one in Credentials Manager.", - toolIntegrationType) - } - } - - // Inject the credential_id if we found one - if credentialID != "" { - record.Arguments["credential_id"] = credentialID - log.Printf("🔐 [AGENT-BLOCK] Auto-injected credential_id=%s for tool=%s (type=%s)", - credentialID, record.Name, toolIntegrationType) - } - } - - // Inject image provider config for generate_image tool - if record.Name == "generate_image" { - imageProviderService := services.GetImageProviderService() - provider := imageProviderService.GetProvider() - if provider != nil { - record.Arguments[tools.ImageProviderConfigKey] = &tools.ImageProviderConfig{ - Name: provider.Name, - BaseURL: provider.BaseURL, - APIKey: provider.APIKey, - DefaultModel: provider.DefaultModel, - } - log.Printf("🎨 [AGENT-BLOCK] Injected image provider: %s (model: %s)", provider.Name, provider.DefaultModel) - } else { - log.Printf("⚠️ [AGENT-BLOCK] No image provider configured for generate_image tool") - } - } - - log.Printf("🔧 [AGENT-BLOCK] Executing tool: %s with args: %+v", record.Name, record.Arguments) - - // Execute the tool - result, err := e.toolRegistry.Execute(record.Name, record.Arguments) - if err != nil { - record.Error = err.Error() - log.Printf("❌ [AGENT-BLOCK] Tool %s failed: %v", record.Name, err) - } else { - record.Result = result - log.Printf("✅ [AGENT-BLOCK] Tool %s succeeded (result length: %d)", record.Name, len(result)) - } - - record.Duration = time.Since(startTime).Milliseconds() - - // Clean up internal keys from Arguments before storing - // These are injected for tool execution but should not be serialized - delete(record.Arguments, tools.CredentialResolverKey) - delete(record.Arguments, tools.UserIDKey) - delete(record.Arguments, tools.ImageProviderConfigKey) - - return record -} - -// getToolName extracts the tool name from a tool call -func (e *AgentBlockExecutor) getToolName(toolCall map[string]any) string { - if fn, ok := toolCall["function"].(map[string]any); ok { - if name, ok := fn["name"].(string); ok { - return name - } - } - return "" -} - -// parseAndValidateOutput parses the LLM response and validates against schema -func (e *AgentBlockExecutor) parseAndValidateOutput( - content string, - schema *models.JSONSchema, - strict bool, -) (map[string]any, error) { - log.Printf("📋 [VALIDATE] parseAndValidateOutput called, schema=%v, strict=%v", schema != nil, strict) - - // If no schema provided, try to parse as JSON or return as-is - if schema == nil { - log.Printf("📋 [VALIDATE] No schema provided, skipping validation") - // Try to parse as JSON - var output map[string]any - if err := json.Unmarshal([]byte(content), &output); err == nil { - return output, nil - } - - // Return content as "response" field - return map[string]any{ - "response": content, - }, nil - } - - // Extract JSON from content (handle markdown code blocks) - jsonContent := extractJSON(content) - - // Parse JSON - support both objects {} and arrays [] - var output any - if err := json.Unmarshal([]byte(jsonContent), &output); err != nil { - if strict { - return nil, fmt.Errorf("failed to parse output as JSON: %w", err) - } - // Non-strict: return content as-is with error - return map[string]any{ - "response": content, - "_parseError": err.Error(), - }, nil - } - - // Validate against schema (basic validation) - if err := e.validateSchema(output, schema); err != nil { - log.Printf("❌ [VALIDATE] Schema validation FAILED: %v", err) - if strict { - return nil, fmt.Errorf("output validation failed: %w", err) - } - // Non-strict: return with validation error at TOP LEVEL for retry loop detection - log.Printf("⚠️ [AGENT-EXEC] Validation warning (non-strict): %v", err) - return map[string]any{ - "response": content, - "data": output, - "_validationError": err.Error(), // TOP LEVEL for retry loop - }, nil - } - - log.Printf("✅ [VALIDATE] Schema validation PASSED") - // Return parsed data as response so downstream blocks can access fields via {{block.response.field}} - // Also include raw JSON string for debugging - return map[string]any{ - "response": output, // Parsed JSON object - allows {{block.response.field}} access - "data": output, // Alias for response - "rawResponse": content, // Raw JSON string for debugging - }, nil -} - -// extractJSON extracts JSON from content (handles markdown code blocks) -func extractJSON(content string) string { - content = strings.TrimSpace(content) - - // Check for markdown JSON code block - jsonBlockRegex := regexp.MustCompile("```(?:json)?\\s*([\\s\\S]*?)```") - if matches := jsonBlockRegex.FindStringSubmatch(content); len(matches) > 1 { - return strings.TrimSpace(matches[1]) - } - - // Try to find JSON object or array - start := strings.IndexAny(content, "{[") - if start == -1 { - return content - } - - // Find matching closing bracket - openBracket := content[start] - closeBracket := byte('}') - if openBracket == '[' { - closeBracket = ']' - } - - depth := 0 - for i := start; i < len(content); i++ { - if content[i] == openBracket { - depth++ - } else if content[i] == closeBracket { - depth-- - if depth == 0 { - return content[start : i+1] - } - } - } - - return content[start:] -} - -// validateSchema performs basic JSON schema validation - supports both objects and arrays -func (e *AgentBlockExecutor) validateSchema(data any, schema *models.JSONSchema) error { - if schema == nil { - return nil - } - - // Handle based on schema type - if schema.Type == "object" { - return e.validateObjectSchema(data, schema) - } else if schema.Type == "array" { - return e.validateArraySchema(data, schema) - } - - // If no type specified, infer from data - if schema.Type == "" { - if _, isMap := data.(map[string]any); isMap { - return e.validateObjectSchema(data, schema) - } - if _, isSlice := data.([]any); isSlice { - return e.validateArraySchema(data, schema) - } - } - - return nil -} - -// validateObjectSchema validates object (map) data against schema -func (e *AgentBlockExecutor) validateObjectSchema(data any, schema *models.JSONSchema) error { - dataMap, ok := data.(map[string]any) - if !ok { - return fmt.Errorf("schema expects object but got %T", data) - } - - // Check required fields - for _, required := range schema.Required { - if _, ok := dataMap[required]; !ok { - return fmt.Errorf("missing required field: %s", required) - } - } - - // Validate property types (basic) - for key, propSchema := range schema.Properties { - if value, ok := dataMap[key]; ok { - if err := e.validateValue(value, propSchema); err != nil { - return fmt.Errorf("field %s: %w", key, err) - } - } - } - - return nil -} - -// validateArraySchema validates array data against schema -func (e *AgentBlockExecutor) validateArraySchema(data any, schema *models.JSONSchema) error { - dataArray, ok := data.([]any) - if !ok { - return fmt.Errorf("schema expects array but got %T", data) - } - - // If no items schema, we can't validate items - if schema.Items == nil { - return nil - } - - // Validate each item against the items schema - for i, item := range dataArray { - if err := e.validateValue(item, schema.Items); err != nil { - return fmt.Errorf("array[%d]: %w", i, err) - } - } - - return nil -} - -// validateValue validates a single value against a schema -func (e *AgentBlockExecutor) validateValue(value any, schema *models.JSONSchema) error { - if schema == nil { - return nil - } - - switch schema.Type { - case "string": - if _, ok := value.(string); !ok { - return fmt.Errorf("expected string, got %T", value) - } - case "number", "integer": - switch value.(type) { - case float64, int, int64: - // OK - default: - return fmt.Errorf("expected number, got %T", value) - } - case "boolean": - if _, ok := value.(bool); !ok { - return fmt.Errorf("expected boolean, got %T", value) - } - case "array": - if _, ok := value.([]interface{}); !ok { - return fmt.Errorf("expected array, got %T", value) - } - case "object": - if _, ok := value.(map[string]interface{}); !ok { - return fmt.Errorf("expected object, got %T", value) - } - } - - return nil -} - -// findCredentialForIntegrationType finds the first credential matching the integration type -// from the list of credential IDs configured for the block. -func findCredentialForIntegrationType(credentialIDs []string, integrationType string, resolver tools.CredentialResolver) string { - for _, credID := range credentialIDs { - cred, err := resolver(credID) - if err != nil { - // Skip invalid credentials - continue - } - if cred.IntegrationType == integrationType { - return credID - } - } - return "" -} diff --git a/backend/internal/execution/block_checker.go b/backend/internal/execution/block_checker.go deleted file mode 100644 index e72b751a..00000000 --- a/backend/internal/execution/block_checker.go +++ /dev/null @@ -1,574 +0,0 @@ -package execution - -import ( - "bytes" - "claraverse/internal/models" - "claraverse/internal/services" - "context" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "strings" - "time" -) - -// BlockCheckResult represents the structured output from the block completion checker -type BlockCheckResult struct { - Passed bool `json:"passed"` // true if block accomplished its job - Reason string `json:"reason"` // explanation of why it passed or failed - ActualOutput string `json:"actual_output"` // truncated actual output for debugging (populated by checker) -} - -// BlockChecker validates whether a block actually accomplished its intended job -// This prevents workflows from continuing when a block technically "completed" -// but didn't actually do what it was supposed to do (e.g., tool errors, timeouts) -type BlockChecker struct { - providerService *services.ProviderService - httpClient *http.Client -} - -// NewBlockChecker creates a new block checker -func NewBlockChecker(providerService *services.ProviderService) *BlockChecker { - return &BlockChecker{ - providerService: providerService, - httpClient: &http.Client{ - Timeout: 30 * time.Second, - }, - } -} - -// CheckBlockCompletion validates if a block actually accomplished its intended task -// Parameters: -// - ctx: context for cancellation -// - workflowGoal: the overall workflow objective (from user's request) -// - block: the block that just executed -// - blockInput: what was passed to the block -// - blockOutput: what the block produced -// - modelID: the model to use for checking (should be a fast, cheap model) -// -// Returns: -// - BlockCheckResult with passed/failed status and reason -// - error if the check itself failed -func (c *BlockChecker) CheckBlockCompletion( - ctx context.Context, - workflowGoal string, - block models.Block, - blockInput map[string]any, - blockOutput map[string]any, - modelID string, -) (*BlockCheckResult, error) { - log.Printf("🔍 [BLOCK-CHECKER] Checking completion for block '%s' (type: %s)", block.Name, block.Type) - - // Skip checking for Start blocks (variable type with read operation) - if block.Type == "variable" { - log.Printf("⏭️ [BLOCK-CHECKER] Skipping Start block '%s'", block.Name) - return &BlockCheckResult{Passed: true, Reason: "Start block - no validation needed"}, nil - } - - // Build the validation prompt - prompt := c.buildValidationPrompt(workflowGoal, block, blockInput, blockOutput) - - // Get provider for the model - provider, err := c.providerService.GetByModelID(modelID) - if err != nil { - log.Printf("⚠️ [BLOCK-CHECKER] Provider error, defaulting to passed: %v", err) - return &BlockCheckResult{Passed: true, Reason: "Provider error - defaulting to passed"}, nil - } - - // Build the request with structured output - requestBody := map[string]interface{}{ - "model": modelID, - "max_tokens": 200, - "temperature": 0.1, // Low temperature for consistent validation - "messages": []map[string]string{ - { - "role": "system", - "content": "You are a workflow block validator. Analyze if the block accomplished its intended job based on its purpose, input, and output. Be strict but fair - if there are clear errors, tool failures, or the output doesn't match the intent, it should fail.", - }, - { - "role": "user", - "content": prompt, - }, - }, - "response_format": map[string]interface{}{ - "type": "json_schema", - "json_schema": map[string]interface{}{ - "name": "block_check_result", - "strict": true, - "schema": map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "passed": map[string]interface{}{ - "type": "boolean", - "description": "true if the block accomplished its intended job, false otherwise", - }, - "reason": map[string]interface{}{ - "type": "string", - "description": "Brief explanation (1-2 sentences) of why the block passed or failed", - }, - }, - "required": []string{"passed", "reason"}, - "additionalProperties": false, - }, - }, - }, - } - - bodyBytes, err := json.Marshal(requestBody) - if err != nil { - return nil, fmt.Errorf("failed to marshal request: %w", err) - } - - // Make the API call - apiURL := fmt.Sprintf("%s/chat/completions", provider.BaseURL) - req, err := http.NewRequestWithContext(ctx, "POST", apiURL, bytes.NewReader(bodyBytes)) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Authorization", fmt.Sprintf("Bearer %s", provider.APIKey)) - - resp, err := c.httpClient.Do(req) - if err != nil { - log.Printf("⚠️ [BLOCK-CHECKER] HTTP error, defaulting to passed: %v", err) - return &BlockCheckResult{Passed: true, Reason: "HTTP error during check - defaulting to passed"}, nil - } - defer resp.Body.Close() - - if resp.StatusCode != http.StatusOK { - body, _ := io.ReadAll(resp.Body) - log.Printf("⚠️ [BLOCK-CHECKER] API error (status %d): %s, defaulting to passed", resp.StatusCode, string(body)) - return &BlockCheckResult{Passed: true, Reason: "API error during check - defaulting to passed"}, nil - } - - // Parse response - var apiResp struct { - Choices []struct { - Message struct { - Content string `json:"content"` - } `json:"message"` - } `json:"choices"` - } - - if err := json.NewDecoder(resp.Body).Decode(&apiResp); err != nil { - log.Printf("⚠️ [BLOCK-CHECKER] Decode error, defaulting to passed: %v", err) - return &BlockCheckResult{Passed: true, Reason: "Response decode error - defaulting to passed"}, nil - } - - if len(apiResp.Choices) == 0 || apiResp.Choices[0].Message.Content == "" { - log.Printf("⚠️ [BLOCK-CHECKER] Empty response, defaulting to passed") - return &BlockCheckResult{Passed: true, Reason: "Empty response from checker - defaulting to passed"}, nil - } - - // Parse the structured output - var result BlockCheckResult - if err := json.Unmarshal([]byte(apiResp.Choices[0].Message.Content), &result); err != nil { - log.Printf("⚠️ [BLOCK-CHECKER] JSON parse error, defaulting to passed: %v", err) - return &BlockCheckResult{Passed: true, Reason: "JSON parse error - defaulting to passed"}, nil - } - - // Always populate ActualOutput with a summary of what the block produced - // This helps with debugging when the block fails - result.ActualOutput = c.summarizeOutputForError(blockOutput) - - if result.Passed { - log.Printf("✅ [BLOCK-CHECKER] Block '%s' PASSED: %s", block.Name, result.Reason) - } else { - log.Printf("❌ [BLOCK-CHECKER] Block '%s' FAILED: %s\n Actual Output: %s", block.Name, result.Reason, result.ActualOutput) - } - - return &result, nil -} - -// buildValidationPrompt creates the prompt for block validation -func (c *BlockChecker) buildValidationPrompt( - workflowGoal string, - block models.Block, - blockInput map[string]any, - blockOutput map[string]any, -) string { - // Extract key information from output - outputSummary := c.summarizeOutput(blockOutput) - - // Check for obvious failures - hasError := false - errorMessages := []string{} - - // Check for tool errors - if toolCalls, ok := blockOutput["toolCalls"].([]interface{}); ok { - for _, tc := range toolCalls { - if tcMap, ok := tc.(map[string]interface{}); ok { - if errMsg, ok := tcMap["error"].(string); ok && errMsg != "" { - hasError = true - errorMessages = append(errorMessages, fmt.Sprintf("Tool '%s' error: %s", tcMap["name"], errMsg)) - } - } - } - } - - // Check for timeout - if timedOut, ok := blockOutput["timedOut"].(bool); ok && timedOut { - hasError = true - errorMessages = append(errorMessages, "Block timed out before completing") - } - - // Check for parse errors - if parseErr, ok := blockOutput["_parseError"].(string); ok && parseErr != "" { - hasError = true - errorMessages = append(errorMessages, fmt.Sprintf("Parse error: %s", parseErr)) - } - - // Check for aggregated tool errors - if toolErr, ok := blockOutput["_toolError"].(string); ok && toolErr != "" { - hasError = true - errorMessages = append(errorMessages, fmt.Sprintf("Tool error: %s", toolErr)) - } - - // Check for empty response - handle both string and object types - hasResponse := false - if respStr, ok := blockOutput["response"].(string); ok && respStr != "" { - hasResponse = true - } else if respObj, ok := blockOutput["response"].(map[string]any); ok && len(respObj) > 0 { - hasResponse = true - } - if !hasResponse && block.Type == "llm_inference" { - hasError = true - errorMessages = append(errorMessages, "Block produced no response") - } - - // Build the prompt - include current date so the model knows what year it is - currentDate := time.Now().Format("January 2, 2006") - prompt := fmt.Sprintf(`## IMPORTANT: CURRENT DATE -**Today's Date:** %s -(This is the actual current date. Do NOT assume dates in the output are in the future.) - -## WORKFLOW CONTEXT -**Overall Goal:** %s - -## BLOCK BEING VALIDATED -**Block Name:** %s -**Block Type:** %s -**Block Description:** %s - -## BLOCK INPUT (what it received) -%s - -## BLOCK OUTPUT (what it produced) -%s -`, - currentDate, - workflowGoal, - block.Name, - block.Type, - block.Description, - c.formatForPrompt(blockInput), - outputSummary, - ) - - // Add error context if any - if hasError { - prompt += fmt.Sprintf(` -## DETECTED ISSUES -The following issues were detected in the block output: -%s - -`, c.formatErrors(errorMessages)) - } - - prompt += ` -## YOUR TASK -Analyze if this block accomplished its intended job within the workflow. - -CRITICAL DISTINCTION - External Failures vs Block Failures: -- **External failures** (API rate limits, service unavailable, network errors): The block DID ITS JOB correctly by calling the right tool with correct parameters. The failure is EXTERNAL. Mark as PASSED if the block handled it gracefully (explained the error, provided fallback info). -- **Block failures** (wrong tool called, missing required data, timeout, empty response): The block FAILED to do its job. - -Consider: -1. Did the block call the correct tool(s) with appropriate parameters? -2. If a tool returned an external error (rate limit, auth error, service down), did the block handle it gracefully? -3. Does the response acknowledge and explain what happened? -4. Is there meaningful information that downstream blocks can use (even if just error context)? - -IMPORTANT: -- External API errors (429 rate limit, 503 service unavailable, etc.) are NOT the block's fault - PASS if handled gracefully -- Parse errors are formatting issues, not functional failures - PASS if the response content is meaningful -- If the block called the right tool and got an external error, it PASSED (the tool worked, the API didn't) -- Only FAIL if: block timed out, produced no response, called wrong tools, or completely failed to attempt its task - -Return your judgment as JSON with "passed" (boolean) and "reason" (brief explanation).` - - return prompt -} - -// summarizeOutput creates a readable summary of block output for the prompt -func (c *BlockChecker) summarizeOutput(output map[string]any) string { - summary := "" - - // Response text - handle both string and object types - if resp, ok := output["response"].(string); ok && resp != "" { - // Truncate long responses - if len(resp) > 500 { - resp = resp[:500] + "... [truncated]" - } - summary += fmt.Sprintf("**Response:** %s\n\n", resp) - } else if respObj, ok := output["response"].(map[string]any); ok && len(respObj) > 0 { - // Response is a structured object (from schema validation) - respJSON, err := json.Marshal(respObj) - if err == nil { - respStr := string(respJSON) - if len(respStr) > 500 { - respStr = respStr[:500] + "... [truncated]" - } - summary += fmt.Sprintf("**Response (structured):** %s\n\n", respStr) - } - } - - // Timeout status - if timedOut, ok := output["timedOut"].(bool); ok && timedOut { - summary += "**Status:** TIMED OUT\n\n" - } - - // Iterations - if iterations, ok := output["iterations"].(int); ok { - summary += fmt.Sprintf("**Iterations:** %d\n\n", iterations) - } else if iterations, ok := output["iterations"].(float64); ok { - summary += fmt.Sprintf("**Iterations:** %.0f\n\n", iterations) - } - - // Tool calls summary - if toolCalls, ok := output["toolCalls"].([]interface{}); ok && len(toolCalls) > 0 { - summary += "**Tool Calls:**\n" - errorCount := 0 - successCount := 0 - for i, tc := range toolCalls { - if i >= 5 { - summary += fmt.Sprintf(" ... and %d more tool calls\n", len(toolCalls)-5) - break - } - if tcMap, ok := tc.(map[string]interface{}); ok { - name, _ := tcMap["name"].(string) - errMsg, hasErr := tcMap["error"].(string) - if hasErr && errMsg != "" { - errorCount++ - summary += fmt.Sprintf(" - %s: ❌ ERROR: %s\n", name, errMsg) - } else { - successCount++ - summary += fmt.Sprintf(" - %s: ✓ Success\n", name) - } - } - } - summary += fmt.Sprintf("\nTotal: %d successful, %d failed\n\n", successCount, errorCount) - } - - // Parse errors - if parseErr, ok := output["_parseError"].(string); ok && parseErr != "" { - summary += fmt.Sprintf("**Parse Error:** %s\n\n", parseErr) - } - - // Tool validation warning - if warning, ok := output["_toolValidationWarning"].(string); ok && warning != "" { - summary += fmt.Sprintf("**Warning:** %s\n\n", warning) - } - - // Artifacts (images, charts, etc.) - if artifacts, ok := output["artifacts"].([]interface{}); ok && len(artifacts) > 0 { - summary += fmt.Sprintf("**Artifacts Generated:** %d artifact(s) created\n", len(artifacts)) - for i, art := range artifacts { - if i >= 3 { - summary += fmt.Sprintf(" ... and %d more artifacts\n", len(artifacts)-3) - break - } - if artMap, ok := art.(map[string]interface{}); ok { - artType, _ := artMap["type"].(string) - artFormat, _ := artMap["format"].(string) - if artType != "" { - summary += fmt.Sprintf(" - Type: %s, Format: %s\n", artType, artFormat) - } else { - summary += fmt.Sprintf(" - Format: %s\n", artFormat) - } - } - } - summary += "\n" - } - - // Generated files - if files, ok := output["generatedFiles"].([]interface{}); ok && len(files) > 0 { - summary += fmt.Sprintf("**Generated Files:** %d file(s) created\n", len(files)) - for i, file := range files { - if i >= 3 { - summary += fmt.Sprintf(" ... and %d more files\n", len(files)-3) - break - } - if fileMap, ok := file.(map[string]interface{}); ok { - fileName, _ := fileMap["name"].(string) - fileType, _ := fileMap["type"].(string) - summary += fmt.Sprintf(" - %s (type: %s)\n", fileName, fileType) - } - } - summary += "\n" - } - - if summary == "" { - // Fallback: dump some of the output - outputBytes, _ := json.MarshalIndent(output, "", " ") - if len(outputBytes) > 1000 { - summary = string(outputBytes[:1000]) + "... [truncated]" - } else { - summary = string(outputBytes) - } - } - - return summary -} - -// formatForPrompt formats input data for the prompt -func (c *BlockChecker) formatForPrompt(data map[string]any) string { - // For input, just show key names and brief values - if len(data) == 0 { - return "(empty)" - } - - result := "" - for k, v := range data { - // Skip internal fields - if k[0] == '_' { - continue - } - valStr := fmt.Sprintf("%v", v) - if len(valStr) > 200 { - valStr = valStr[:200] + "..." - } - result += fmt.Sprintf("- **%s:** %s\n", k, valStr) - } - - if result == "" { - return "(internal data only)" - } - return result -} - -// formatErrors formats error messages for the prompt -func (c *BlockChecker) formatErrors(errors []string) string { - result := "" - for _, err := range errors { - result += fmt.Sprintf("- %s\n", err) - } - return result -} - -// ShouldCheckBlock determines if a block should be validated -// Some blocks (like Start/variable blocks) don't need validation -func ShouldCheckBlock(block models.Block) bool { - // Skip Start blocks (variable type with read operation) - if block.Type == "variable" { - if op, ok := block.Config["operation"].(string); ok && op == "read" { - return false - } - } - - // Only check LLM blocks (they're the ones that can fail in complex ways) - return block.Type == "llm_inference" -} - -// summarizeOutputForError creates a concise summary of block output for error messages -// This helps developers understand what went wrong when a block fails validation -func (c *BlockChecker) summarizeOutputForError(output map[string]any) string { - var parts []string - - // Include the response (truncated) - handle both string and object types - if resp, ok := output["response"].(string); ok && resp != "" { - truncated := resp - if len(truncated) > 300 { - truncated = truncated[:300] + "..." - } - parts = append(parts, fmt.Sprintf("Response: %q", truncated)) - } else if respObj, ok := output["response"].(map[string]any); ok && len(respObj) > 0 { - // Response is a structured object (from schema validation) - respJSON, err := json.Marshal(respObj) - if err == nil { - truncated := string(respJSON) - if len(truncated) > 300 { - truncated = truncated[:300] + "..." - } - parts = append(parts, fmt.Sprintf("Response: %s", truncated)) - } else { - parts = append(parts, fmt.Sprintf("Response: (object with %d keys)", len(respObj))) - } - } else { - parts = append(parts, "Response: (empty)") - } - - // Include parse error if present - if parseErr, ok := output["_parseError"].(string); ok && parseErr != "" { - parts = append(parts, fmt.Sprintf("Parse Error: %s", parseErr)) - } - - // Include tool validation warning if present - if warning, ok := output["_toolValidationWarning"].(string); ok && warning != "" { - parts = append(parts, fmt.Sprintf("Tool Warning: %s", warning)) - } - - // Summarize tool calls - handle both []models.ToolCallRecord and []interface{} - toolCallsSummarized := false - if toolCalls, ok := output["toolCalls"].([]models.ToolCallRecord); ok && len(toolCalls) > 0 { - successCount := 0 - failCount := 0 - var failedTools []string - for _, tc := range toolCalls { - if tc.Error != "" { - failCount++ - failedTools = append(failedTools, fmt.Sprintf("%s: %s", tc.Name, tc.Error)) - } else { - successCount++ - } - } - parts = append(parts, fmt.Sprintf("Tools: %d called (%d success, %d failed)", len(toolCalls), successCount, failCount)) - if len(failedTools) > 0 && len(failedTools) <= 3 { - for _, ft := range failedTools { - parts = append(parts, fmt.Sprintf(" - %s", ft)) - } - } - toolCallsSummarized = true - } else if toolCalls, ok := output["toolCalls"].([]interface{}); ok && len(toolCalls) > 0 { - // Fallback for []interface{} type (e.g., from JSON unmarshaling) - successCount := 0 - failCount := 0 - var failedTools []string - for _, tc := range toolCalls { - if tcMap, ok := tc.(map[string]interface{}); ok { - name, _ := tcMap["name"].(string) - if errMsg, hasErr := tcMap["error"].(string); hasErr && errMsg != "" { - failCount++ - failedTools = append(failedTools, fmt.Sprintf("%s: %s", name, errMsg)) - } else { - successCount++ - } - } - } - parts = append(parts, fmt.Sprintf("Tools: %d called (%d success, %d failed)", len(toolCalls), successCount, failCount)) - if len(failedTools) > 0 && len(failedTools) <= 3 { - for _, ft := range failedTools { - parts = append(parts, fmt.Sprintf(" - %s", ft)) - } - } - toolCallsSummarized = true - } - if !toolCallsSummarized { - parts = append(parts, "Tools: none called") - } - - // Include structured data summary if present - if data, ok := output["data"]; ok && data != nil { - dataBytes, _ := json.Marshal(data) - if len(dataBytes) > 200 { - parts = append(parts, fmt.Sprintf("Data: %s...", string(dataBytes[:200]))) - } else if len(dataBytes) > 0 { - parts = append(parts, fmt.Sprintf("Data: %s", string(dataBytes))) - } - } - - return strings.Join(parts, " | ") -} diff --git a/backend/internal/execution/engine.go b/backend/internal/execution/engine.go deleted file mode 100644 index 7c2e6815..00000000 --- a/backend/internal/execution/engine.go +++ /dev/null @@ -1,830 +0,0 @@ -package execution - -import ( - "claraverse/internal/models" - "claraverse/internal/services" - "context" - "fmt" - "log" - "sync" - "time" -) - -// WorkflowEngine executes workflows as DAGs with parallel execution -type WorkflowEngine struct { - registry *ExecutorRegistry - blockChecker *BlockChecker -} - -// NewWorkflowEngine creates a new workflow engine -func NewWorkflowEngine(registry *ExecutorRegistry) *WorkflowEngine { - return &WorkflowEngine{registry: registry} -} - -// NewWorkflowEngineWithChecker creates a workflow engine with block completion checking -func NewWorkflowEngineWithChecker(registry *ExecutorRegistry, providerService *services.ProviderService) *WorkflowEngine { - return &WorkflowEngine{ - registry: registry, - blockChecker: NewBlockChecker(providerService), - } -} - -// SetBlockChecker allows setting the block checker after creation -func (e *WorkflowEngine) SetBlockChecker(checker *BlockChecker) { - e.blockChecker = checker -} - -// ExecutionResult contains the final result of a workflow execution -type ExecutionResult struct { - Status string `json:"status"` // completed, failed, partial - Output map[string]any `json:"output"` - BlockStates map[string]*models.BlockState `json:"block_states"` - Error string `json:"error,omitempty"` -} - -// ExecutionOptions contains optional settings for workflow execution -type ExecutionOptions struct { - // WorkflowGoal is the high-level objective of the workflow (used for block checking) - WorkflowGoal string - // CheckerModelID is the model to use for block completion checking - // If empty, block checking is disabled - CheckerModelID string - // EnableBlockChecker enables/disables block completion validation - EnableBlockChecker bool -} - -// Execute runs a workflow and streams updates via the statusChan -// This is the backwards-compatible version without block checking -func (e *WorkflowEngine) Execute( - ctx context.Context, - workflow *models.Workflow, - input map[string]any, - statusChan chan<- models.ExecutionUpdate, -) (*ExecutionResult, error) { - return e.ExecuteWithOptions(ctx, workflow, input, statusChan, nil) -} - -// ExecuteWithOptions runs a workflow with optional block completion checking -func (e *WorkflowEngine) ExecuteWithOptions( - ctx context.Context, - workflow *models.Workflow, - input map[string]any, - statusChan chan<- models.ExecutionUpdate, - options *ExecutionOptions, -) (*ExecutionResult, error) { - log.Printf("🚀 [ENGINE] Starting workflow execution with %d blocks", len(workflow.Blocks)) - - // Build block index - blockIndex := make(map[string]models.Block) - for _, block := range workflow.Blocks { - blockIndex[block.ID] = block - } - - // Build dependency graph - // dependencies[blockID] = list of block IDs that must complete before this block - dependencies := make(map[string][]string) - // dependents[blockID] = list of block IDs that depend on this block - dependents := make(map[string][]string) - - for _, block := range workflow.Blocks { - dependencies[block.ID] = []string{} - dependents[block.ID] = []string{} - } - - for _, conn := range workflow.Connections { - // conn.SourceBlockID -> conn.TargetBlockID - dependencies[conn.TargetBlockID] = append(dependencies[conn.TargetBlockID], conn.SourceBlockID) - dependents[conn.SourceBlockID] = append(dependents[conn.SourceBlockID], conn.TargetBlockID) - } - - // Find start blocks (no dependencies) - var startBlocks []string - for blockID, deps := range dependencies { - if len(deps) == 0 { - startBlocks = append(startBlocks, blockID) - } - } - - if len(startBlocks) == 0 && len(workflow.Blocks) > 0 { - return nil, fmt.Errorf("workflow has no start blocks (circular dependency?)") - } - - log.Printf("📊 [ENGINE] Found %d start blocks: %v", len(startBlocks), startBlocks) - - // Initialize block states and outputs - blockStates := make(map[string]*models.BlockState) - blockOutputs := make(map[string]map[string]any) - var statesMu sync.RWMutex - - for _, block := range workflow.Blocks { - blockStates[block.ID] = &models.BlockState{ - Status: "pending", - } - } - - // Initialize with workflow variables and input - globalInputs := make(map[string]any) - log.Printf("🔍 [ENGINE] Workflow input received: %+v", input) - - // First, set workflow variable defaults - for _, variable := range workflow.Variables { - if variable.DefaultValue != nil { - globalInputs[variable.Name] = variable.DefaultValue - log.Printf("🔍 [ENGINE] Added workflow variable default: %s = %v", variable.Name, variable.DefaultValue) - } - } - - // Then, override with execution input (takes precedence over defaults) - for k, v := range input { - globalInputs[k] = v - log.Printf("🔍 [ENGINE] Added/overrode from execution input: %s = %v", k, v) - } - - // Extract workflow-level model override from Start block - for _, block := range workflow.Blocks { - if block.Type == "variable" { - if op, ok := block.Config["operation"].(string); ok && op == "read" { - if varName, ok := block.Config["variableName"].(string); ok && varName == "input" { - // This is the Start block - check for workflowModelId - if modelID, ok := block.Config["workflowModelId"].(string); ok && modelID != "" { - globalInputs["_workflowModelId"] = modelID - log.Printf("🎯 [ENGINE] Using workflow model override: %s", modelID) - } - } - } - } - } - - // Track completed blocks for dependency resolution - completedBlocks := make(map[string]bool) - failedBlocks := make(map[string]bool) - var completedMu sync.Mutex - - // Error tracking - var executionErrors []string - var errorsMu sync.Mutex - - // WaitGroup for tracking all goroutines - var wg sync.WaitGroup - - // Recursive function to execute a block and schedule dependents - var executeBlock func(blockID string) - executeBlock = func(blockID string) { - block := blockIndex[blockID] - - // Update status to running - statesMu.Lock() - blockStates[blockID].Status = "running" - blockStates[blockID].StartedAt = timePtr(time.Now()) - statesMu.Unlock() - - // Send status update (without inputs yet - will send after building them) - statusChan <- models.ExecutionUpdate{ - Type: "execution_update", - BlockID: blockID, - Status: "running", - } - - log.Printf("▶️ [ENGINE] Executing block '%s' (type: %s)", block.Name, block.Type) - - // Build inputs for this block from: - // 1. Global inputs (workflow input + variables) - // 2. Outputs from upstream blocks - blockInputs := make(map[string]any) - log.Printf("🔍 [ENGINE] Block '%s': globalInputs keys: %v", block.Name, getMapKeys(globalInputs)) - for k, v := range globalInputs { - blockInputs[k] = v - } - log.Printf("🔍 [ENGINE] Block '%s': blockInputs after globalInputs: %v", block.Name, getMapKeys(blockInputs)) - - // Make ALL completed block outputs available for template resolution - // This allows blocks to reference any upstream block, not just directly connected ones - // Example: Final block can use {{start.response}}, {{research-overview.response}}, etc. - statesMu.RLock() - - essentialKeys := []string{ - "response", "data", "output", "value", "result", - "artifacts", "toolResults", "tokens", "model", - "iterations", "_parseError", "rawResponse", - "generatedFiles", "toolCalls", "timedOut", - } - - // Track which block is directly connected (for flattening priority) - directlyConnectedBlockID := "" - for _, conn := range workflow.Connections { - if conn.TargetBlockID == blockID { - directlyConnectedBlockID = conn.SourceBlockID - break - } - } - - // Add ALL completed block outputs (for template access like {{block-name.response}}) - for completedBlockID, output := range blockOutputs { - sourceBlock, exists := blockIndex[completedBlockID] - if !exists { - continue - } - - // Create clean output (only essential keys) - cleanOutput := make(map[string]any) - for _, key := range essentialKeys { - if val, exists := output[key]; exists { - cleanOutput[key] = val - } - } - - // Store under normalizedId (e.g., "research-overview") - if sourceBlock.NormalizedID != "" { - blockInputs[sourceBlock.NormalizedID] = cleanOutput - } - - // Also store under block ID if different - if sourceBlock.ID != "" && sourceBlock.ID != sourceBlock.NormalizedID { - blockInputs[sourceBlock.ID] = cleanOutput - } - } - - // Log available block references - log.Printf("🔗 [ENGINE] Block '%s' can access %d upstream blocks", block.Name, len(blockOutputs)) - - // Flatten essential keys from DIRECTLY CONNECTED block only (for {{response}} shorthand) - if directlyConnectedBlockID != "" { - if output, ok := blockOutputs[directlyConnectedBlockID]; ok { - for _, key := range essentialKeys { - if val, exists := output[key]; exists { - blockInputs[key] = val - } - } - log.Printf("🔗 [ENGINE] Flattened keys from directly connected block '%s'", blockIndex[directlyConnectedBlockID].Name) - } - } - - statesMu.RUnlock() - - // Store the available inputs in BlockState for debugging - statesMu.Lock() - blockStates[blockID].Inputs = blockInputs - statesMu.Unlock() - - log.Printf("🔍 [ENGINE] Block '%s': stored %d input keys for debugging: %v", block.Name, len(blockInputs), getMapKeys(blockInputs)) - - // Send updated status with inputs for debugging - statusChan <- models.ExecutionUpdate{ - Type: "execution_update", - BlockID: blockID, - Status: "running", - Inputs: blockInputs, - } - - // Get executor for this block type - executor, execErr := e.registry.Get(block.Type) - if execErr != nil { - handleBlockError(blockID, block.Name, execErr, blockStates, &statesMu, statusChan, &executionErrors, &errorsMu) - completedMu.Lock() - failedBlocks[blockID] = true - completedMu.Unlock() - return - } - - // Create timeout context - // Default: 30s for most blocks, 120s for LLM blocks (they need more time for API calls) - timeout := 30 * time.Second - if block.Type == "llm_inference" { - timeout = 120 * time.Second // LLM blocks get 2 minutes by default - } - // User-specified timeout can override, but LLM blocks get at least 120s - if block.Timeout > 0 { - userTimeout := time.Duration(block.Timeout) * time.Second - if block.Type == "llm_inference" && userTimeout < 120*time.Second { - // LLM blocks need at least 120s for reasoning/streaming - timeout = 120 * time.Second - } else { - timeout = userTimeout - } - } - blockCtx, cancel := context.WithTimeout(ctx, timeout) - defer cancel() - - // Execute the block - output, execErr := executor.Execute(blockCtx, block, blockInputs) - if execErr != nil { - handleBlockError(blockID, block.Name, execErr, blockStates, &statesMu, statusChan, &executionErrors, &errorsMu) - completedMu.Lock() - failedBlocks[blockID] = true - completedMu.Unlock() - return - } - - // Block Completion Check: Validate if block actually accomplished its job - // This catches cases where a block "completed" but didn't actually succeed - // (e.g., repeated tool errors, timeouts, empty responses) - if options != nil && options.EnableBlockChecker && e.blockChecker != nil && ShouldCheckBlock(block) { - log.Printf("🔍 [ENGINE] Running block completion check for '%s'", block.Name) - - checkerModelID := options.CheckerModelID - if checkerModelID == "" { - // Default to a fast model for checking - checkerModelID = "gpt-4.1" - } - - checkResult, checkErr := e.blockChecker.CheckBlockCompletion( - ctx, - options.WorkflowGoal, - block, - blockInputs, - output, - checkerModelID, - ) - - if checkErr != nil { - log.Printf("⚠️ [ENGINE] Block checker error (continuing): %v", checkErr) - } else if !checkResult.Passed { - // Block failed the completion check - treat as failure - log.Printf("❌ [ENGINE] Block '%s' failed completion check: %s\n Actual Output: %s", block.Name, checkResult.Reason, checkResult.ActualOutput) - - // Add check failure info to output for visibility - output["_blockCheckFailed"] = true - output["_blockCheckReason"] = checkResult.Reason - output["_blockActualOutput"] = checkResult.ActualOutput - - checkError := fmt.Errorf("block did not accomplish its job: %s\n\nActual Output: %s", checkResult.Reason, checkResult.ActualOutput) - handleBlockError(blockID, block.Name, checkError, blockStates, &statesMu, statusChan, &executionErrors, &errorsMu) - completedMu.Lock() - failedBlocks[blockID] = true - completedMu.Unlock() - return - } else { - log.Printf("✓ [ENGINE] Block '%s' passed completion check: %s", block.Name, checkResult.Reason) - } - } - - // Store output and mark completed - statesMu.Lock() - blockOutputs[blockID] = output - blockStates[blockID].Status = "completed" - blockStates[blockID].CompletedAt = timePtr(time.Now()) - blockStates[blockID].Outputs = output - statesMu.Unlock() - - // Send completion update with inputs for debugging - statusChan <- models.ExecutionUpdate{ - Type: "execution_update", - BlockID: blockID, - Status: "completed", - Inputs: blockInputs, - Output: output, - } - - log.Printf("✅ [ENGINE] Block '%s' completed", block.Name) - - // Mark as completed and check dependents - completedMu.Lock() - completedBlocks[blockID] = true - - // Check if any dependent blocks can now run - for _, depBlockID := range dependents[blockID] { - canRun := true - for _, reqBlockID := range dependencies[depBlockID] { - if !completedBlocks[reqBlockID] { - // Check if the required block failed - if so, we can't run - if failedBlocks[reqBlockID] { - canRun = false - break - } - // Required block hasn't completed yet - canRun = false - break - } - } - if canRun { - // Queue this block for execution - wg.Add(1) - go func(bid string) { - defer wg.Done() - executeBlock(bid) - }(depBlockID) - } - } - completedMu.Unlock() - } - - // Start execution with start blocks - for _, blockID := range startBlocks { - wg.Add(1) - go func(bid string) { - defer wg.Done() - executeBlock(bid) - }(blockID) - } - - // Wait for all blocks to complete - wg.Wait() - - // Determine final status - finalStatus := "completed" - var failedBlockIDs []string - var completedCount, failedCount int - - statesMu.RLock() - for blockID, state := range blockStates { - if state.Status == "completed" { - completedCount++ - } else if state.Status == "failed" { - failedCount++ - failedBlockIDs = append(failedBlockIDs, blockID) - } - } - statesMu.RUnlock() - - if failedCount > 0 { - if completedCount > 0 { - finalStatus = "partial" - } else { - finalStatus = "failed" - } - } - - // Collect final output from terminal blocks (blocks with no dependents) - finalOutput := make(map[string]any) - statesMu.RLock() - for blockID, deps := range dependents { - if len(deps) == 0 { - if output, ok := blockOutputs[blockID]; ok { - block := blockIndex[blockID] - finalOutput[block.Name] = output - } - } - } - statesMu.RUnlock() - - // Build error message if any - var errorMsg string - errorsMu.Lock() - if len(executionErrors) > 0 { - errorMsg = fmt.Sprintf("%d block(s) failed: %v", len(executionErrors), executionErrors) - } - errorsMu.Unlock() - - log.Printf("🏁 [ENGINE] Workflow execution %s: %d completed, %d failed", - finalStatus, completedCount, failedCount) - - return &ExecutionResult{ - Status: finalStatus, - Output: finalOutput, - BlockStates: blockStates, - Error: errorMsg, - }, nil -} - -// handleBlockError handles block execution errors with classification for debugging -func handleBlockError( - blockID, blockName string, - err error, - blockStates map[string]*models.BlockState, - statesMu *sync.RWMutex, - statusChan chan<- models.ExecutionUpdate, - executionErrors *[]string, - errorsMu *sync.Mutex, -) { - // Try to extract error classification for better debugging - var errorType string - var retryable bool - - if execErr, ok := err.(*ExecutionError); ok { - errorType = execErr.Category.String() - retryable = execErr.Retryable - log.Printf("❌ [ENGINE] Block '%s' failed: %v [type=%s, retryable=%v]", blockName, err, errorType, retryable) - } else { - errorType = "unknown" - retryable = false - log.Printf("❌ [ENGINE] Block '%s' failed: %v", blockName, err) - } - - statesMu.Lock() - blockStates[blockID].Status = "failed" - blockStates[blockID].CompletedAt = timePtr(time.Now()) - blockStates[blockID].Error = err.Error() - statesMu.Unlock() - - // Include error classification in status update for frontend visibility - statusChan <- models.ExecutionUpdate{ - Type: "execution_update", - BlockID: blockID, - Status: "failed", - Error: err.Error(), - Output: map[string]any{ - "errorType": errorType, - "retryable": retryable, - }, - } - - errorsMu.Lock() - *executionErrors = append(*executionErrors, fmt.Sprintf("%s: %s", blockName, err.Error())) - errorsMu.Unlock() -} - -// timePtr returns a pointer to a time.Time -func timePtr(t time.Time) *time.Time { - return &t -} - -// getMapKeys returns the keys of a map as a slice -func getMapKeys(m map[string]any) []string { - keys := make([]string, 0, len(m)) - for k := range m { - keys = append(keys, k) - } - return keys -} - -// BuildAPIResponse converts an ExecutionResult into a clean, structured API response -// This provides a standardized output format for API consumers -func (e *WorkflowEngine) BuildAPIResponse( - result *ExecutionResult, - workflow *models.Workflow, - executionID string, - durationMs int64, -) *models.ExecutionAPIResponse { - response := &models.ExecutionAPIResponse{ - Status: result.Status, - Artifacts: []models.APIArtifact{}, - Files: []models.APIFile{}, - Blocks: make(map[string]models.APIBlockOutput), - Metadata: models.ExecutionMetadata{ - ExecutionID: executionID, - DurationMs: durationMs, - }, - Error: result.Error, - } - - // Build block index for lookups - blockIndex := make(map[string]models.Block) - for _, block := range workflow.Blocks { - blockIndex[block.ID] = block - } - - // Track totals - var totalTokens int - var blocksExecuted, blocksFailed int - - // Process block states - for blockID, state := range result.BlockStates { - block, exists := blockIndex[blockID] - if !exists { - continue - } - - // Create clean block output - blockOutput := models.APIBlockOutput{ - Name: block.Name, - Type: block.Type, - Status: state.Status, - } - - if state.Status == "completed" { - blocksExecuted++ - } else if state.Status == "failed" { - blocksFailed++ - blockOutput.Error = state.Error - } - - // Extract response text from outputs - if state.Outputs != nil { - // Primary response - if resp, ok := state.Outputs["response"].(string); ok { - blockOutput.Response = resp - } - - // Extract tokens (for metadata, but don't expose) - if tokens, ok := state.Outputs["tokens"].(map[string]any); ok { - if total, ok := tokens["total"].(int); ok { - totalTokens += total - } else if total, ok := tokens["total"].(float64); ok { - totalTokens += int(total) - } - } - - // Calculate duration from timestamps - if state.StartedAt != nil && state.CompletedAt != nil { - blockOutput.DurationMs = state.CompletedAt.Sub(*state.StartedAt).Milliseconds() - } - - // Extract structured data - filter all outputs except response - cleanData := make(map[string]any) - for k, v := range state.Outputs { - // Skip internal fields and the response (already extracted) - if !isInternalField(k) && k != "response" { - // Also check nested output object - if k == "output" { - if outputMap, ok := v.(map[string]any); ok { - for ok, ov := range outputMap { - if !isInternalField(ok) && ok != "response" { - cleanData[ok] = ov - } - } - } - } else { - cleanData[k] = v - } - } - } - if len(cleanData) > 0 { - blockOutput.Data = cleanData - } - - // Extract artifacts from this block - artifacts := extractArtifactsFromBlockOutput(state.Outputs, block.Name) - response.Artifacts = append(response.Artifacts, artifacts...) - - // Extract files from this block - files := extractFilesFromBlockOutput(state.Outputs, block.Name) - response.Files = append(response.Files, files...) - } - - response.Blocks[block.ID] = blockOutput - } - - // Set metadata - response.Metadata.TotalTokens = totalTokens - response.Metadata.BlocksExecuted = blocksExecuted - response.Metadata.BlocksFailed = blocksFailed - if workflow != nil { - response.Metadata.WorkflowVersion = workflow.Version - } - - // Extract the primary result and structured data from terminal blocks - response.Result, response.Data = extractPrimaryResultAndData(result.Output, result.BlockStates) - - log.Printf("📦 [ENGINE] Built API response: status=%s, result_length=%d, has_data=%v, artifacts=%d, files=%d", - response.Status, len(response.Result), response.Data != nil, len(response.Artifacts), len(response.Files)) - - return response -} - -// extractPrimaryResultAndData gets the main text result AND structured data from the workflow output -// For structured output blocks, the "data" field contains the parsed JSON which we return separately -func extractPrimaryResultAndData(output map[string]any, blockStates map[string]*models.BlockState) (string, any) { - // First, try to get from the final output (terminal blocks) - for blockName, blockOutput := range output { - if blockData, ok := blockOutput.(map[string]any); ok { - var resultStr string - var structuredData any - - // Look for response field (the text/JSON string) - if resp, ok := blockData["response"].(string); ok && resp != "" { - resultStr = resp - log.Printf("📝 [ENGINE] Extracted primary result from block '%s' (%d chars)", blockName, len(resp)) - } else if resp, ok := blockData["rawResponse"].(string); ok && resp != "" { - // Fallback to rawResponse - resultStr = resp - } - - // Look for structured data field (parsed JSON from structured output blocks) - // This is populated when outputFormat="json" and the response was successfully parsed - if data, ok := blockData["data"]; ok && data != nil { - structuredData = data - log.Printf("📊 [ENGINE] Extracted structured data from block '%s'", blockName) - } - - if resultStr != "" { - return resultStr, structuredData - } - } - } - - // Fallback: find the last completed block with a response - var lastResponse string - var lastData any - for _, state := range blockStates { - if state.Status == "completed" && state.Outputs != nil { - if resp, ok := state.Outputs["response"].(string); ok && resp != "" { - lastResponse = resp - } - if data, ok := state.Outputs["data"]; ok && data != nil { - lastData = data - } - } - } - - return lastResponse, lastData -} - -// extractArtifactsFromBlockOutput extracts artifacts from a block's output -func extractArtifactsFromBlockOutput(outputs map[string]any, blockName string) []models.APIArtifact { - var artifacts []models.APIArtifact - - // Check for artifacts array - if rawArtifacts, ok := outputs["artifacts"]; ok { - switch arts := rawArtifacts.(type) { - case []any: - for _, a := range arts { - if artMap, ok := a.(map[string]any); ok { - artifact := models.APIArtifact{ - SourceBlock: blockName, - } - if t, ok := artMap["type"].(string); ok { - artifact.Type = t - } - if f, ok := artMap["format"].(string); ok { - artifact.Format = f - } - if d, ok := artMap["data"].(string); ok { - artifact.Data = d - } - if t, ok := artMap["title"].(string); ok { - artifact.Title = t - } - if artifact.Data != "" && len(artifact.Data) > 100 { - artifacts = append(artifacts, artifact) - } - } - } - } - } - - return artifacts -} - -// extractFilesFromBlockOutput extracts generated files from a block's output -func extractFilesFromBlockOutput(outputs map[string]any, blockName string) []models.APIFile { - var files []models.APIFile - - // Check for generatedFiles array - if rawFiles, ok := outputs["generatedFiles"]; ok { - switch fs := rawFiles.(type) { - case []any: - for _, f := range fs { - if fileMap, ok := f.(map[string]any); ok { - file := models.APIFile{ - SourceBlock: blockName, - } - if id, ok := fileMap["file_id"].(string); ok { - file.FileID = id - } - if fn, ok := fileMap["filename"].(string); ok { - file.Filename = fn - } - if url, ok := fileMap["download_url"].(string); ok { - file.DownloadURL = url - } - if mt, ok := fileMap["mime_type"].(string); ok { - file.MimeType = mt - } - if sz, ok := fileMap["size"].(float64); ok { - file.Size = int64(sz) - } - if file.FileID != "" || file.DownloadURL != "" { - files = append(files, file) - } - } - } - } - } - - // Also check for single file reference - if fileURL, ok := outputs["file_url"].(string); ok && fileURL != "" { - file := models.APIFile{ - DownloadURL: fileURL, - SourceBlock: blockName, - } - if fn, ok := outputs["file_name"].(string); ok { - file.Filename = fn - } - files = append(files, file) - } - - return files -} - -// isInternalField checks if a field name is internal and should be hidden from API response -func isInternalField(key string) bool { - // Any field starting with _ or __ is internal - if len(key) > 0 && key[0] == '_' { - return true - } - - internalFields := map[string]bool{ - // Response duplicates - "rawResponse": true, - "output": true, // Duplicate of response - - // Execution internals - "tokens": true, - "toolCalls": true, - "iterations": true, - "model": true, // Internal model ID - never expose - - // Already extracted separately - "artifacts": true, - "generatedFiles": true, - "file_url": true, - "file_name": true, - - // Passthrough noise - "start": true, - "input": true, // Passthrough from workflow input - "value": true, // Duplicate of input - "timedOut": true, - } - return internalFields[key] -} \ No newline at end of file diff --git a/backend/internal/execution/errors.go b/backend/internal/execution/errors.go deleted file mode 100644 index 05127382..00000000 --- a/backend/internal/execution/errors.go +++ /dev/null @@ -1,312 +0,0 @@ -package execution - -import ( - "fmt" - "math" - "math/rand" - "net/http" - "strings" - "time" -) - -// ErrorCategory classifies errors for retry decisions -type ErrorCategory int - -const ( - // ErrorCategoryUnknown - unclassified error, default to not retryable - ErrorCategoryUnknown ErrorCategory = iota - - // ErrorCategoryTransient - temporary failures that may succeed on retry - // Examples: timeout, rate limit (429), server error (5xx), network error - ErrorCategoryTransient - - // ErrorCategoryPermanent - errors that will not succeed on retry - // Examples: auth error (401/403), bad request (400), parse error - ErrorCategoryPermanent - - // ErrorCategoryValidation - business logic validation failures - // Examples: tool not called, required tool missing - ErrorCategoryValidation -) - -// String returns a human-readable category name -func (c ErrorCategory) String() string { - switch c { - case ErrorCategoryTransient: - return "transient" - case ErrorCategoryPermanent: - return "permanent" - case ErrorCategoryValidation: - return "validation" - default: - return "unknown" - } -} - -// ExecutionError wraps errors with classification for retry logic -type ExecutionError struct { - Category ErrorCategory - Message string - StatusCode int // HTTP status code if applicable - Retryable bool // Explicit retryable flag - RetryAfter int // Seconds to wait before retry (from Retry-After header) - Cause error // Original error -} - -func (e *ExecutionError) Error() string { - if e.StatusCode > 0 { - return fmt.Sprintf("[%d] %s", e.StatusCode, e.Message) - } - return e.Message -} - -func (e *ExecutionError) Unwrap() error { - return e.Cause -} - -// IsRetryable determines if an error should be retried -func (e *ExecutionError) IsRetryable() bool { - return e.Retryable -} - -// ClassifyHTTPError classifies an HTTP response error -func ClassifyHTTPError(statusCode int, body string) *ExecutionError { - err := &ExecutionError{ - StatusCode: statusCode, - Message: fmt.Sprintf("HTTP %d: %s", statusCode, truncateString(body, 200)), - } - - switch { - // Rate limiting - always retryable - case statusCode == http.StatusTooManyRequests: - err.Category = ErrorCategoryTransient - err.Retryable = true - err.RetryAfter = 60 // Default 60 seconds for rate limiting - - // Server errors - retryable - case statusCode >= 500 && statusCode < 600: - err.Category = ErrorCategoryTransient - err.Retryable = true - - // Request timeout - retryable - case statusCode == http.StatusRequestTimeout: - err.Category = ErrorCategoryTransient - err.Retryable = true - - // Gateway errors - retryable - case statusCode == http.StatusBadGateway || - statusCode == http.StatusServiceUnavailable || - statusCode == http.StatusGatewayTimeout: - err.Category = ErrorCategoryTransient - err.Retryable = true - - // Auth errors - NOT retryable - case statusCode == http.StatusUnauthorized || statusCode == http.StatusForbidden: - err.Category = ErrorCategoryPermanent - err.Retryable = false - - // Bad request - NOT retryable - case statusCode == http.StatusBadRequest: - err.Category = ErrorCategoryPermanent - err.Retryable = false - - // Not found - NOT retryable - case statusCode == http.StatusNotFound: - err.Category = ErrorCategoryPermanent - err.Retryable = false - - // Unprocessable entity - NOT retryable - case statusCode == http.StatusUnprocessableEntity: - err.Category = ErrorCategoryPermanent - err.Retryable = false - - default: - err.Category = ErrorCategoryUnknown - err.Retryable = false - } - - return err -} - -// ClassifyError classifies a general error -func ClassifyError(err error) *ExecutionError { - if err == nil { - return nil - } - - // If already an ExecutionError, return as-is - if execErr, ok := err.(*ExecutionError); ok { - return execErr - } - - errStr := err.Error() - - // Context timeout/cancellation - if strings.Contains(errStr, "context deadline exceeded") || - strings.Contains(errStr, "context canceled") { - return &ExecutionError{ - Category: ErrorCategoryTransient, - Message: "Request timed out", - Retryable: true, - Cause: err, - } - } - - // Network errors - connection issues - if strings.Contains(errStr, "connection refused") || - strings.Contains(errStr, "connection reset") || - strings.Contains(errStr, "no such host") || - strings.Contains(errStr, "network is unreachable") || - strings.Contains(errStr, "i/o timeout") || - strings.Contains(errStr, "EOF") { - return &ExecutionError{ - Category: ErrorCategoryTransient, - Message: fmt.Sprintf("Network error: %s", truncateString(errStr, 100)), - Retryable: true, - Cause: err, - } - } - - // TLS errors - usually permanent - if strings.Contains(errStr, "certificate") || - strings.Contains(errStr, "tls:") || - strings.Contains(errStr, "x509:") { - return &ExecutionError{ - Category: ErrorCategoryPermanent, - Message: "TLS/Certificate error", - Retryable: false, - Cause: err, - } - } - - // DNS errors - may be transient - if strings.Contains(errStr, "no such host") || - strings.Contains(errStr, "dns") { - return &ExecutionError{ - Category: ErrorCategoryTransient, - Message: "DNS resolution error", - Retryable: true, - Cause: err, - } - } - - // Default: unknown, not retryable - return &ExecutionError{ - Category: ErrorCategoryUnknown, - Message: truncateString(errStr, 200), - Retryable: false, - Cause: err, - } -} - -// BackoffCalculator computes retry delays with exponential backoff and jitter -type BackoffCalculator struct { - initialDelay time.Duration - maxDelay time.Duration - multiplier float64 - jitterPercent int -} - -// NewBackoffCalculator creates a calculator with specified parameters -func NewBackoffCalculator(initialDelayMs, maxDelayMs int, multiplier float64, jitterPercent int) *BackoffCalculator { - // Apply defaults if not specified - if initialDelayMs <= 0 { - initialDelayMs = 1000 // 1 second default - } - if maxDelayMs <= 0 { - maxDelayMs = 30000 // 30 seconds default - } - if multiplier <= 0 { - multiplier = 2.0 // Double each time - } - if jitterPercent < 0 { - jitterPercent = 20 // 20% jitter default - } - - return &BackoffCalculator{ - initialDelay: time.Duration(initialDelayMs) * time.Millisecond, - maxDelay: time.Duration(maxDelayMs) * time.Millisecond, - multiplier: multiplier, - jitterPercent: jitterPercent, - } -} - -// NextDelay calculates the delay for the given attempt number (0-indexed) -func (b *BackoffCalculator) NextDelay(attempt int) time.Duration { - if attempt < 0 { - attempt = 0 - } - - // Calculate exponential delay: initialDelay * (multiplier ^ attempt) - delay := float64(b.initialDelay) * math.Pow(b.multiplier, float64(attempt)) - - // Cap at max delay - if delay > float64(b.maxDelay) { - delay = float64(b.maxDelay) - } - - // Add jitter to prevent thundering herd - if b.jitterPercent > 0 { - jitterRange := delay * float64(b.jitterPercent) / 100.0 - jitter := (rand.Float64()*2 - 1) * jitterRange // -jitterRange to +jitterRange - delay += jitter - } - - // Ensure non-negative - if delay < 0 { - delay = float64(b.initialDelay) - } - - return time.Duration(delay) -} - -// ShouldRetry determines if the error type should be retried based on policy -func ShouldRetry(err *ExecutionError, retryOn []string) bool { - if err == nil || !err.Retryable { - return false - } - - // If no specific retry types configured, retry all retryable errors - if len(retryOn) == 0 { - return err.Retryable - } - - // Map error to type string - errorType := getErrorType(err) - - // Check if this error type is in the retry list - for _, retryType := range retryOn { - if retryType == errorType || retryType == "all_transient" { - return true - } - } - - return false -} - -// getErrorType maps an ExecutionError to a type string for retry matching -func getErrorType(err *ExecutionError) string { - if err.StatusCode == 429 { - return "rate_limit" - } - if err.StatusCode >= 500 { - return "server_error" - } - if strings.Contains(strings.ToLower(err.Message), "timeout") || - strings.Contains(strings.ToLower(err.Message), "deadline exceeded") { - return "timeout" - } - if strings.Contains(strings.ToLower(err.Message), "network") || - strings.Contains(strings.ToLower(err.Message), "connection") { - return "network_error" - } - return "unknown" -} - -// truncateString truncates a string to maxLen characters -func truncateString(s string, maxLen int) string { - if len(s) <= maxLen { - return s - } - return s[:maxLen] + "..." -} diff --git a/backend/internal/execution/executor.go b/backend/internal/execution/executor.go deleted file mode 100644 index a7789141..00000000 --- a/backend/internal/execution/executor.go +++ /dev/null @@ -1,58 +0,0 @@ -package execution - -import ( - "claraverse/internal/models" - "claraverse/internal/services" - "claraverse/internal/tools" - "context" - "fmt" -) - -// BlockExecutor interface for all block types -type BlockExecutor interface { - Execute(ctx context.Context, block models.Block, inputs map[string]any) (map[string]any, error) -} - -// ExecutorRegistry maps block types to executors -type ExecutorRegistry struct { - executors map[string]BlockExecutor -} - -// NewExecutorRegistry creates a new executor registry with all block type executors -// Hybrid Architecture: Supports variable, llm_inference, and code_block types. -// - variable: Input/output data handling -// - llm_inference: AI reasoning with tool access -// - code_block: Direct tool execution (no LLM, faster & deterministic) -func NewExecutorRegistry( - chatService *services.ChatService, - providerService *services.ProviderService, - toolRegistry *tools.Registry, - credentialService *services.CredentialService, -) *ExecutorRegistry { - return &ExecutorRegistry{ - executors: map[string]BlockExecutor{ - // Variable blocks handle input/output data - "variable": NewVariableExecutor(), - // LLM blocks handle all intelligent actions via tools - // Tools available: search_web, scrape_web, send_webhook, send_discord_message, send_slack_message, etc. - "llm_inference": NewAgentBlockExecutor(chatService, providerService, toolRegistry, credentialService), - // Code blocks execute tools directly without LLM (faster, deterministic) - // Use for mechanical tasks that don't need AI reasoning - "code_block": NewToolExecutor(toolRegistry, credentialService), - }, - } -} - -// Get retrieves an executor for a block type -func (r *ExecutorRegistry) Get(blockType string) (BlockExecutor, error) { - exec, ok := r.executors[blockType] - if !ok { - return nil, fmt.Errorf("no executor registered for block type: %s", blockType) - } - return exec, nil -} - -// Register adds a new executor for a block type -func (r *ExecutorRegistry) Register(blockType string, executor BlockExecutor) { - r.executors[blockType] = executor -} diff --git a/backend/internal/execution/format_to_schema.go b/backend/internal/execution/format_to_schema.go deleted file mode 100644 index bf7e6154..00000000 --- a/backend/internal/execution/format_to_schema.go +++ /dev/null @@ -1,313 +0,0 @@ -package execution - -import ( - "claraverse/internal/models" - "context" - "encoding/json" - "fmt" - "log" - "regexp" - "strings" -) - -// FormatInput represents the input data to be formatted -type FormatInput struct { - // RawData is the unstructured data from task execution (can be string, map, slice, etc.) - RawData any - - // ToolResults contains results from tool executions (if any) - ToolResults []map[string]any - - // LLMResponse is the final LLM text response (if any) - LLMResponse string - - // Context provides additional context for formatting (e.g., original task description) - Context string -} - -// FormatOutput represents the result of schema formatting -type FormatOutput struct { - // Data is the validated, schema-compliant structured output - Data map[string]any - - // RawJSON is the raw JSON string returned by the formatter - RawJSON string - - // Model is the model ID used for formatting - Model string - - // Tokens contains token usage information - Tokens models.TokenUsage - - // Success indicates whether formatting succeeded - Success bool - - // Error contains any error message if formatting failed - Error string -} - -// FormatToSchema formats the given input data into the specified JSON schema -// This is a method on AgentBlockExecutor so it can reuse the existing LLM call infrastructure -func (e *AgentBlockExecutor) FormatToSchema( - ctx context.Context, - input FormatInput, - schema *models.JSONSchema, - modelID string, -) (*FormatOutput, error) { - log.Printf("📐 [FORMAT-SCHEMA] Starting schema formatting with model=%s", modelID) - - if schema == nil { - return nil, fmt.Errorf("schema is required for FormatToSchema") - } - - // Resolve model using existing method - provider, resolvedModelID, err := e.resolveModel(modelID) - if err != nil { - return nil, fmt.Errorf("failed to resolve model: %w", err) - } - - log.Printf("📐 [FORMAT-SCHEMA] Resolved model %s -> %s (provider: %s)", - modelID, resolvedModelID, provider.Name) - - // Build the formatting prompt - systemPrompt, userPrompt := buildFormattingPrompts(input, schema) - - // Build messages - messages := []map[string]any{ - {"role": "system", "content": systemPrompt}, - {"role": "user", "content": userPrompt}, - } - - // Make the LLM call with native structured output if supported - // Use existing callLLMWithRetryAndSchema for consistency - response, _, err := e.callLLMWithRetryAndSchema(ctx, provider, resolvedModelID, messages, nil, 0.1, nil, schema) - if err != nil { - return &FormatOutput{ - Success: false, - Error: fmt.Sprintf("LLM call failed: %v", err), - Model: resolvedModelID, - }, nil - } - - log.Printf("📐 [FORMAT-SCHEMA] LLM response received, length=%d chars", len(response.Content)) - - // Parse and validate the output - output, err := parseAndValidateSchema(response.Content, schema) - if err != nil { - log.Printf("⚠️ [FORMAT-SCHEMA] Validation failed: %v", err) - return &FormatOutput{ - Success: false, - Error: fmt.Sprintf("validation failed: %v", err), - RawJSON: response.Content, - Model: resolvedModelID, - Tokens: models.TokenUsage{ - Input: response.InputTokens, - Output: response.OutputTokens, - }, - }, nil - } - - log.Printf("✅ [FORMAT-SCHEMA] Successfully formatted data to schema") - - return &FormatOutput{ - Data: output, - RawJSON: response.Content, - Model: resolvedModelID, - Tokens: models.TokenUsage{ - Input: response.InputTokens, - Output: response.OutputTokens, - }, - Success: true, - }, nil -} - -// buildFormattingPrompts creates the system and user prompts for schema formatting -func buildFormattingPrompts(input FormatInput, schema *models.JSONSchema) (string, string) { - // Build schema description - schemaJSON, _ := json.MarshalIndent(schema, "", " ") - - // Build data description - var dataBuilder strings.Builder - - // Add tool results if present - if len(input.ToolResults) > 0 { - dataBuilder.WriteString("## Tool Execution Results\n") - for i, tr := range input.ToolResults { - trJSON, _ := json.MarshalIndent(tr, "", " ") - dataBuilder.WriteString(fmt.Sprintf("### Tool Result %d\n```json\n%s\n```\n\n", i+1, string(trJSON))) - } - } - - // Add LLM response if present - if input.LLMResponse != "" { - dataBuilder.WriteString("## LLM Response\n") - dataBuilder.WriteString("```\n") - dataBuilder.WriteString(input.LLMResponse) - dataBuilder.WriteString("\n```\n\n") - } - - // Add raw data if present and different from LLM response - if input.RawData != nil { - rawStr := fmt.Sprintf("%v", input.RawData) - if rawStr != input.LLMResponse { - dataBuilder.WriteString("## Additional Data\n") - if rawJSON, err := json.MarshalIndent(input.RawData, "", " "); err == nil { - dataBuilder.WriteString("```json\n") - dataBuilder.WriteString(string(rawJSON)) - dataBuilder.WriteString("\n```\n\n") - } else { - dataBuilder.WriteString("```\n") - dataBuilder.WriteString(rawStr) - dataBuilder.WriteString("\n```\n\n") - } - } - } - - systemPrompt := fmt.Sprintf(`You are a precise data formatter. Your ONLY task is to extract data from the provided sources and format it as JSON matching the required schema. - -## CRITICAL RULES -1. Respond with ONLY valid JSON - no explanations, no markdown code blocks, no extra text -2. The JSON must exactly match the required schema structure -3. Extract ALL relevant data from the provided sources -4. If data is missing for a required field, use reasonable defaults or null -5. DO NOT invent or fabricate data - only use what's provided -6. DO NOT include fields not in the schema - -## Required Output Schema -%s - -## Data Fields Explanation -%s`, string(schemaJSON), buildSchemaFieldsExplanation(schema)) - - contextNote := "" - if input.Context != "" { - contextNote = fmt.Sprintf("\n\n## Context\n%s", input.Context) - } - - userPrompt := fmt.Sprintf(`Format the following data into the required JSON schema. - -%s%s - -Respond with ONLY the JSON object, nothing else.`, dataBuilder.String(), contextNote) - - return systemPrompt, userPrompt -} - -// buildSchemaFieldsExplanation creates a human-readable explanation of schema fields -func buildSchemaFieldsExplanation(schema *models.JSONSchema) string { - if schema == nil || schema.Properties == nil { - return "No specific field requirements." - } - - var builder strings.Builder - for fieldName, fieldSchema := range schema.Properties { - builder.WriteString(fmt.Sprintf("- **%s**: ", fieldName)) - if fieldSchema.Description != "" { - builder.WriteString(fieldSchema.Description) - } else if fieldSchema.Type != "" { - builder.WriteString(fmt.Sprintf("(%s)", fieldSchema.Type)) - } - builder.WriteString("\n") - } - return builder.String() -} - -// parseAndValidateSchema parses JSON and validates against schema -func parseAndValidateSchema(content string, schema *models.JSONSchema) (map[string]any, error) { - // Extract JSON from content (handle markdown code blocks if any slipped through) - jsonContent := extractJSONFromContent(content) - - // Parse JSON - var output map[string]any - if err := json.Unmarshal([]byte(jsonContent), &output); err != nil { - return nil, fmt.Errorf("failed to parse JSON: %w", err) - } - - // Validate required fields - for _, required := range schema.Required { - if _, exists := output[required]; !exists { - return nil, fmt.Errorf("missing required field: %s", required) - } - } - - // Validate property types - for propName, propSchema := range schema.Properties { - val, exists := output[propName] - if !exists { - continue // Not required, skip - } - - if err := validateFieldType(val, propSchema.Type); err != nil { - return nil, fmt.Errorf("field %s: %w", propName, err) - } - } - - return output, nil -} - -// extractJSONFromContent extracts JSON from content that may have markdown wrappers -func extractJSONFromContent(content string) string { - content = strings.TrimSpace(content) - - // Check for markdown JSON code block - jsonBlockRegex := regexp.MustCompile("```(?:json)?\\s*([\\s\\S]*?)```") - if matches := jsonBlockRegex.FindStringSubmatch(content); len(matches) > 1 { - return strings.TrimSpace(matches[1]) - } - - // Try to find JSON object - start := strings.Index(content, "{") - if start == -1 { - return content - } - - // Find matching closing brace - depth := 0 - for i := start; i < len(content); i++ { - if content[i] == '{' { - depth++ - } else if content[i] == '}' { - depth-- - if depth == 0 { - return content[start : i+1] - } - } - } - - return content[start:] -} - -// validateFieldType checks if a value matches the expected JSON schema type -func validateFieldType(val any, expectedType string) error { - if val == nil { - return nil // null is valid for any type - } - - switch expectedType { - case "string": - if _, ok := val.(string); !ok { - return fmt.Errorf("expected string, got %T", val) - } - case "number", "integer": - switch val.(type) { - case float64, float32, int, int32, int64: - // OK - default: - return fmt.Errorf("expected number, got %T", val) - } - case "boolean": - if _, ok := val.(bool); !ok { - return fmt.Errorf("expected boolean, got %T", val) - } - case "array": - if _, ok := val.([]any); !ok { - return fmt.Errorf("expected array, got %T", val) - } - case "object": - if _, ok := val.(map[string]any); !ok { - return fmt.Errorf("expected object, got %T", val) - } - } - - return nil -} diff --git a/backend/internal/execution/llm_executor.go b/backend/internal/execution/llm_executor.go deleted file mode 100644 index 26bb613c..00000000 --- a/backend/internal/execution/llm_executor.go +++ /dev/null @@ -1,527 +0,0 @@ -package execution - -import ( - "bytes" - "claraverse/internal/models" - "claraverse/internal/services" - "context" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "regexp" - "strings" - "time" -) - -// LLMExecutor executes LLM inference blocks -type LLMExecutor struct { - chatService *services.ChatService - providerService *services.ProviderService - httpClient *http.Client -} - -// NewLLMExecutor creates a new LLM executor -func NewLLMExecutor(chatService *services.ChatService, providerService *services.ProviderService) *LLMExecutor { - return &LLMExecutor{ - chatService: chatService, - providerService: providerService, - httpClient: &http.Client{ - Timeout: 120 * time.Second, - }, - } -} - -// getDefaultModel returns the first available model from database -func (e *LLMExecutor) getDefaultModel() string { - // Use chatService to get optimal text model - provider, modelID, err := e.chatService.GetTextProviderWithModel() - if err == nil && modelID != "" { - log.Printf("🎯 [LLM-EXEC] Using dynamic default model: %s (provider: %s)", modelID, provider.Name) - return modelID - } - - // If that fails, try to get default provider with model - provider, modelID, err = e.chatService.GetDefaultProviderWithModel() - if err == nil && modelID != "" { - log.Printf("🎯 [LLM-EXEC] Using fallback default model: %s", modelID) - return modelID - } - - // Last resort: return empty string (will cause error later if no model specified) - log.Printf("⚠️ [LLM-EXEC] No default model available - LLM execution will require explicit model") - return "" -} - -// Execute runs an LLM inference block -func (e *LLMExecutor) Execute(ctx context.Context, block models.Block, inputs map[string]any) (map[string]any, error) { - config := block.Config - - // Get configuration - support both "model" and "modelId" field names - modelID := getString(config, "model", "") - if modelID == "" { - modelID = getString(config, "modelId", "") - } - - // Check for workflow-level model override (set in Start block) - if workflowModelID, ok := inputs["_workflowModelId"].(string); ok && workflowModelID != "" { - log.Printf("🎯 [LLM-EXEC] Block '%s': Using workflow model override: %s", block.Name, workflowModelID) - modelID = workflowModelID - } - - // Default to a sensible model if not specified - if modelID == "" { - modelID = e.getDefaultModel() - if modelID != "" { - log.Printf("⚠️ [LLM-EXEC] No model specified for block '%s', using default: %s", block.Name, modelID) - } else { - return nil, fmt.Errorf("no model specified and no default model available") - } - } - - // Support both "systemPrompt" and "system_prompt" field names - systemPrompt := getString(config, "systemPrompt", "") - if systemPrompt == "" { - systemPrompt = getString(config, "system_prompt", "") - } - - // Support "userPrompt", "userPromptTemplate", "user_prompt" field names - userPromptTemplate := getString(config, "userPrompt", "") - if userPromptTemplate == "" { - userPromptTemplate = getString(config, "userPromptTemplate", "") - } - if userPromptTemplate == "" { - userPromptTemplate = getString(config, "user_prompt", "") - } - temperature := getFloat(config, "temperature", 0.7) - - // Get structured output configuration - outputFormat := getString(config, "outputFormat", "text") - var outputSchema map[string]interface{} - if schema, ok := config["outputSchema"].(map[string]interface{}); ok { - outputSchema = schema - } - - // Interpolate variables in prompts - userPrompt := interpolateTemplate(userPromptTemplate, inputs) - systemPrompt = interpolateTemplate(systemPrompt, inputs) - - log.Printf("🤖 [LLM-EXEC] Block '%s': model=%s, prompt_len=%d", block.Name, modelID, len(userPrompt)) - - // Find provider for this model using multi-step resolution - var provider *models.Provider - var actualModelID string - var err error - - // Step 1: Try direct lookup in models table - provider, err = e.providerService.GetByModelID(modelID) - if err == nil { - actualModelID = modelID - log.Printf("✅ [LLM-EXEC] Found model '%s' via direct lookup", modelID) - } else { - // Step 2: Try model alias resolution - log.Printf("🔄 [LLM-EXEC] Model '%s' not found directly, trying alias resolution...", modelID) - aliasProvider, aliasModel, found := e.chatService.ResolveModelAlias(modelID) - if found { - provider = aliasProvider - actualModelID = aliasModel - log.Printf("✅ [LLM-EXEC] Resolved alias '%s' -> '%s'", modelID, actualModelID) - } else { - // Step 3: Fallback to default provider with an actual model from the database - log.Printf("⚠️ [LLM-EXEC] Model '%s' not found, using default provider with default model", modelID) - defaultProvider, defaultModel, defaultErr := e.chatService.GetDefaultProviderWithModel() - if defaultErr != nil { - return nil, fmt.Errorf("failed to find provider for model %s and no default provider available: %w", modelID, defaultErr) - } - provider = defaultProvider - actualModelID = defaultModel - log.Printf("⚠️ [LLM-EXEC] Using default provider '%s' with model '%s'", provider.Name, actualModelID) - } - } - - // Use the resolved model ID - modelID = actualModelID - - // Build request - messages := []map[string]string{ - {"role": "user", "content": userPrompt}, - } - if systemPrompt != "" { - messages = append([]map[string]string{{"role": "system", "content": systemPrompt}}, messages...) - } - - requestBody := map[string]interface{}{ - "model": modelID, - "messages": messages, - "temperature": temperature, - "stream": false, // Non-streaming for block execution - } - - // Add structured output if configured (provider-aware implementation) - if outputFormat == "json" && outputSchema != nil { - // Detect provider capability for strict schema support - supportsStrictSchema := supportsStrictJSONSchema(provider.Name, provider.BaseURL) - - if supportsStrictSchema { - // Full support: Use strict JSON schema mode (OpenAI, some OpenRouter models) - requestBody["response_format"] = map[string]interface{}{ - "type": "json_schema", - "json_schema": map[string]interface{}{ - "name": fmt.Sprintf("%s_output", block.NormalizedID), - "strict": true, - "schema": outputSchema, - }, - } - log.Printf("🎯 [LLM-EXEC] Block '%s': Using strict JSON schema mode", block.Name) - } else { - // Fallback: JSON mode + schema in system prompt - requestBody["response_format"] = map[string]interface{}{ - "type": "json_object", - } - - // Add schema to system prompt for better compliance - schemaJSON, _ := json.Marshal(outputSchema) - systemPrompt += fmt.Sprintf("\n\nIMPORTANT: Return your response as valid JSON matching this EXACT schema:\n%s\n\nDo not add any extra fields. Include all required fields.", string(schemaJSON)) - - // Update messages with enhanced prompt - messages = []map[string]string{ - {"role": "user", "content": userPrompt}, - } - if systemPrompt != "" { - messages = append([]map[string]string{{"role": "system", "content": systemPrompt}}, messages...) - } - requestBody["messages"] = messages - - log.Printf("⚠️ [LLM-EXEC] Block '%s': Using JSON mode with schema in prompt (provider fallback)", block.Name) - } - } else if outputFormat == "json" { - // Fallback to basic JSON mode if no schema provided - requestBody["response_format"] = map[string]interface{}{ - "type": "json_object", - } - log.Printf("🎯 [LLM-EXEC] Block '%s': Using basic JSON output mode", block.Name) - } - - bodyBytes, err := json.Marshal(requestBody) - if err != nil { - return nil, fmt.Errorf("failed to marshal request: %w", err) - } - - // Create request - endpoint := strings.TrimSuffix(provider.BaseURL, "/") + "/chat/completions" - req, err := http.NewRequestWithContext(ctx, "POST", endpoint, bytes.NewReader(bodyBytes)) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Authorization", "Bearer "+provider.APIKey) - - // Execute request - resp, err := e.httpClient.Do(req) - if err != nil { - return nil, fmt.Errorf("LLM request failed: %w", err) - } - defer resp.Body.Close() - - if resp.StatusCode != http.StatusOK { - body, _ := io.ReadAll(resp.Body) - return nil, fmt.Errorf("LLM request failed with status %d: %s", resp.StatusCode, string(body)) - } - - // Parse response - var result map[string]interface{} - if err := json.NewDecoder(resp.Body).Decode(&result); err != nil { - return nil, fmt.Errorf("failed to parse LLM response: %w", err) - } - - // Extract content from OpenAI-style response - content := "" - if choices, ok := result["choices"].([]interface{}); ok && len(choices) > 0 { - if choice, ok := choices[0].(map[string]interface{}); ok { - if message, ok := choice["message"].(map[string]interface{}); ok { - if c, ok := message["content"].(string); ok { - content = c - } - } - } - } - - // Extract token usage - var inputTokens, outputTokens int - if usage, ok := result["usage"].(map[string]interface{}); ok { - if pt, ok := usage["prompt_tokens"].(float64); ok { - inputTokens = int(pt) - } - if ct, ok := usage["completion_tokens"].(float64); ok { - outputTokens = int(ct) - } - } - - log.Printf("✅ [LLM-EXEC] Block '%s': completed, response_len=%d, tokens=%d/%d", - block.Name, len(content), inputTokens, outputTokens) - - // Parse JSON output if structured output was requested - if outputFormat == "json" { - var parsedJSON map[string]interface{} - if err := json.Unmarshal([]byte(content), &parsedJSON); err != nil { - log.Printf("⚠️ [LLM-EXEC] Block '%s': Failed to parse JSON output: %v", block.Name, err) - // Return raw content if JSON parsing fails - return map[string]any{ - "response": content, - "model": modelID, - "tokens": map[string]int{ - "input": inputTokens, - "output": outputTokens, - }, - "parseError": err.Error(), - }, nil - } - - log.Printf("✅ [LLM-EXEC] Block '%s': Successfully parsed JSON output with %d keys", block.Name, len(parsedJSON)) - - // Return both raw and parsed data - return map[string]any{ - "response": content, // Raw JSON string (for debugging/logging) - "data": parsedJSON, // Parsed JSON object (for downstream blocks) - "model": modelID, - "tokens": map[string]int{ - "input": inputTokens, - "output": outputTokens, - }, - }, nil - } - - // Text output (default) - return map[string]any{ - "response": content, - "model": modelID, - "tokens": map[string]int{ - "input": inputTokens, - "output": outputTokens, - }, - }, nil -} - -// interpolateTemplate replaces {{variable}} placeholders with actual values -func interpolateTemplate(template string, inputs map[string]any) string { - if template == "" { - return "" - } - - // Match {{path.to.value}} or {{path[0].value}} - re := regexp.MustCompile(`\{\{([^}]+)\}\}`) - - return re.ReplaceAllStringFunc(template, func(match string) string { - // Extract the path (remove {{ and }}) - path := strings.TrimPrefix(strings.TrimSuffix(match, "}}"), "{{") - path = strings.TrimSpace(path) - - // Debug logging - log.Printf("🔍 [INTERPOLATE] Resolving '%s' from inputs: %+v", path, inputs) - - // Resolve the path in inputs - value := resolvePath(inputs, path) - if value == nil { - log.Printf("⚠️ [INTERPOLATE] Failed to resolve '%s', keeping original", path) - return match // Keep original if not found - } - - log.Printf("✅ [INTERPOLATE] Resolved '%s' = %v", path, value) - - // Convert to string - switch v := value.(type) { - case string: - return v - case float64: - if v == float64(int(v)) { - return fmt.Sprintf("%d", int(v)) - } - return fmt.Sprintf("%g", v) - case int: - return fmt.Sprintf("%d", v) - case bool: - return fmt.Sprintf("%t", v) - default: - // For complex types, JSON encode - jsonBytes, err := json.Marshal(v) - if err != nil { - return match - } - return string(jsonBytes) - } - }) -} - -// interpolateMapValues recursively interpolates template strings in map values -func interpolateMapValues(data map[string]any, inputs map[string]any) map[string]any { - result := make(map[string]any) - - for key, value := range data { - result[key] = interpolateValue(value, inputs) - } - - return result -} - -// interpolateValue interpolates a single value (handles strings, maps, slices) -func interpolateValue(value any, inputs map[string]any) any { - switch v := value.(type) { - case string: - // Interpolate string templates - return interpolateTemplate(v, inputs) - case map[string]any: - // Recursively interpolate nested maps - return interpolateMapValues(v, inputs) - case []any: - // Interpolate each element in slices - result := make([]any, len(v)) - for i, elem := range v { - result[i] = interpolateValue(elem, inputs) - } - return result - default: - // Return as-is for other types - return value - } -} - -// resolvePath resolves a dot-notation path in a map -// Supports: input.field, input.nested.field, input[0].field -// Uses exact string matching with normalized block IDs -func resolvePath(data map[string]any, path string) any { - parts := strings.Split(path, ".") - var current any = data - - for _, part := range parts { - if current == nil { - return nil - } - - // Check for array access: field[0] - if idx := strings.Index(part, "["); idx != -1 { - fieldName := part[:idx] - indexStr := strings.TrimSuffix(part[idx+1:], "]") - - // Get the field - if m, ok := current.(map[string]any); ok { - current = m[fieldName] - } else { - return nil - } - - // Get the array element - if arr, ok := current.([]any); ok { - var index int - fmt.Sscanf(indexStr, "%d", &index) - if index >= 0 && index < len(arr) { - current = arr[index] - } else { - return nil - } - } else { - return nil - } - } else { - // Simple field access - exact match only - if m, ok := current.(map[string]any); ok { - val, exists := m[part] - if !exists { - return nil - } - current = val - } else { - return nil - } - } - } - - return current -} - -// Helper functions for config access -func getString(config map[string]any, key, defaultVal string) string { - if v, ok := config[key]; ok { - if s, ok := v.(string); ok { - return s - } - } - return defaultVal -} - -func getFloat(config map[string]any, key string, defaultVal float64) float64 { - if v, ok := config[key]; ok { - switch f := v.(type) { - case float64: - return f - case int: - return float64(f) - } - } - return defaultVal -} - -func getMap(config map[string]any, key string) map[string]any { - if v, ok := config[key]; ok { - if m, ok := v.(map[string]any); ok { - return m - } - } - return nil -} - -// supportsStrictJSONSchema determines if a provider supports OpenAI's strict JSON schema mode -// Based on comprehensive testing results (Jan 2026) with 100% compliance validation -func supportsStrictJSONSchema(providerName, baseURL string) bool { - // Normalize provider name and base URL for comparison - name := strings.ToLower(providerName) - url := strings.ToLower(baseURL) - - // ✅ TIER 1: Proven 100% compliance with strict mode - - // OpenAI - 100% compliance (tested: gpt-4.1, gpt-4.1-mini) - // Response time: 3.8-4.1s - if strings.Contains(name, "openai") || strings.Contains(url, "api.openai.com") { - return true - } - - // Gemini via OpenRouter - 100% compliance, FASTEST (819ms-1.4s) - // Models: gemini-3-flash-preview, gemini-2.5-flash-lite-preview - if strings.Contains(url, "openrouter.ai") { - // Enable strict mode for OpenRouter - Gemini models have proven 100% compliance - return true - } - - // ClaraVerse Cloud (private TEE) - Mixed results, use fallback for safety - // ✅ 100% compliance: Kimi-K2-Thinking-TEE, MiMo-V2-Flash - // ❌ 0% compliance: GLM-4.7-TEE (accepts strict mode but returns invalid JSON) - // Decision: Use fallback mode to ensure consistency across all models - if strings.Contains(url, "llm.chutes.ai") || strings.Contains(url, "chutes.ai") { - return false // Use fallback mode with prompt-based schema - } - - // ❌ TIER 2: Providers that claim support but fail compliance - - // Z.AI - Accepts strict mode but 0% compliance (returns invalid JSON) - // Models tested: glm-4.5, glm-4.7 (both 0% compliance) - if strings.Contains(name, "z.ai") || strings.Contains(url, "api.z.ai") { - return false - } - - // 0G AI - Mixed results, some models 0% compliance - // Use fallback mode for reliability - if strings.Contains(url, "13.235.83.18:4002") { - return false - } - - // Groq - supports json_object but not strict json_schema (as of Jan 2026) - if strings.Contains(name, "groq") || strings.Contains(url, "groq.com") { - return false - } - - // Default: Conservative fallback for untested providers - // Use JSON mode + prompt-based schema enforcement - return false -} - diff --git a/backend/internal/execution/tool_executor.go b/backend/internal/execution/tool_executor.go deleted file mode 100644 index bf4e7376..00000000 --- a/backend/internal/execution/tool_executor.go +++ /dev/null @@ -1,229 +0,0 @@ -package execution - -import ( - "claraverse/internal/models" - "claraverse/internal/services" - "claraverse/internal/tools" - "context" - "encoding/json" - "fmt" - "log" - "strings" -) - -// ToolExecutor executes tool blocks using the tool registry -// This executor runs tools directly without LLM involvement for faster, deterministic execution -type ToolExecutor struct { - registry *tools.Registry - credentialService *services.CredentialService -} - -// NewToolExecutor creates a new tool executor -func NewToolExecutor(registry *tools.Registry, credentialService *services.CredentialService) *ToolExecutor { - return &ToolExecutor{ - registry: registry, - credentialService: credentialService, - } -} - -// deepInterpolate recursively interpolates {{...}} templates in nested structures -func deepInterpolate(value interface{}, inputs map[string]any) interface{} { - switch v := value.(type) { - case string: - // Handle string interpolation - if strings.HasPrefix(v, "{{") && strings.HasSuffix(v, "}}") { - resolvedPath := strings.TrimPrefix(v, "{{") - resolvedPath = strings.TrimSuffix(resolvedPath, "}}") - resolvedPath = strings.TrimSpace(resolvedPath) - resolved := resolvePath(inputs, resolvedPath) - if resolved != nil { - return resolved - } - } - return v - case map[string]interface{}: - // Handle nested maps - result := make(map[string]interface{}) - for k, val := range v { - result[k] = deepInterpolate(val, inputs) - } - return result - case []interface{}: - // Handle arrays - result := make([]interface{}, len(v)) - for i, val := range v { - result[i] = deepInterpolate(val, inputs) - } - return result - default: - // Return primitives as-is (numbers, booleans, nil) - return v - } -} - -// Execute runs a tool block -func (e *ToolExecutor) Execute(ctx context.Context, block models.Block, inputs map[string]any) (map[string]any, error) { - config := block.Config - - toolName := getString(config, "toolName", "") - if toolName == "" { - return nil, fmt.Errorf("toolName is required for tool execution block") - } - - log.Printf("🔧 [TOOL-EXEC] Block '%s': executing tool '%s'", block.Name, toolName) - - // Get tool from registry - tool, exists := e.registry.Get(toolName) - if !exists { - return nil, fmt.Errorf("tool not found: %s", toolName) - } - - // Map inputs to tool arguments based on argumentMapping config - argMapping := getMap(config, "argumentMapping") - args := make(map[string]interface{}) - - log.Printf("🔧 [TOOL-EXEC] Block '%s' config: %+v", block.Name, config) - log.Printf("🔧 [TOOL-EXEC] Block '%s' argumentMapping: %+v", block.Name, argMapping) - log.Printf("🔧 [TOOL-EXEC] Block '%s' inputs keys: %v", block.Name, getInputKeys(inputs)) - - if argMapping != nil { - for argName, inputPath := range argMapping { - // Use deep interpolation to handle nested structures - interpolated := deepInterpolate(inputPath, inputs) - - // Log the interpolation result - if pathStr, ok := inputPath.(string); ok && strings.HasPrefix(pathStr, "{{") { - log.Printf("🔧 [TOOL-EXEC] Interpolated '%s': %v", argName, interpolated) - } else if _, isMap := inputPath.(map[string]interface{}); isMap { - log.Printf("🔧 [TOOL-EXEC] Deep interpolated object '%s'", argName) - } else { - log.Printf("🔧 [TOOL-EXEC] Using literal value for '%s': %v", argName, inputPath) - } - - if interpolated != nil { - args[argName] = interpolated - } - } - } else { - // If no argument mapping, pass all inputs as args - for k, v := range inputs { - // Skip internal fields - if len(k) > 0 && k[0] == '_' { - continue - } - args[k] = v - } - } - - // Extract userID from inputs for credential resolution (uses __user_id__ convention) - userID, _ := inputs["__user_id__"].(string) - - // Inject credentials for tools that need them - e.injectCredentials(ctx, toolName, args, userID, config) - - log.Printf("🔧 [TOOL-EXEC] Tool '%s' args: %v", toolName, args) - - // Execute tool - result, err := tool.Execute(args) - - // Clean up internal keys from args (don't log them) - delete(args, tools.CredentialResolverKey) - delete(args, tools.UserIDKey) - - if err != nil { - log.Printf("❌ [TOOL-EXEC] Tool '%s' failed: %v", toolName, err) - return nil, fmt.Errorf("tool execution failed: %w", err) - } - - log.Printf("✅ [TOOL-EXEC] Tool '%s' completed, result_len=%d", toolName, len(result)) - - // Try to parse result as JSON for structured output - var parsedResult any - if err := json.Unmarshal([]byte(result), &parsedResult); err != nil { - // Not JSON, use as string - parsedResult = result - } - - return map[string]any{ - "response": parsedResult, // Primary output key for consistency with other blocks - "result": parsedResult, // Kept for backwards compatibility - "data": parsedResult, // For structured data access - "toolName": toolName, - "raw": result, - }, nil -} - -// injectCredentials adds credential resolver and auto-discovers credentials for tools that need them -func (e *ToolExecutor) injectCredentials(ctx context.Context, toolName string, args map[string]interface{}, userID string, config map[string]any) { - if e.credentialService == nil || userID == "" { - return - } - - // Inject credential resolver for tools that need authentication - // Cast to tools.CredentialResolver type for proper type assertion in credential_helper.go - resolver := tools.CredentialResolver(e.credentialService.CreateCredentialResolver(userID)) - args[tools.CredentialResolverKey] = resolver - args[tools.UserIDKey] = userID - - // Auto-inject credential_id for tools that need it - toolIntegrationType := tools.GetIntegrationTypeForTool(toolName) - if toolIntegrationType == "" { - return - } - - var credentialID string - - // First, try to find from explicitly configured credentials in block config - if credentials, ok := config["credentials"].([]interface{}); ok && len(credentials) > 0 { - for _, credID := range credentials { - if credIDStr, ok := credID.(string); ok { - cred, err := resolver(credIDStr) - if err == nil && cred != nil && cred.IntegrationType == toolIntegrationType { - credentialID = credIDStr - log.Printf("🔐 [TOOL-EXEC] Found credential_id=%s from block config for tool=%s", - credentialID, toolName) - break - } - } - } - } - - // If no credential found in block config, try runtime auto-discovery from user's credentials - if credentialID == "" { - log.Printf("🔍 [TOOL-EXEC] No credentials in block config for tool=%s, trying runtime auto-discovery...", toolName) - userCreds, err := e.credentialService.ListByUserAndType(ctx, userID, toolIntegrationType) - if err != nil { - log.Printf("⚠️ [TOOL-EXEC] Failed to fetch user credentials: %v", err) - } else if len(userCreds) == 1 { - // Exactly one credential of this type - auto-use it - credentialID = userCreds[0].ID - log.Printf("🔐 [TOOL-EXEC] Runtime auto-discovered single credential: %s (%s) for tool=%s", - userCreds[0].Name, credentialID, toolName) - } else if len(userCreds) > 1 { - log.Printf("⚠️ [TOOL-EXEC] Multiple credentials (%d) found for %s - cannot auto-select. Configure in block settings.", - len(userCreds), toolIntegrationType) - } else { - log.Printf("⚠️ [TOOL-EXEC] No %s credentials found for user. Please add one in Credentials Manager.", - toolIntegrationType) - } - } - - // Inject the credential_id if we found one - if credentialID != "" { - args["credential_id"] = credentialID - log.Printf("🔐 [TOOL-EXEC] Auto-injected credential_id=%s for tool=%s (type=%s)", - credentialID, toolName, toolIntegrationType) - } -} - -// getInputKeys returns sorted keys from inputs map for logging -func getInputKeys(inputs map[string]any) []string { - keys := make([]string, 0, len(inputs)) - for k := range inputs { - // Skip internal fields for cleaner logging - if !strings.HasPrefix(k, "__") { - keys = append(keys, k) - } - } - return keys -} diff --git a/backend/internal/execution/variable_executor.go b/backend/internal/execution/variable_executor.go deleted file mode 100644 index 73c7bffe..00000000 --- a/backend/internal/execution/variable_executor.go +++ /dev/null @@ -1,318 +0,0 @@ -package execution - -import ( - "claraverse/internal/filecache" - "claraverse/internal/models" - "claraverse/internal/security" - "context" - "fmt" - "log" - "os" - "path/filepath" - "strings" - "time" -) - -// FileReference represents a file that can be passed between workflow blocks -type FileReference struct { - FileID string `json:"file_id"` - Filename string `json:"filename"` - MimeType string `json:"mime_type"` - Size int64 `json:"size"` - Type string `json:"type"` // "image", "document", "audio", "data" -} - -// isFileReference checks if a value is a file reference (map with file_id) -func isFileReference(value any) bool { - if m, ok := value.(map[string]any); ok { - _, hasFileID := m["file_id"] - return hasFileID - } - return false -} - -// getFileType determines the file type category from MIME type -func getFileType(mimeType string) string { - switch { - case strings.HasPrefix(mimeType, "image/"): - return "image" - case strings.HasPrefix(mimeType, "audio/"): - return "audio" - case mimeType == "application/pdf", - mimeType == "application/vnd.openxmlformats-officedocument.wordprocessingml.document", - mimeType == "application/vnd.openxmlformats-officedocument.presentationml.presentation", - mimeType == "application/msword": - return "document" - case mimeType == "application/json", - mimeType == "text/csv", - mimeType == "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet", - strings.HasPrefix(mimeType, "text/"): - return "data" - default: - return "data" - } -} - -// validateFileReference validates a file reference and enriches it with metadata -func validateFileReference(value map[string]any, userID string) (*FileReference, error) { - fileID, ok := value["file_id"].(string) - if !ok || fileID == "" { - return nil, fmt.Errorf("invalid file reference: missing file_id") - } - - // SECURITY: Validate fileID to prevent path traversal attacks - if err := security.ValidateFileID(fileID); err != nil { - return nil, fmt.Errorf("invalid file reference: %w", err) - } - - // Get file from cache service - fileCacheService := filecache.GetService() - file, found := fileCacheService.Get(fileID) - - // If not in cache, try to find on disk and restore cache entry - if !found { - log.Printf("⚠️ [VAR-EXEC] File %s not in cache, attempting disk recovery...", fileID) - - // Try to find the file on disk - uploadDir := os.Getenv("UPLOAD_DIR") - if uploadDir == "" { - uploadDir = "./uploads" - } - - // Try common extensions for data files - extensions := []string{".csv", ".xlsx", ".xls", ".json", ".txt", ".png", ".jpg", ".jpeg", ""} - var foundPath string - var foundFilename string - - for _, ext := range extensions { - testPath := filepath.Join(uploadDir, fileID+ext) - if info, err := os.Stat(testPath); err == nil { - foundPath = testPath - foundFilename = fileID + ext - log.Printf("✅ [VAR-EXEC] Found file on disk: %s (size: %d bytes)", testPath, info.Size()) - - // Restore cache entry - mimeType := getMimeTypeFromExtension(ext) - cachedFile := &filecache.CachedFile{ - FileID: fileID, - UserID: userID, // Use current user since original is unknown - Filename: foundFilename, - MimeType: mimeType, - Size: info.Size(), - FilePath: foundPath, - UploadedAt: time.Now(), - } - fileCacheService.Store(cachedFile) - file = cachedFile - found = true - break - } - } - - if !found { - return nil, fmt.Errorf("file not found or has expired: %s (checked disk at %s)", fileID, uploadDir) - } - } - - // For workflow context, we allow access if userID matches or if no userID check is needed - if userID != "" && file.UserID != "" && file.UserID != userID { - return nil, fmt.Errorf("access denied: you don't have permission to access this file") - } - - return &FileReference{ - FileID: file.FileID, - Filename: file.Filename, - MimeType: file.MimeType, - Size: file.Size, - Type: getFileType(file.MimeType), - }, nil -} - -// getMimeTypeFromExtension returns MIME type based on file extension -func getMimeTypeFromExtension(ext string) string { - switch strings.ToLower(ext) { - case ".csv": - return "text/csv" - case ".xlsx": - return "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet" - case ".xls": - return "application/vnd.ms-excel" - case ".json": - return "application/json" - case ".txt": - return "text/plain" - case ".png": - return "image/png" - case ".jpg", ".jpeg": - return "image/jpeg" - case ".gif": - return "image/gif" - case ".webp": - return "image/webp" - default: - return "application/octet-stream" - } -} - -// VariableExecutor executes variable blocks (read/set workflow variables) -type VariableExecutor struct{} - -// NewVariableExecutor creates a new variable executor -func NewVariableExecutor() *VariableExecutor { - return &VariableExecutor{} -} - -// Execute runs a variable block -func (e *VariableExecutor) Execute(ctx context.Context, block models.Block, inputs map[string]any) (map[string]any, error) { - config := block.Config - - operation := getString(config, "operation", "read") - variableName := getString(config, "variableName", "") - - // Extract user context for file validation - userID, _ := inputs["__user_id__"].(string) - - if variableName == "" { - return nil, fmt.Errorf("variableName is required for variable block") - } - - log.Printf("📦 [VAR-EXEC] Block '%s': %s variable '%s'", block.Name, operation, variableName) - - switch operation { - case "read": - // Read from inputs (workflow variables are passed as inputs) - value, ok := inputs[variableName] - if !ok || value == nil || value == "" { - // Check inputType in config to determine if we should use defaultValue (text) or fileValue (file) - inputType := getString(config, "inputType", "text") - - if inputType == "file" { - // Check for fileValue in config (used for Start block file input) - if fileValue, hasFile := config["fileValue"]; hasFile && fileValue != nil { - if fileMap, isMap := fileValue.(map[string]any); isMap { - // Validate and use file reference - fileRef, err := validateFileReference(fileMap, userID) - if err != nil { - log.Printf("⚠️ [VAR-EXEC] File reference validation failed: %v", err) - } else { - value = map[string]any{ - "file_id": fileRef.FileID, - "filename": fileRef.Filename, - "mime_type": fileRef.MimeType, - "size": fileRef.Size, - "type": fileRef.Type, - } - log.Printf("📁 [VAR-EXEC] Using fileValue for '%s': %s (%s)", variableName, fileRef.Filename, fileRef.Type) - output := map[string]any{ - "value": value, - variableName: value, - } - log.Printf("🔍 [VAR-EXEC] Output keys: %v", getKeys(output)) - return output, nil - } - } - } - } - - // Check for defaultValue in config (used for Start block text input) - defaultValue := getString(config, "defaultValue", "") - if defaultValue != "" { - log.Printf("📦 [VAR-EXEC] Using defaultValue for '%s': %s", variableName, defaultValue) - output := map[string]any{ - "value": defaultValue, - variableName: defaultValue, - } - log.Printf("🔍 [VAR-EXEC] Output keys: %v", getKeys(output)) - return output, nil - } - log.Printf("⚠️ [VAR-EXEC] Variable '%s' not found and no defaultValue/fileValue, returning nil", variableName) - output := map[string]any{ - "value": nil, - variableName: nil, - } - log.Printf("🔍 [VAR-EXEC] Output keys: %v", getKeys(output)) - return output, nil - } - - // Handle file references - validate and enrich with metadata - if isFileReference(value) { - fileRef, err := validateFileReference(value.(map[string]any), userID) - if err != nil { - log.Printf("⚠️ [VAR-EXEC] File reference validation failed: %v", err) - // Return the original value but log the warning - } else { - // Convert FileReference to map for downstream use - value = map[string]any{ - "file_id": fileRef.FileID, - "filename": fileRef.Filename, - "mime_type": fileRef.MimeType, - "size": fileRef.Size, - "type": fileRef.Type, - } - log.Printf("📁 [VAR-EXEC] Validated file reference: %s (%s)", fileRef.Filename, fileRef.Type) - } - } - - log.Printf("✅ [VAR-EXEC] Read variable '%s': %v", variableName, value) - output := map[string]any{ - "value": value, - variableName: value, - } - log.Printf("🔍 [VAR-EXEC] Output keys: %v", getKeys(output)) - return output, nil - - case "set": - // Set/transform a value - valueExpr := getString(config, "valueExpression", "") - var value any - - if valueExpr != "" { - // Resolve value from expression (path in inputs) - value = resolvePath(inputs, valueExpr) - } else { - // Check for a direct value in config - if v, ok := config["value"]; ok { - value = v - } - } - - // Handle file references - validate and enrich with metadata - if isFileReference(value) { - fileRef, err := validateFileReference(value.(map[string]any), userID) - if err != nil { - log.Printf("⚠️ [VAR-EXEC] File reference validation failed: %v", err) - // Return the original value but log the warning - } else { - // Convert FileReference to map for downstream use - value = map[string]any{ - "file_id": fileRef.FileID, - "filename": fileRef.Filename, - "mime_type": fileRef.MimeType, - "size": fileRef.Size, - "type": fileRef.Type, - } - log.Printf("📁 [VAR-EXEC] Validated file reference: %s (%s)", fileRef.Filename, fileRef.Type) - } - } - - log.Printf("✅ [VAR-EXEC] Set variable '%s' = %v", variableName, value) - output := map[string]any{ - "value": value, - variableName: value, - } - log.Printf("🔍 [VAR-EXEC] Output keys: %v", getKeys(output)) - return output, nil - - default: - return nil, fmt.Errorf("unknown variable operation: %s", operation) - } -} - -// getKeys returns the keys of a map as a slice -func getKeys(m map[string]any) []string { - keys := make([]string, 0, len(m)) - for k := range m { - keys = append(keys, k) - } - return keys -} diff --git a/backend/internal/execution/variable_executor_test.go b/backend/internal/execution/variable_executor_test.go deleted file mode 100644 index 9adad976..00000000 --- a/backend/internal/execution/variable_executor_test.go +++ /dev/null @@ -1,421 +0,0 @@ -package execution - -import ( - "context" - "testing" - - "claraverse/internal/models" -) - -// TestFileReferenceDetection tests isFileReference helper -func TestFileReferenceDetection(t *testing.T) { - testCases := []struct { - name string - value any - expected bool - }{ - { - name: "valid file reference", - value: map[string]any{"file_id": "abc123", "filename": "test.pdf"}, - expected: true, - }, - { - name: "file_id only", - value: map[string]any{"file_id": "abc123"}, - expected: true, - }, - { - name: "no file_id", - value: map[string]any{"filename": "test.pdf"}, - expected: false, - }, - { - name: "string value", - value: "just a string", - expected: false, - }, - { - name: "number value", - value: 123, - expected: false, - }, - { - name: "nil value", - value: nil, - expected: false, - }, - { - name: "empty map", - value: map[string]any{}, - expected: false, - }, - } - - for _, tc := range testCases { - t.Run(tc.name, func(t *testing.T) { - result := isFileReference(tc.value) - if result != tc.expected { - t.Errorf("isFileReference(%v) = %v, expected %v", tc.value, result, tc.expected) - } - }) - } -} - -// TestGetFileType tests MIME type categorization -func TestGetFileType(t *testing.T) { - testCases := []struct { - mimeType string - expected string - }{ - // Images - {"image/jpeg", "image"}, - {"image/png", "image"}, - {"image/gif", "image"}, - {"image/webp", "image"}, - {"image/svg+xml", "image"}, - - // Audio - {"audio/mpeg", "audio"}, - {"audio/wav", "audio"}, - {"audio/mp4", "audio"}, - {"audio/ogg", "audio"}, - - // Documents - {"application/pdf", "document"}, - {"application/vnd.openxmlformats-officedocument.wordprocessingml.document", "document"}, - {"application/vnd.openxmlformats-officedocument.presentationml.presentation", "document"}, - {"application/msword", "document"}, - - // Data files - {"application/json", "data"}, - {"text/csv", "data"}, - {"text/plain", "data"}, - {"application/vnd.openxmlformats-officedocument.spreadsheetml.sheet", "data"}, - - // Unknown defaults to data - {"application/octet-stream", "data"}, - {"video/mp4", "data"}, - } - - for _, tc := range testCases { - t.Run(tc.mimeType, func(t *testing.T) { - result := getFileType(tc.mimeType) - if result != tc.expected { - t.Errorf("getFileType(%s) = %s, expected %s", tc.mimeType, result, tc.expected) - } - }) - } -} - -// TestFileReferenceStruct tests FileReference structure -func TestFileReferenceStruct(t *testing.T) { - ref := FileReference{ - FileID: "file-123", - Filename: "document.pdf", - MimeType: "application/pdf", - Size: 12345, - Type: "document", - } - - if ref.FileID == "" { - t.Error("FileID should be set") - } - if ref.Type != "document" { - t.Errorf("Type should be 'document', got %s", ref.Type) - } -} - -// TestVariableExecutorCreation tests executor creation -func TestVariableExecutorCreation(t *testing.T) { - executor := NewVariableExecutor() - if executor == nil { - t.Fatal("NewVariableExecutor should return non-nil executor") - } -} - -// TestVariableReadOperation tests read operation -func TestVariableReadOperation(t *testing.T) { - executor := NewVariableExecutor() - - block := models.Block{ - ID: "var-block", - Name: "Read Variable", - Config: map[string]any{ - "operation": "read", - "variableName": "testVar", - }, - } - - inputs := map[string]any{ - "testVar": "hello world", - } - - result, err := executor.Execute(context.Background(), block, inputs) - if err != nil { - t.Fatalf("Execute failed: %v", err) - } - - if result["value"] != "hello world" { - t.Errorf("Expected 'hello world', got %v", result["value"]) - } - if result["testVar"] != "hello world" { - t.Errorf("Expected testVar='hello world', got %v", result["testVar"]) - } -} - -// TestVariableReadWithDefault tests read operation with default value -func TestVariableReadWithDefault(t *testing.T) { - executor := NewVariableExecutor() - - block := models.Block{ - ID: "var-block", - Name: "Read Variable", - Config: map[string]any{ - "operation": "read", - "variableName": "missingVar", - "defaultValue": "default value", - }, - } - - inputs := map[string]any{} - - result, err := executor.Execute(context.Background(), block, inputs) - if err != nil { - t.Fatalf("Execute failed: %v", err) - } - - if result["value"] != "default value" { - t.Errorf("Expected 'default value', got %v", result["value"]) - } -} - -// TestVariableReadMissing tests read of missing variable without default -func TestVariableReadMissing(t *testing.T) { - executor := NewVariableExecutor() - - block := models.Block{ - ID: "var-block", - Name: "Read Variable", - Config: map[string]any{ - "operation": "read", - "variableName": "missingVar", - }, - } - - inputs := map[string]any{} - - result, err := executor.Execute(context.Background(), block, inputs) - if err != nil { - t.Fatalf("Execute failed: %v", err) - } - - // Should return nil for missing variable - if result["value"] != nil { - t.Errorf("Expected nil, got %v", result["value"]) - } -} - -// TestVariableSetOperation tests set operation -func TestVariableSetOperation(t *testing.T) { - executor := NewVariableExecutor() - - block := models.Block{ - ID: "var-block", - Name: "Set Variable", - Config: map[string]any{ - "operation": "set", - "variableName": "newVar", - "value": "set value", - }, - } - - inputs := map[string]any{} - - result, err := executor.Execute(context.Background(), block, inputs) - if err != nil { - t.Fatalf("Execute failed: %v", err) - } - - if result["value"] != "set value" { - t.Errorf("Expected 'set value', got %v", result["value"]) - } - if result["newVar"] != "set value" { - t.Errorf("Expected newVar='set value', got %v", result["newVar"]) - } -} - -// TestVariableSetFromExpression tests set operation with expression -func TestVariableSetFromExpression(t *testing.T) { - executor := NewVariableExecutor() - - block := models.Block{ - ID: "var-block", - Name: "Set Variable", - Config: map[string]any{ - "operation": "set", - "variableName": "result", - "valueExpression": "sourceData", - }, - } - - inputs := map[string]any{ - "sourceData": "value from source", - } - - result, err := executor.Execute(context.Background(), block, inputs) - if err != nil { - t.Fatalf("Execute failed: %v", err) - } - - if result["value"] != "value from source" { - t.Errorf("Expected 'value from source', got %v", result["value"]) - } -} - -// TestVariableMissingName tests error for missing variable name -func TestVariableMissingName(t *testing.T) { - executor := NewVariableExecutor() - - block := models.Block{ - ID: "var-block", - Name: "Bad Block", - Config: map[string]any{ - "operation": "read", - // Missing variableName - }, - } - - inputs := map[string]any{} - - _, err := executor.Execute(context.Background(), block, inputs) - if err == nil { - t.Error("Expected error for missing variableName") - } -} - -// TestVariableUnknownOperation tests error for unknown operation -func TestVariableUnknownOperation(t *testing.T) { - executor := NewVariableExecutor() - - block := models.Block{ - ID: "var-block", - Name: "Bad Block", - Config: map[string]any{ - "operation": "invalid", - "variableName": "test", - }, - } - - inputs := map[string]any{} - - _, err := executor.Execute(context.Background(), block, inputs) - if err == nil { - t.Error("Expected error for unknown operation") - } -} - -// TestGetKeysHelper tests getKeys helper function -func TestGetKeysHelper(t *testing.T) { - m := map[string]any{ - "a": 1, - "b": 2, - "c": 3, - } - - keys := getKeys(m) - if len(keys) != 3 { - t.Errorf("Expected 3 keys, got %d", len(keys)) - } - - // Check all keys are present (order doesn't matter) - keyMap := make(map[string]bool) - for _, k := range keys { - keyMap[k] = true - } - - for _, expected := range []string{"a", "b", "c"} { - if !keyMap[expected] { - t.Errorf("Expected key %s not found", expected) - } - } -} - -// Benchmark tests -func BenchmarkVariableRead(b *testing.B) { - executor := NewVariableExecutor() - - block := models.Block{ - ID: "var-block", - Name: "Read Variable", - Config: map[string]any{ - "operation": "read", - "variableName": "testVar", - }, - } - - inputs := map[string]any{ - "testVar": "test value", - } - - ctx := context.Background() - - b.ResetTimer() - for i := 0; i < b.N; i++ { - executor.Execute(ctx, block, inputs) - } -} - -func BenchmarkVariableSet(b *testing.B) { - executor := NewVariableExecutor() - - block := models.Block{ - ID: "var-block", - Name: "Set Variable", - Config: map[string]any{ - "operation": "set", - "variableName": "testVar", - "value": "test value", - }, - } - - inputs := map[string]any{} - ctx := context.Background() - - b.ResetTimer() - for i := 0; i < b.N; i++ { - executor.Execute(ctx, block, inputs) - } -} - -func BenchmarkIsFileReference(b *testing.B) { - testCases := []any{ - map[string]any{"file_id": "abc123"}, - "just a string", - 123, - nil, - } - - b.ResetTimer() - for i := 0; i < b.N; i++ { - for _, tc := range testCases { - isFileReference(tc) - } - } -} - -func BenchmarkGetFileType(b *testing.B) { - mimeTypes := []string{ - "image/jpeg", - "audio/mpeg", - "application/pdf", - "application/json", - "text/plain", - } - - b.ResetTimer() - for i := 0; i < b.N; i++ { - for _, mime := range mimeTypes { - getFileType(mime) - } - } -} diff --git a/backend/internal/execution/webhook_executor.go b/backend/internal/execution/webhook_executor.go deleted file mode 100644 index d0b204bd..00000000 --- a/backend/internal/execution/webhook_executor.go +++ /dev/null @@ -1,112 +0,0 @@ -package execution - -import ( - "claraverse/internal/models" - "context" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "strings" - "time" -) - -// WebhookExecutor executes webhook blocks (HTTP requests) -type WebhookExecutor struct { - client *http.Client -} - -// NewWebhookExecutor creates a new webhook executor -func NewWebhookExecutor() *WebhookExecutor { - return &WebhookExecutor{ - client: &http.Client{Timeout: 30 * time.Second}, - } -} - -// Execute runs a webhook block -func (e *WebhookExecutor) Execute(ctx context.Context, block models.Block, inputs map[string]any) (map[string]any, error) { - config := block.Config - - url := getString(config, "url", "") - method := strings.ToUpper(getString(config, "method", "GET")) - headers := getMap(config, "headers") - bodyTemplate := getString(config, "bodyTemplate", "") - - if url == "" { - return nil, fmt.Errorf("url is required for webhook block") - } - - // Interpolate variables in URL - url = interpolateTemplate(url, inputs) - - // Interpolate variables in body - body := interpolateTemplate(bodyTemplate, inputs) - - log.Printf("🌐 [WEBHOOK-EXEC] Block '%s': %s %s", block.Name, method, url) - - // Create request - var bodyReader io.Reader - if body != "" { - bodyReader = strings.NewReader(body) - } - - req, err := http.NewRequestWithContext(ctx, method, url, bodyReader) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - // Set headers - if headers != nil { - for key, value := range headers { - if strVal, ok := value.(string); ok { - // Interpolate variables in header values (for secrets) - strVal = interpolateTemplate(strVal, inputs) - req.Header.Set(key, strVal) - } - } - } - - // Default content type for POST/PUT with body - if body != "" && req.Header.Get("Content-Type") == "" { - req.Header.Set("Content-Type", "application/json") - } - - // Execute request - resp, err := e.client.Do(req) - if err != nil { - log.Printf("❌ [WEBHOOK-EXEC] Request failed: %v", err) - return nil, fmt.Errorf("webhook request failed: %w", err) - } - defer resp.Body.Close() - - // Read response body - responseBody, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - log.Printf("✅ [WEBHOOK-EXEC] Block '%s': status=%d, body_len=%d", block.Name, resp.StatusCode, len(responseBody)) - - // Try to parse response as JSON - var parsedBody any - if err := json.Unmarshal(responseBody, &parsedBody); err != nil { - // Not JSON, use as string - parsedBody = string(responseBody) - } - - // Convert response headers to map - respHeaders := make(map[string]string) - for key, values := range resp.Header { - if len(values) > 0 { - respHeaders[key] = values[0] - } - } - - return map[string]any{ - "status": resp.StatusCode, - "body": parsedBody, - "headers": respHeaders, - "raw": string(responseBody), - }, nil -} diff --git a/backend/internal/execution/workflow_test.go b/backend/internal/execution/workflow_test.go deleted file mode 100644 index 8e9e583c..00000000 --- a/backend/internal/execution/workflow_test.go +++ /dev/null @@ -1,390 +0,0 @@ -package execution - -import ( - "claraverse/internal/models" - "context" - "testing" -) - -// TestVariableExecutorOutputsCorrectKeys tests that variable blocks output both "value" and the variable name -func TestVariableExecutorOutputsCorrectKeys(t *testing.T) { - executor := NewVariableExecutor() - - tests := []struct { - name string - block models.Block - inputs map[string]any - expectedKeys []string - }{ - { - name: "Read operation with defaultValue should output both value and variableName", - block: models.Block{ - Name: "Start", - Type: "variable", - Config: map[string]any{ - "operation": "read", - "variableName": "input", - "defaultValue": "test value", - }, - }, - inputs: map[string]any{}, - expectedKeys: []string{"value", "input"}, - }, - { - name: "Read operation with existing input should output both keys", - block: models.Block{ - Name: "Read Input", - Type: "variable", - Config: map[string]any{ - "operation": "read", - "variableName": "query", - }, - }, - inputs: map[string]any{"query": "search term"}, - expectedKeys: []string{"value", "query"}, - }, - { - name: "Set operation should output both keys", - block: models.Block{ - Name: "Set Variable", - Type: "variable", - Config: map[string]any{ - "operation": "set", - "variableName": "result", - "value": "new value", - }, - }, - inputs: map[string]any{}, - expectedKeys: []string{"value", "result"}, - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - ctx := context.Background() - output, err := executor.Execute(ctx, tt.block, tt.inputs) - - if err != nil { - t.Fatalf("Execute failed: %v", err) - } - - // Check that all expected keys are present - for _, key := range tt.expectedKeys { - if _, ok := output[key]; !ok { - t.Errorf("Expected key '%s' not found in output. Got: %+v", key, output) - } - } - - // For "read" operation, both keys should have the same value - if tt.block.Config["operation"] == "read" { - if output["value"] != output[tt.block.Config["variableName"].(string)] { - t.Errorf("'value' and variable name key should have same value. Got: %+v", output) - } - } - }) - } -} - -// TestInterpolateTemplate tests the template interpolation function -func TestInterpolateTemplate(t *testing.T) { - tests := []struct { - name string - template string - inputs map[string]any - expected string - }{ - { - name: "Simple variable interpolation", - template: "Search for {{input}}", - inputs: map[string]any{"input": "test query"}, - expected: "Search for test query", - }, - { - name: "Multiple variables", - template: "{{user}} searched for {{query}}", - inputs: map[string]any{"user": "John", "query": "golang"}, - expected: "John searched for golang", - }, - { - name: "Nested object access", - template: "Result: {{output.response}}", - inputs: map[string]any{ - "output": map[string]any{ - "response": "success", - }, - }, - expected: "Result: success", - }, - { - name: "Missing variable should keep original", - template: "Value: {{missing}}", - inputs: map[string]any{"other": "value"}, - expected: "Value: {{missing}}", - }, - { - name: "Number conversion", - template: "Count: {{count}}", - inputs: map[string]any{"count": 42}, - expected: "Count: 42", - }, - { - name: "Boolean conversion", - template: "Active: {{active}}", - inputs: map[string]any{"active": true}, - expected: "Active: true", - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - result := interpolateTemplate(tt.template, tt.inputs) - if result != tt.expected { - t.Errorf("Expected '%s', got '%s'", tt.expected, result) - } - }) - } -} - -// TestInterpolateMapValues tests the map value interpolation function -func TestInterpolateMapValues(t *testing.T) { - tests := []struct { - name string - data map[string]any - inputs map[string]any - expected map[string]any - }{ - { - name: "Interpolate string values in map", - data: map[string]any{ - "query": "{{input}}", - "type": "search", - }, - inputs: map[string]any{"input": "test query"}, - expected: map[string]any{ - "query": "test query", - "type": "search", - }, - }, - { - name: "Nested map interpolation", - data: map[string]any{ - "params": map[string]any{ - "q": "{{query}}", - }, - }, - inputs: map[string]any{"query": "golang"}, - expected: map[string]any{ - "params": map[string]any{ - "q": "golang", - }, - }, - }, - { - name: "Array interpolation", - data: map[string]any{ - "items": []any{"{{first}}", "{{second}}"}, - }, - inputs: map[string]any{"first": "a", "second": "b"}, - expected: map[string]any{ - "items": []any{"a", "b"}, - }, - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - result := interpolateMapValues(tt.data, tt.inputs) - - // Deep comparison - if !mapsEqual(result, tt.expected) { - t.Errorf("Expected %+v, got %+v", tt.expected, result) - } - }) - } -} - -// TestWorkflowDataFlow tests end-to-end data flow through a simple workflow -func TestWorkflowDataFlow(t *testing.T) { - // Create a simple workflow: Start -> Block A -> Block B - workflow := &models.Workflow{ - ID: "test-workflow", - Blocks: []models.Block{ - { - ID: "start", - Name: "Start", - Type: "variable", - Config: map[string]any{ - "operation": "read", - "variableName": "input", - "defaultValue": "test value", - }, - }, - }, - Connections: []models.Connection{}, - Variables: []models.Variable{}, - } - - // Test variable executor output - varExec := NewVariableExecutor() - ctx := context.Background() - - startOutput, err := varExec.Execute(ctx, workflow.Blocks[0], map[string]any{}) - if err != nil { - t.Fatalf("Start block failed: %v", err) - } - - t.Logf("Start block output: %+v", startOutput) - - // Verify Start block outputs both keys - if _, ok := startOutput["value"]; !ok { - t.Error("Start block should output 'value' key") - } - if _, ok := startOutput["input"]; !ok { - t.Error("Start block should output 'input' key") - } - - // Simulate engine passing Start output to next block - nextBlockInputs := map[string]any{} - // Copy globalInputs (workflow input) - nextBlockInputs["input"] = "test value" - // Add Start block outputs - for k, v := range startOutput { - nextBlockInputs[k] = v - } - - t.Logf("Next block would receive inputs: %+v", nextBlockInputs) - - // Test interpolation with these inputs - template := "Process {{input}}" - result := interpolateTemplate(template, nextBlockInputs) - expected := "Process test value" - - if result != expected { - t.Errorf("Interpolation failed. Expected '%s', got '%s'", expected, result) - } -} - -// TestWorkflowEngineBlockInputConstruction tests how engine.go constructs blockInputs -func TestWorkflowEngineBlockInputConstruction(t *testing.T) { - // Simulate what engine.go does at lines 129-154 - workflow := &models.Workflow{ - ID: "test", - Blocks: []models.Block{ - {ID: "start", Name: "Start", Type: "variable"}, - {ID: "block2", Name: "Block 2", Type: "llm_inference"}, - }, - Connections: []models.Connection{ - { - ID: "conn1", - SourceBlockID: "start", - TargetBlockID: "block2", - SourceOutput: "output", - TargetInput: "input", - }, - }, - } - - // Initial workflow input - workflowInput := map[string]any{ - "input": "GUVI HCL Scam", - } - - // Global inputs (what engine.go builds) - globalInputs := make(map[string]any) - for k, v := range workflowInput { - globalInputs[k] = v - } - - // Start block outputs - startBlockOutput := map[string]any{ - "value": "GUVI HCL Scam", - "input": "GUVI HCL Scam", - } - - // Block outputs storage - blockOutputs := map[string]map[string]any{ - "start": startBlockOutput, - } - - // Build inputs for block2 (what engine.go does) - blockInputs := make(map[string]any) - for k, v := range globalInputs { - blockInputs[k] = v - } - - // Add outputs from connected upstream blocks - for _, conn := range workflow.Connections { - if conn.TargetBlockID == "block2" { - if output, ok := blockOutputs[conn.SourceBlockID]; ok { - // Add under source block name - blockInputs["Start"] = output - - // Also add fields directly - for k, v := range output { - blockInputs[k] = v - } - } - } - } - - t.Logf("Block2 inputs: %+v", blockInputs) - - // Verify block2 has access to "input" key - if _, ok := blockInputs["input"]; !ok { - t.Error("Block2 should have 'input' key in inputs") - } - - if blockInputs["input"] != "GUVI HCL Scam" { - t.Errorf("Block2 input should be 'GUVI HCL Scam', got: %v", blockInputs["input"]) - } - - // Test interpolation would work - template := "{{input}}" - result := interpolateTemplate(template, blockInputs) - if result != "GUVI HCL Scam" { - t.Errorf("Interpolation should resolve to 'GUVI HCL Scam', got: '%s'", result) - } -} - -// Helper function for deep map comparison -func mapsEqual(a, b map[string]any) bool { - if len(a) != len(b) { - return false - } - for k, v := range a { - bv, ok := b[k] - if !ok { - return false - } - // Handle different types - switch av := v.(type) { - case map[string]any: - bvm, ok := bv.(map[string]any) - if !ok || !mapsEqual(av, bvm) { - return false - } - case []any: - bva, ok := bv.([]any) - if !ok || !slicesEqual(av, bva) { - return false - } - default: - if v != bv { - return false - } - } - } - return true -} - -func slicesEqual(a, b []any) bool { - if len(a) != len(b) { - return false - } - for i := range a { - if a[i] != b[i] { - return false - } - } - return true -} diff --git a/backend/internal/filecache/filecache.go b/backend/internal/filecache/filecache.go deleted file mode 100644 index b9d54334..00000000 --- a/backend/internal/filecache/filecache.go +++ /dev/null @@ -1,426 +0,0 @@ -package filecache - -import ( - "claraverse/internal/security" - "fmt" - "log" - "os" - "sync" - "time" - - "github.com/patrickmn/go-cache" -) - -// CachedFile represents a file stored in memory cache -type CachedFile struct { - FileID string - UserID string - ConversationID string - ExtractedText *security.SecureString // For PDFs - FileHash security.Hash - Filename string - MimeType string - Size int64 - PageCount int // For PDFs - WordCount int // For PDFs - FilePath string // For images (disk location) - UploadedAt time.Time -} - -// Service manages uploaded files in memory -type Service struct { - cache *cache.Cache - mu sync.RWMutex -} - -var ( - instance *Service - once sync.Once -) - -// GetService returns the singleton file cache service -func GetService() *Service { - once.Do(func() { - instance = NewService() - }) - return instance -} - -// NewService creates a new file cache service -func NewService() *Service { - c := cache.New(30*time.Minute, 10*time.Minute) - - // Set eviction handler for secure wiping - c.OnEvicted(func(key string, value interface{}) { - if file, ok := value.(*CachedFile); ok { - log.Printf("🗑️ [FILE-CACHE] Evicting file %s (%s) - secure wiping memory", file.FileID, file.Filename) - file.SecureWipe() - } - }) - - return &Service{ - cache: c, - } -} - -// Store stores a file in the cache -func (s *Service) Store(file *CachedFile) { - s.mu.Lock() - defer s.mu.Unlock() - s.cache.Set(file.FileID, file, cache.DefaultExpiration) - log.Printf("📦 [FILE-CACHE] Stored file %s (%s) - %d bytes, %d words", - file.FileID, file.Filename, file.Size, file.WordCount) -} - -// Get retrieves a file from the cache -func (s *Service) Get(fileID string) (*CachedFile, bool) { - s.mu.RLock() - defer s.mu.RUnlock() - - value, found := s.cache.Get(fileID) - if !found { - return nil, false - } - - file, ok := value.(*CachedFile) - if !ok { - return nil, false - } - - return file, true -} - -// GetByUserAndConversation retrieves a file if it belongs to the user and conversation -func (s *Service) GetByUserAndConversation(fileID, userID, conversationID string) (*CachedFile, error) { - file, found := s.Get(fileID) - if !found { - return nil, fmt.Errorf("file not found or expired") - } - - // Verify ownership - if file.UserID != userID { - return nil, fmt.Errorf("access denied: file belongs to different user") - } - - // Verify conversation - if file.ConversationID != conversationID { - return nil, fmt.Errorf("file belongs to different conversation") - } - - return file, nil -} - -// GetByUser retrieves a file if it belongs to the user (ignores conversation) -func (s *Service) GetByUser(fileID, userID string) (*CachedFile, error) { - file, found := s.Get(fileID) - if !found { - return nil, fmt.Errorf("file not found or expired") - } - - // Verify ownership - if file.UserID != userID { - return nil, fmt.Errorf("access denied: file belongs to different user") - } - - return file, nil -} - -// GetFilesForConversation returns all files for a conversation -func (s *Service) GetFilesForConversation(conversationID string) []*CachedFile { - s.mu.RLock() - defer s.mu.RUnlock() - - var files []*CachedFile - for _, item := range s.cache.Items() { - if file, ok := item.Object.(*CachedFile); ok { - if file.ConversationID == conversationID { - files = append(files, file) - } - } - } - - return files -} - -// GetConversationFiles returns all file IDs for a conversation -func (s *Service) GetConversationFiles(conversationID string) []string { - files := s.GetFilesForConversation(conversationID) - fileIDs := make([]string, 0, len(files)) - for _, file := range files { - fileIDs = append(fileIDs, file.FileID) - } - return fileIDs -} - -// Delete removes a file from the cache and securely wipes it -func (s *Service) Delete(fileID string) { - s.mu.Lock() - defer s.mu.Unlock() - - // Get the file first to wipe it - if value, found := s.cache.Get(fileID); found { - if file, ok := value.(*CachedFile); ok { - log.Printf("🗑️ [FILE-CACHE] Deleting file %s (%s)", file.FileID, file.Filename) - file.SecureWipe() - } - } - - s.cache.Delete(fileID) -} - -// DeleteConversationFiles deletes all files for a conversation -func (s *Service) DeleteConversationFiles(conversationID string) { - s.mu.Lock() - defer s.mu.Unlock() - - log.Printf("🗑️ [FILE-CACHE] Deleting all files for conversation %s", conversationID) - - for key, item := range s.cache.Items() { - if file, ok := item.Object.(*CachedFile); ok { - if file.ConversationID == conversationID { - file.SecureWipe() - s.cache.Delete(key) - } - } - } -} - -// ExtendTTL extends the TTL of a file to match conversation lifetime -func (s *Service) ExtendTTL(fileID string, duration time.Duration) { - s.mu.Lock() - defer s.mu.Unlock() - - if value, found := s.cache.Get(fileID); found { - s.cache.Set(fileID, value, duration) - log.Printf("⏰ [FILE-CACHE] Extended TTL for file %s to %v", fileID, duration) - } -} - -// SecureWipe securely wipes the file's sensitive data -func (f *CachedFile) SecureWipe() { - if f.ExtractedText != nil { - f.ExtractedText.Wipe() - f.ExtractedText = nil - } - - // Delete physical file if it exists (for images) - if f.FilePath != "" { - if err := os.Remove(f.FilePath); err != nil && !os.IsNotExist(err) { - log.Printf("⚠️ Failed to delete file %s: %v", f.FilePath, err) - } else { - log.Printf("🗑️ Deleted file from disk: %s", f.FilePath) - } - } - - // Wipe hash - for i := range f.FileHash { - f.FileHash[i] = 0 - } - - // Clear other fields - f.FileID = "" - f.UserID = "" - f.ConversationID = "" - f.Filename = "" - f.FilePath = "" -} - -// CleanupExpiredFiles deletes files older than 1 hour -func (s *Service) CleanupExpiredFiles() { - s.mu.Lock() - defer s.mu.Unlock() - - now := time.Now() - expiredCount := 0 - - for key, item := range s.cache.Items() { - if file, ok := item.Object.(*CachedFile); ok { - if file.FilePath != "" { - if now.Sub(file.UploadedAt) > 1*time.Hour { - log.Printf("🗑️ [FILE-CACHE] Deleting expired file: %s (uploaded %v ago)", - file.Filename, now.Sub(file.UploadedAt)) - - if err := os.Remove(file.FilePath); err != nil && !os.IsNotExist(err) { - log.Printf("⚠️ Failed to delete expired file %s: %v", file.FilePath, err) - } - - s.cache.Delete(key) - expiredCount++ - } - } - } - } - - if expiredCount > 0 { - log.Printf("✅ [FILE-CACHE] Cleaned up %d expired files", expiredCount) - } -} - -// CleanupOrphanedFiles scans the uploads directory and deletes orphaned files -func (s *Service) CleanupOrphanedFiles(uploadDir string, maxAge time.Duration) { - s.mu.RLock() - trackedFiles := make(map[string]bool) - for _, item := range s.cache.Items() { - if file, ok := item.Object.(*CachedFile); ok { - if file.FilePath != "" { - trackedFiles[file.FilePath] = true - } - } - } - s.mu.RUnlock() - - entries, err := os.ReadDir(uploadDir) - if err != nil { - log.Printf("⚠️ [CLEANUP] Failed to read uploads directory: %v", err) - return - } - - now := time.Now() - orphanedCount := 0 - - for _, entry := range entries { - if entry.IsDir() { - continue - } - - filePath := fmt.Sprintf("%s/%s", uploadDir, entry.Name()) - - info, err := entry.Info() - if err != nil { - continue - } - - fileAge := now.Sub(info.ModTime()) - - if !trackedFiles[filePath] { - if fileAge > 5*time.Minute { - log.Printf("🗑️ [CLEANUP] Deleting orphaned file: %s (age: %v)", entry.Name(), fileAge) - if err := os.Remove(filePath); err != nil && !os.IsNotExist(err) { - log.Printf("⚠️ [CLEANUP] Failed to delete orphaned file %s: %v", entry.Name(), err) - } else { - orphanedCount++ - } - } - } - } - - if orphanedCount > 0 { - log.Printf("✅ [CLEANUP] Deleted %d orphaned files", orphanedCount) - } -} - -// RunStartupCleanup performs initial cleanup when server starts -func (s *Service) RunStartupCleanup(uploadDir string) { - log.Printf("🧹 [STARTUP] Running startup file cleanup in %s...", uploadDir) - - entries, err := os.ReadDir(uploadDir) - if err != nil { - log.Printf("⚠️ [STARTUP] Failed to read uploads directory: %v", err) - return - } - - now := time.Now() - deletedCount := 0 - - for _, entry := range entries { - if entry.IsDir() { - continue - } - - filePath := fmt.Sprintf("%s/%s", uploadDir, entry.Name()) - - info, err := entry.Info() - if err != nil { - continue - } - - if now.Sub(info.ModTime()) > 1*time.Hour { - log.Printf("🗑️ [STARTUP] Deleting stale file: %s (modified: %v ago)", - entry.Name(), now.Sub(info.ModTime())) - if err := os.Remove(filePath); err != nil && !os.IsNotExist(err) { - log.Printf("⚠️ [STARTUP] Failed to delete file %s: %v", entry.Name(), err) - } else { - deletedCount++ - } - } - } - - log.Printf("✅ [STARTUP] Startup cleanup complete: deleted %d stale files", deletedCount) -} - -// GetStats returns cache statistics -func (s *Service) GetStats() map[string]interface{} { - s.mu.RLock() - defer s.mu.RUnlock() - - items := s.cache.Items() - totalSize := int64(0) - totalWords := 0 - - for _, item := range items { - if file, ok := item.Object.(*CachedFile); ok { - totalSize += file.Size - totalWords += file.WordCount - } - } - - return map[string]interface{}{ - "total_files": len(items), - "total_size": totalSize, - "total_words": totalWords, - } -} - -// GetAllFilesByUser returns metadata for all files owned by a user -func (s *Service) GetAllFilesByUser(userID string) []map[string]interface{} { - s.mu.RLock() - defer s.mu.RUnlock() - - var fileMetadata []map[string]interface{} - - for _, item := range s.cache.Items() { - if file, ok := item.Object.(*CachedFile); ok { - if file.UserID == userID { - metadata := map[string]interface{}{ - "file_id": file.FileID, - "filename": file.Filename, - "mime_type": file.MimeType, - "size": file.Size, - "uploaded_at": file.UploadedAt.Format(time.RFC3339), - "conversation_id": file.ConversationID, - } - - if file.MimeType == "application/pdf" { - metadata["page_count"] = file.PageCount - metadata["word_count"] = file.WordCount - } - - fileMetadata = append(fileMetadata, metadata) - } - } - } - - return fileMetadata -} - -// DeleteAllFilesByUser deletes all files owned by a user -func (s *Service) DeleteAllFilesByUser(userID string) (int, error) { - s.mu.Lock() - defer s.mu.Unlock() - - deletedCount := 0 - - for key, item := range s.cache.Items() { - if file, ok := item.Object.(*CachedFile); ok { - if file.UserID == userID { - log.Printf("🗑️ [GDPR] Deleting file %s (%s) for user %s", file.FileID, file.Filename, userID) - file.SecureWipe() - s.cache.Delete(key) - deletedCount++ - } - } - } - - log.Printf("✅ [GDPR] Deleted %d files for user %s", deletedCount, userID) - return deletedCount, nil -} diff --git a/backend/internal/filecache/filecache_test.go b/backend/internal/filecache/filecache_test.go deleted file mode 100644 index ec4d2088..00000000 --- a/backend/internal/filecache/filecache_test.go +++ /dev/null @@ -1,501 +0,0 @@ -package filecache - -import ( - "claraverse/internal/security" - "os" - "path/filepath" - "testing" - "time" -) - -// TestNewService verifies service creation -func TestNewService(t *testing.T) { - svc := NewService() - if svc == nil { - t.Fatal("NewService should return non-nil service") - } - if svc.cache == nil { - t.Error("Service should have cache initialized") - } -} - -// TestGetServiceSingleton verifies singleton pattern -func TestGetServiceSingleton(t *testing.T) { - svc1 := GetService() - svc2 := GetService() - - if svc1 != svc2 { - t.Error("GetService should return the same instance") - } -} - -// TestStoreAndGet tests basic store and retrieve -func TestStoreAndGet(t *testing.T) { - svc := NewService() - - file := &CachedFile{ - FileID: "test-file-123", - UserID: "user-456", - ConversationID: "conv-789", - Filename: "test.pdf", - MimeType: "application/pdf", - Size: 1024, - UploadedAt: time.Now(), - } - - svc.Store(file) - - retrieved, found := svc.Get("test-file-123") - if !found { - t.Fatal("File should be found after store") - } - - if retrieved.FileID != file.FileID { - t.Errorf("Expected FileID %s, got %s", file.FileID, retrieved.FileID) - } - if retrieved.UserID != file.UserID { - t.Errorf("Expected UserID %s, got %s", file.UserID, retrieved.UserID) - } - if retrieved.Filename != file.Filename { - t.Errorf("Expected Filename %s, got %s", file.Filename, retrieved.Filename) - } -} - -// TestGetNotFound tests retrieval of non-existent file -func TestGetNotFound(t *testing.T) { - svc := NewService() - - _, found := svc.Get("non-existent-file") - if found { - t.Error("Non-existent file should not be found") - } -} - -// TestGetByUser tests user-scoped retrieval -func TestGetByUser(t *testing.T) { - svc := NewService() - - file := &CachedFile{ - FileID: "file-for-user", - UserID: "user-123", - Filename: "user-file.pdf", - } - svc.Store(file) - - // Same user should be able to retrieve - retrieved, err := svc.GetByUser("file-for-user", "user-123") - if err != nil { - t.Errorf("Same user should retrieve file: %v", err) - } - if retrieved == nil { - t.Fatal("Retrieved file should not be nil") - } - - // Different user should be denied - _, err = svc.GetByUser("file-for-user", "different-user") - if err == nil { - t.Error("Different user should be denied access") - } -} - -// TestGetByUserAndConversation tests user+conversation scoped retrieval -func TestGetByUserAndConversation(t *testing.T) { - svc := NewService() - - file := &CachedFile{ - FileID: "conv-file", - UserID: "user-123", - ConversationID: "conv-456", - Filename: "conv-file.pdf", - } - svc.Store(file) - - // Same user + conversation should work - retrieved, err := svc.GetByUserAndConversation("conv-file", "user-123", "conv-456") - if err != nil { - t.Errorf("Same user+conversation should retrieve file: %v", err) - } - if retrieved == nil { - t.Fatal("Retrieved file should not be nil") - } - - // Same user, different conversation should fail - _, err = svc.GetByUserAndConversation("conv-file", "user-123", "different-conv") - if err == nil { - t.Error("Different conversation should be denied") - } - - // Different user, same conversation should fail - _, err = svc.GetByUserAndConversation("conv-file", "different-user", "conv-456") - if err == nil { - t.Error("Different user should be denied") - } -} - -// TestDelete tests file deletion -func TestDelete(t *testing.T) { - svc := NewService() - - file := &CachedFile{ - FileID: "delete-me", - UserID: "user-123", - Filename: "to-delete.pdf", - } - svc.Store(file) - - // Verify it exists - _, found := svc.Get("delete-me") - if !found { - t.Fatal("File should exist before deletion") - } - - // Delete it - svc.Delete("delete-me") - - // Verify it's gone - _, found = svc.Get("delete-me") - if found { - t.Error("File should not exist after deletion") - } -} - -// TestDeleteConversationFiles tests conversation-level deletion -func TestDeleteConversationFiles(t *testing.T) { - svc := NewService() - - // Store multiple files for same conversation - for i := 0; i < 3; i++ { - svc.Store(&CachedFile{ - FileID: "conv-file-" + string(rune('a'+i)), - UserID: "user-123", - ConversationID: "conv-to-delete", - Filename: "file.pdf", - }) - } - - // Store file for different conversation - svc.Store(&CachedFile{ - FileID: "other-file", - UserID: "user-123", - ConversationID: "other-conv", - Filename: "other.pdf", - }) - - // Delete conversation files - svc.DeleteConversationFiles("conv-to-delete") - - // Conversation files should be gone - files := svc.GetFilesForConversation("conv-to-delete") - if len(files) != 0 { - t.Errorf("Expected 0 files after deletion, got %d", len(files)) - } - - // Other conversation's file should remain - _, found := svc.Get("other-file") - if !found { - t.Error("File from other conversation should still exist") - } -} - -// TestGetFilesForConversation tests conversation-level retrieval -func TestGetFilesForConversation(t *testing.T) { - svc := NewService() - - targetConv := "target-conv" - - // Store files for target conversation - for i := 0; i < 3; i++ { - svc.Store(&CachedFile{ - FileID: "target-file-" + string(rune('a'+i)), - UserID: "user-123", - ConversationID: targetConv, - Filename: "file.pdf", - }) - } - - // Store files for other conversation - svc.Store(&CachedFile{ - FileID: "other-file", - UserID: "user-123", - ConversationID: "other-conv", - Filename: "other.pdf", - }) - - files := svc.GetFilesForConversation(targetConv) - if len(files) != 3 { - t.Errorf("Expected 3 files for conversation, got %d", len(files)) - } - - for _, file := range files { - if file.ConversationID != targetConv { - t.Errorf("File %s has wrong conversation %s", file.FileID, file.ConversationID) - } - } -} - -// TestGetConversationFiles tests file ID retrieval -func TestGetConversationFiles(t *testing.T) { - svc := NewService() - - conv := "my-conv" - expectedIDs := []string{"file-1", "file-2", "file-3"} - - for _, id := range expectedIDs { - svc.Store(&CachedFile{ - FileID: id, - ConversationID: conv, - }) - } - - fileIDs := svc.GetConversationFiles(conv) - if len(fileIDs) != len(expectedIDs) { - t.Errorf("Expected %d file IDs, got %d", len(expectedIDs), len(fileIDs)) - } -} - -// TestExtendTTL tests TTL extension -func TestExtendTTL(t *testing.T) { - svc := NewService() - - file := &CachedFile{ - FileID: "ttl-file", - Filename: "ttl.pdf", - } - svc.Store(file) - - // Extend TTL (should not error) - svc.ExtendTTL("ttl-file", 1*time.Hour) - - // Verify file still exists - _, found := svc.Get("ttl-file") - if !found { - t.Error("File should still exist after TTL extension") - } - - // Extend non-existent file (should not error) - svc.ExtendTTL("non-existent", 1*time.Hour) -} - -// TestSecureWipe tests secure wiping -func TestSecureWipe(t *testing.T) { - // Create temp file - tmpDir := t.TempDir() - tmpFile := filepath.Join(tmpDir, "wipe-test.txt") - if err := os.WriteFile(tmpFile, []byte("test content"), 0644); err != nil { - t.Fatalf("Failed to create temp file: %v", err) - } - - file := &CachedFile{ - FileID: "wipe-file", - UserID: "user-123", - Filename: "wipe-test.txt", - FilePath: tmpFile, - ExtractedText: security.NewSecureString("sensitive text"), - FileHash: security.Hash{1, 2, 3, 4, 5}, - } - - file.SecureWipe() - - // Verify memory is cleared - if file.FileID != "" { - t.Error("FileID should be cleared") - } - if file.UserID != "" { - t.Error("UserID should be cleared") - } - if file.ExtractedText != nil { - t.Error("ExtractedText should be nil") - } - - // Verify hash is zeroed - for i, b := range file.FileHash { - if b != 0 { - t.Errorf("FileHash[%d] should be 0, got %d", i, b) - } - } - - // Verify physical file is deleted - if _, err := os.Stat(tmpFile); !os.IsNotExist(err) { - t.Error("Physical file should be deleted") - } -} - -// TestGetStats tests statistics retrieval -func TestGetStats(t *testing.T) { - svc := NewService() - - // Store some files - svc.Store(&CachedFile{ - FileID: "stats-file-1", - Size: 1000, - WordCount: 100, - }) - svc.Store(&CachedFile{ - FileID: "stats-file-2", - Size: 2000, - WordCount: 200, - }) - - stats := svc.GetStats() - - if stats["total_files"].(int) != 2 { - t.Errorf("Expected 2 files, got %v", stats["total_files"]) - } - if stats["total_size"].(int64) != 3000 { - t.Errorf("Expected total size 3000, got %v", stats["total_size"]) - } - if stats["total_words"].(int) != 300 { - t.Errorf("Expected 300 words, got %v", stats["total_words"]) - } -} - -// TestGetAllFilesByUser tests user file listing -func TestGetAllFilesByUser(t *testing.T) { - svc := NewService() - - targetUser := "target-user" - - // Store files for target user - svc.Store(&CachedFile{ - FileID: "user-file-1", - UserID: targetUser, - Filename: "file1.pdf", - MimeType: "application/pdf", - Size: 1000, - UploadedAt: time.Now(), - }) - svc.Store(&CachedFile{ - FileID: "user-file-2", - UserID: targetUser, - Filename: "file2.jpg", - MimeType: "image/jpeg", - Size: 2000, - UploadedAt: time.Now(), - }) - - // Store file for different user - svc.Store(&CachedFile{ - FileID: "other-file", - UserID: "other-user", - Filename: "other.pdf", - }) - - files := svc.GetAllFilesByUser(targetUser) - if len(files) != 2 { - t.Errorf("Expected 2 files for user, got %d", len(files)) - } - - for _, metadata := range files { - if metadata["file_id"] == "other-file" { - t.Error("Should not return files from other users") - } - } -} - -// TestDeleteAllFilesByUser tests GDPR-style user data deletion -func TestDeleteAllFilesByUser(t *testing.T) { - svc := NewService() - - targetUser := "delete-user" - - // Store files for target user - for i := 0; i < 5; i++ { - svc.Store(&CachedFile{ - FileID: "delete-file-" + string(rune('a'+i)), - UserID: targetUser, - Filename: "file.pdf", - }) - } - - // Store file for different user - svc.Store(&CachedFile{ - FileID: "keep-file", - UserID: "other-user", - Filename: "keep.pdf", - }) - - deleted, err := svc.DeleteAllFilesByUser(targetUser) - if err != nil { - t.Errorf("DeleteAllFilesByUser should not error: %v", err) - } - if deleted != 5 { - t.Errorf("Expected 5 files deleted, got %d", deleted) - } - - // Verify target user's files are gone - files := svc.GetAllFilesByUser(targetUser) - if len(files) != 0 { - t.Errorf("Expected 0 files for deleted user, got %d", len(files)) - } - - // Verify other user's file remains - _, found := svc.Get("keep-file") - if !found { - t.Error("Other user's file should still exist") - } -} - -// TestCachedFileStructure tests CachedFile struct -func TestCachedFileStructure(t *testing.T) { - file := &CachedFile{ - FileID: "test-id", - UserID: "user-id", - ConversationID: "conv-id", - Filename: "test.pdf", - MimeType: "application/pdf", - Size: 12345, - PageCount: 10, - WordCount: 500, - FilePath: "/tmp/test.pdf", - UploadedAt: time.Now(), - } - - if file.FileID == "" { - t.Error("FileID should be set") - } - if file.MimeType != "application/pdf" { - t.Errorf("MimeType should be application/pdf, got %s", file.MimeType) - } -} - -// Benchmark tests -func BenchmarkStore(b *testing.B) { - svc := NewService() - - b.ResetTimer() - for i := 0; i < b.N; i++ { - svc.Store(&CachedFile{ - FileID: "bench-file", - UserID: "user", - Filename: "bench.pdf", - }) - } -} - -func BenchmarkGet(b *testing.B) { - svc := NewService() - svc.Store(&CachedFile{ - FileID: "bench-get-file", - UserID: "user", - Filename: "bench.pdf", - }) - - b.ResetTimer() - for i := 0; i < b.N; i++ { - svc.Get("bench-get-file") - } -} - -func BenchmarkGetByUser(b *testing.B) { - svc := NewService() - svc.Store(&CachedFile{ - FileID: "bench-user-file", - UserID: "target-user", - Filename: "bench.pdf", - }) - - b.ResetTimer() - for i := 0; i < b.N; i++ { - svc.GetByUser("bench-user-file", "target-user") - } -} diff --git a/backend/internal/handlers/admin.go b/backend/internal/handlers/admin.go deleted file mode 100644 index 25147f37..00000000 --- a/backend/internal/handlers/admin.go +++ /dev/null @@ -1,746 +0,0 @@ -package handlers - -import ( - "claraverse/internal/models" - "claraverse/internal/services" - "fmt" - "log" - - "github.com/gofiber/fiber/v2" -) - -// AdminHandler handles admin operations -type AdminHandler struct { - userService *services.UserService - tierService *services.TierService - analyticsService *services.AnalyticsService - providerService *services.ProviderService - modelService *services.ModelService -} - -// NewAdminHandler creates a new admin handler -func NewAdminHandler(userService *services.UserService, tierService *services.TierService, analyticsService *services.AnalyticsService, providerService *services.ProviderService, modelService *services.ModelService) *AdminHandler { - return &AdminHandler{ - userService: userService, - tierService: tierService, - analyticsService: analyticsService, - providerService: providerService, - modelService: modelService, - } -} - -// GetUserDetails returns detailed user information (admin only) -// GET /api/admin/users/:userID -func (h *AdminHandler) GetUserDetails(c *fiber.Ctx) error { - targetUserID := c.Params("userID") - if targetUserID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "User ID is required", - }) - } - - adminUserID := c.Locals("user_id").(string) - log.Printf("🔍 Admin %s viewing details for user %s", adminUserID, targetUserID) - - userDetails, err := h.userService.GetAdminUserDetails(c.Context(), targetUserID, h.tierService) - if err != nil { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "User not found", - }) - } - - return c.JSON(userDetails) -} - -// SetLimitOverrides sets tier OR granular limit overrides for a user (admin only) -// POST /api/admin/users/:userID/overrides -func (h *AdminHandler) SetLimitOverrides(c *fiber.Ctx) error { - targetUserID := c.Params("userID") - if targetUserID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "User ID is required", - }) - } - - var req models.SetLimitOverridesRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // Validate: must provide either tier OR limits - if req.Tier == nil && req.Limits == nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Must provide either 'tier' or 'limits'", - }) - } - - if req.Tier != nil && req.Limits != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Cannot set both 'tier' and 'limits' at the same time", - }) - } - - // Validate tier if provided - if req.Tier != nil { - validTiers := []string{ - models.TierFree, - models.TierPro, - models.TierMax, - models.TierEnterprise, - models.TierLegacyUnlimited, - } - isValid := false - for _, validTier := range validTiers { - if *req.Tier == validTier { - isValid = true - break - } - } - if !isValid { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid tier", - }) - } - } - - adminUserID := c.Locals("user_id").(string) - - err := h.userService.SetLimitOverrides( - c.Context(), - targetUserID, - adminUserID, - req.Reason, - req.Tier, - req.Limits, - ) - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": err.Error(), - }) - } - - // Invalidate cache - h.tierService.InvalidateCache(targetUserID) - - var message string - if req.Tier != nil { - message = fmt.Sprintf("Tier override set to %s", *req.Tier) - } else { - message = "Granular limit overrides set successfully" - } - - return c.JSON(fiber.Map{ - "success": true, - "message": message, - }) -} - -// RemoveAllOverrides removes all overrides (tier and limits) for a user (admin only) -// DELETE /api/admin/users/:userID/overrides -func (h *AdminHandler) RemoveAllOverrides(c *fiber.Ctx) error { - targetUserID := c.Params("userID") - if targetUserID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "User ID is required", - }) - } - - adminUserID := c.Locals("user_id").(string) - - err := h.userService.RemoveAllOverrides(c.Context(), targetUserID, adminUserID) - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": err.Error(), - }) - } - - // Invalidate cache - h.tierService.InvalidateCache(targetUserID) - - return c.JSON(fiber.Map{ - "success": true, - "message": "All overrides removed successfully", - }) -} - -// ListUsers returns a GDPR-compliant paginated list of users (admin only) -// GET /api/admin/users -func (h *AdminHandler) ListUsers(c *fiber.Ctx) error { - if h.analyticsService == nil { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "Analytics service not available", - }) - } - - // Parse query parameters - page := c.QueryInt("page", 1) - pageSize := c.QueryInt("page_size", 50) - tier := c.Query("tier", "") - search := c.Query("search", "") - - // Get aggregated user analytics (GDPR-compliant - no PII) - users, totalCount, err := h.analyticsService.GetUserListGDPR(c.Context(), page, pageSize, tier, search) - if err != nil { - log.Printf("❌ [ADMIN] Failed to get user list: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to fetch user list", - }) - } - - return c.JSON(fiber.Map{ - "users": users, - "total_count": totalCount, - "page": page, - "page_size": pageSize, - "gdpr_notice": "This data is aggregated and anonymized. Full email addresses are hashed for privacy. Only domains are shown for trend analysis.", - }) -} - -// GetGDPRPolicy returns the GDPR data policy -// GET /api/admin/gdpr-policy -func (h *AdminHandler) GetGDPRPolicy(c *fiber.Ctx) error { - return c.JSON(fiber.Map{ - "data_collected": []string{ - "User ID (anonymized)", - "Email domain (for trend analysis only)", - "Subscription tier", - "Usage counts (chats, messages, agent runs)", - "Activity timestamps", - }, - "data_retention_days": 90, - "purpose": "Product analytics and performance monitoring", - "legal_basis": "Legitimate interest (GDPR Art. 6(1)(f))", - "user_rights": []string{ - "Right to access (Art. 15)", - "Right to rectification (Art. 16)", - "Right to erasure (Art. 17)", - "Right to data portability (Art. 20)", - "Right to object (Art. 21)", - }, - }) -} - -// GetSystemStats returns system statistics (admin only) -// GET /api/admin/stats -func (h *AdminHandler) GetSystemStats(c *fiber.Ctx) error { - // TODO: Implement stats like user count by tier, active subscriptions, etc. - return c.JSON(fiber.Map{ - "message": "System stats endpoint - to be implemented", - }) -} - -// GetAdminStatus returns admin status for the authenticated user -// GET /api/admin/me -func (h *AdminHandler) GetAdminStatus(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - email := c.Locals("email") - if email == nil { - email = "" - } - - return c.JSON(fiber.Map{ - "is_admin": true, // If this endpoint is reached, user is admin (middleware validated) - "user_id": userID, - "email": email, - }) -} - -// GetOverviewAnalytics returns overview analytics -// GET /api/admin/analytics/overview -func (h *AdminHandler) GetOverviewAnalytics(c *fiber.Ctx) error { - if h.analyticsService == nil { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "Analytics service not available", - }) - } - - stats, err := h.analyticsService.GetOverviewStats(c.Context()) - if err != nil { - log.Printf("❌ [ADMIN] Failed to get overview analytics: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to fetch overview analytics", - }) - } - - return c.JSON(stats) -} - -// GetProviderAnalytics returns provider usage analytics -// GET /api/admin/analytics/providers -func (h *AdminHandler) GetProviderAnalytics(c *fiber.Ctx) error { - if h.analyticsService == nil { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "Analytics service not available", - }) - } - - analytics, err := h.analyticsService.GetProviderAnalytics(c.Context()) - if err != nil { - log.Printf("❌ [ADMIN] Failed to get provider analytics: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to fetch provider analytics", - }) - } - - return c.JSON(analytics) -} - -// GetChatAnalytics returns chat usage analytics -// GET /api/admin/analytics/chats -func (h *AdminHandler) GetChatAnalytics(c *fiber.Ctx) error { - if h.analyticsService == nil { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "Analytics service not available", - }) - } - - analytics, err := h.analyticsService.GetChatAnalytics(c.Context()) - if err != nil { - log.Printf("❌ [ADMIN] Failed to get chat analytics: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to fetch chat analytics", - }) - } - - return c.JSON(analytics) -} - -// GetModelAnalytics returns model usage analytics (placeholder) -// GET /api/admin/analytics/models -func (h *AdminHandler) GetModelAnalytics(c *fiber.Ctx) error { - // TODO: Implement model analytics - return c.JSON([]fiber.Map{}) -} - -// GetAgentAnalytics returns comprehensive agent activity analytics -// GET /api/admin/analytics/agents -func (h *AdminHandler) GetAgentAnalytics(c *fiber.Ctx) error { - if h.analyticsService == nil { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "Analytics service not available", - }) - } - - analytics, err := h.analyticsService.GetAgentAnalytics(c.Context()) - if err != nil { - log.Printf("❌ [ADMIN] Failed to get agent analytics: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to fetch agent analytics", - }) - } - - return c.JSON(analytics) -} - -// MigrateChatSessionTimestamps fixes existing chat sessions without proper timestamps -// POST /api/admin/analytics/migrate-timestamps -func (h *AdminHandler) MigrateChatSessionTimestamps(c *fiber.Ctx) error { - if h.analyticsService == nil { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "Analytics service not available", - }) - } - - count, err := h.analyticsService.MigrateChatSessionTimestamps(c.Context()) - if err != nil { - log.Printf("❌ [ADMIN] Failed to migrate chat session timestamps: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to migrate chat session timestamps", - "details": err.Error(), - }) - } - - return c.JSON(fiber.Map{ - "success": true, - "message": fmt.Sprintf("Successfully migrated %d chat sessions", count), - "sessions_updated": count, - }) -} - -// GetProviders returns all providers from database -// GET /api/admin/providers -func (h *AdminHandler) GetProviders(c *fiber.Ctx) error { - providers, err := h.providerService.GetAllIncludingDisabled() - if err != nil { - log.Printf("❌ [ADMIN] Failed to get providers: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get providers", - }) - } - - // Get model counts, aliases, and filters for each provider - var providerViews []fiber.Map - for _, provider := range providers { - // Get model count - models, err := h.modelService.GetByProvider(provider.ID, false) - modelCount := 0 - if err == nil { - modelCount = len(models) - } - - // Get aliases - aliases, err := h.modelService.LoadAllAliasesFromDB() - providerAliases := make(map[string]interface{}) - if err == nil && aliases[provider.ID] != nil { - providerAliases = convertAliasesMapToInterface(aliases[provider.ID]) - } - - // Get filters - filters, _ := h.providerService.GetFilters(provider.ID) - - // Get recommended models - recommended, _ := h.modelService.LoadAllRecommendedModelsFromDB() - var recommendedModels interface{} - if recommended[provider.ID] != nil { - recommendedModels = recommended[provider.ID] - } - - providerView := fiber.Map{ - "id": provider.ID, - "name": provider.Name, - "base_url": provider.BaseURL, - "enabled": provider.Enabled, - "audio_only": provider.AudioOnly, - "favicon": provider.Favicon, - "model_count": modelCount, - "model_aliases": providerAliases, - "filters": filters, - "recommended_models": recommendedModels, - } - - providerViews = append(providerViews, providerView) - } - - return c.JSON(fiber.Map{ - "providers": providerViews, - }) -} - -// CreateProvider creates a new provider -// POST /api/admin/providers -func (h *AdminHandler) CreateProvider(c *fiber.Ctx) error { - var req struct { - Name string `json:"name"` - BaseURL string `json:"base_url"` - APIKey string `json:"api_key"` - Enabled *bool `json:"enabled"` - AudioOnly *bool `json:"audio_only"` - ImageOnly *bool `json:"image_only"` - ImageEditOnly *bool `json:"image_edit_only"` - Secure *bool `json:"secure"` - DefaultModel string `json:"default_model"` - SystemPrompt string `json:"system_prompt"` - Favicon string `json:"favicon"` - } - - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // Validate required fields - if req.Name == "" || req.BaseURL == "" || req.APIKey == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Name, base_url, and api_key are required", - }) - } - - // Build provider config - config := models.ProviderConfig{ - Name: req.Name, - BaseURL: req.BaseURL, - APIKey: req.APIKey, - Enabled: req.Enabled != nil && *req.Enabled, - AudioOnly: req.AudioOnly != nil && *req.AudioOnly, - ImageOnly: req.ImageOnly != nil && *req.ImageOnly, - ImageEditOnly: req.ImageEditOnly != nil && *req.ImageEditOnly, - Secure: req.Secure != nil && *req.Secure, - DefaultModel: req.DefaultModel, - SystemPrompt: req.SystemPrompt, - Favicon: req.Favicon, - } - - provider, err := h.providerService.Create(config) - if err != nil { - log.Printf("❌ [ADMIN] Failed to create provider: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": fmt.Sprintf("Failed to create provider: %v", err), - }) - } - - log.Printf("✅ [ADMIN] Created provider: %s (ID %d)", provider.Name, provider.ID) - - // Reload image providers if this is an image provider - if config.ImageOnly || config.ImageEditOnly { - h.reloadImageProviders() - } - - return c.Status(fiber.StatusCreated).JSON(provider) -} - -// UpdateProvider updates an existing provider -// PUT /api/admin/providers/:id -func (h *AdminHandler) UpdateProvider(c *fiber.Ctx) error { - providerID, err := c.ParamsInt("id") - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid provider ID", - }) - } - - var req struct { - Name *string `json:"name"` - BaseURL *string `json:"base_url"` - APIKey *string `json:"api_key"` - Enabled *bool `json:"enabled"` - AudioOnly *bool `json:"audio_only"` - ImageOnly *bool `json:"image_only"` - ImageEditOnly *bool `json:"image_edit_only"` - Secure *bool `json:"secure"` - DefaultModel *string `json:"default_model"` - SystemPrompt *string `json:"system_prompt"` - Favicon *string `json:"favicon"` - } - - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // Get existing provider - existing, err := h.providerService.GetByID(providerID) - if err != nil { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Provider not found", - }) - } - - // Build update config with existing values as defaults - config := models.ProviderConfig{ - Name: existing.Name, - BaseURL: existing.BaseURL, - APIKey: existing.APIKey, - Enabled: existing.Enabled, - AudioOnly: existing.AudioOnly, - SystemPrompt: existing.SystemPrompt, - Favicon: existing.Favicon, - } - - // Apply updates - if req.Name != nil { - config.Name = *req.Name - } - if req.BaseURL != nil { - config.BaseURL = *req.BaseURL - } - if req.APIKey != nil { - config.APIKey = *req.APIKey - } - if req.Enabled != nil { - config.Enabled = *req.Enabled - } - if req.AudioOnly != nil { - config.AudioOnly = *req.AudioOnly - } - if req.ImageOnly != nil { - config.ImageOnly = *req.ImageOnly - } - if req.ImageEditOnly != nil { - config.ImageEditOnly = *req.ImageEditOnly - } - if req.Secure != nil { - config.Secure = *req.Secure - } - if req.DefaultModel != nil { - config.DefaultModel = *req.DefaultModel - } - if req.SystemPrompt != nil { - config.SystemPrompt = *req.SystemPrompt - } - if req.Favicon != nil { - config.Favicon = *req.Favicon - } - - if err := h.providerService.Update(providerID, config); err != nil { - log.Printf("❌ [ADMIN] Failed to update provider %d: %v", providerID, err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": fmt.Sprintf("Failed to update provider: %v", err), - }) - } - - // Get updated provider - updated, err := h.providerService.GetByID(providerID) - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to retrieve updated provider", - }) - } - - log.Printf("✅ [ADMIN] Updated provider: %s (ID %d)", updated.Name, updated.ID) - - // Reload image providers if image flags changed - if (req.ImageOnly != nil && *req.ImageOnly) || (req.ImageEditOnly != nil && *req.ImageEditOnly) || - updated.ImageOnly || updated.ImageEditOnly { - h.reloadImageProviders() - } - - return c.JSON(updated) -} - -// DeleteProvider deletes a provider -// DELETE /api/admin/providers/:id -func (h *AdminHandler) DeleteProvider(c *fiber.Ctx) error { - providerID, err := c.ParamsInt("id") - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid provider ID", - }) - } - - // Get provider before deleting for logging - provider, err := h.providerService.GetByID(providerID) - if err != nil { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Provider not found", - }) - } - - if err := h.providerService.Delete(providerID); err != nil { - log.Printf("❌ [ADMIN] Failed to delete provider %d: %v", providerID, err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": fmt.Sprintf("Failed to delete provider: %v", err), - }) - } - - log.Printf("✅ [ADMIN] Deleted provider: %s (ID %d)", provider.Name, provider.ID) - return c.JSON(fiber.Map{ - "message": "Provider deleted successfully", - }) -} - -// ToggleProvider toggles a provider's enabled state -// PUT /api/admin/providers/:id/toggle -func (h *AdminHandler) ToggleProvider(c *fiber.Ctx) error { - providerID, err := c.ParamsInt("id") - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid provider ID", - }) - } - - var req struct { - Enabled bool `json:"enabled"` - } - - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // Get existing provider - provider, err := h.providerService.GetByID(providerID) - if err != nil { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Provider not found", - }) - } - - // Update enabled state - config := models.ProviderConfig{ - Name: provider.Name, - BaseURL: provider.BaseURL, - APIKey: provider.APIKey, - Enabled: req.Enabled, - AudioOnly: provider.AudioOnly, - SystemPrompt: provider.SystemPrompt, - Favicon: provider.Favicon, - } - - if err := h.providerService.Update(providerID, config); err != nil { - log.Printf("❌ [ADMIN] Failed to toggle provider %d: %v", providerID, err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": fmt.Sprintf("Failed to toggle provider: %v", err), - }) - } - - // Get updated provider - updated, err := h.providerService.GetByID(providerID) - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to retrieve updated provider", - }) - } - - log.Printf("✅ [ADMIN] Toggled provider %s to enabled=%v", updated.Name, updated.Enabled) - return c.JSON(updated) -} - -// Helper function to convert ModelAlias map to interface{} map for JSON -func convertAliasesMapToInterface(aliases map[string]models.ModelAlias) map[string]interface{} { - result := make(map[string]interface{}) - for key, alias := range aliases { - result[key] = fiber.Map{ - "actual_model": alias.ActualModel, - "display_name": alias.DisplayName, - "description": alias.Description, - "supports_vision": alias.SupportsVision, - "agents": alias.Agents, - "smart_tool_router": alias.SmartToolRouter, - "free_tier": alias.FreeTier, - "structured_output_support": alias.StructuredOutputSupport, - "structured_output_compliance": alias.StructuredOutputCompliance, - "structured_output_warning": alias.StructuredOutputWarning, - "structured_output_speed_ms": alias.StructuredOutputSpeedMs, - "structured_output_badge": alias.StructuredOutputBadge, - "memory_extractor": alias.MemoryExtractor, - "memory_selector": alias.MemorySelector, - } - } - return result -} - -// reloadImageProviders reloads image providers from the database -// Called after creating/updating providers with image_only or image_edit_only flags -func (h *AdminHandler) reloadImageProviders() { - log.Println("🔄 [ADMIN] Reloading image providers...") - - // Get all providers from database - allProviders, err := h.providerService.GetAll() - if err != nil { - log.Printf("⚠️ [ADMIN] Failed to reload providers: %v", err) - return - } - - // Convert to ProviderConfig format - var providerConfigs []models.ProviderConfig - for _, p := range allProviders { - providerConfigs = append(providerConfigs, models.ProviderConfig{ - Name: p.Name, - BaseURL: p.BaseURL, - APIKey: p.APIKey, - Enabled: p.Enabled, - Secure: p.Secure, - AudioOnly: p.AudioOnly, - ImageOnly: p.ImageOnly, - ImageEditOnly: p.ImageEditOnly, - DefaultModel: p.DefaultModel, - SystemPrompt: p.SystemPrompt, - Favicon: p.Favicon, - }) - } - - // Reload image providers - imageProviderService := services.GetImageProviderService() - imageProviderService.LoadFromProviders(providerConfigs) - - // Reload image edit providers - imageEditProviderService := services.GetImageEditProviderService() - imageEditProviderService.LoadFromProviders(providerConfigs) - - log.Println("✅ [ADMIN] Image providers reloaded") -} diff --git a/backend/internal/handlers/agent.go b/backend/internal/handlers/agent.go deleted file mode 100644 index 8cde2262..00000000 --- a/backend/internal/handlers/agent.go +++ /dev/null @@ -1,1243 +0,0 @@ -package handlers - -import ( - "bytes" - "claraverse/internal/models" - "claraverse/internal/services" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "strings" - "time" - - "github.com/gofiber/fiber/v2" -) - -// isPlaceholderDescription checks if a description is empty or a placeholder -func isPlaceholderDescription(desc string) bool { - if desc == "" { - return true - } - // Normalize for comparison - lower := strings.ToLower(strings.TrimSpace(desc)) - // Common placeholder patterns - placeholders := []string{ - "describe what this agent does", - "description", - "add a description", - "enter description", - "agent description", - "no description", - "...", - "-", - } - for _, p := range placeholders { - if lower == p || strings.HasPrefix(lower, p) { - return true - } - } - return false -} - -// AgentHandler handles agent-related HTTP requests -type AgentHandler struct { - agentService *services.AgentService - workflowGeneratorService *services.WorkflowGeneratorService - workflowGeneratorV2Service *services.WorkflowGeneratorV2Service - builderConvService *services.BuilderConversationService - providerService *services.ProviderService -} - -// NewAgentHandler creates a new agent handler -func NewAgentHandler(agentService *services.AgentService, workflowGenerator *services.WorkflowGeneratorService) *AgentHandler { - return &AgentHandler{ - agentService: agentService, - workflowGeneratorService: workflowGenerator, - } -} - -// SetWorkflowGeneratorV2Service sets the v2 workflow generator service -func (h *AgentHandler) SetWorkflowGeneratorV2Service(svc *services.WorkflowGeneratorV2Service) { - h.workflowGeneratorV2Service = svc -} - -// SetBuilderConversationService sets the builder conversation service (for sync endpoint) -func (h *AgentHandler) SetBuilderConversationService(svc *services.BuilderConversationService) { - h.builderConvService = svc -} - -// SetProviderService sets the provider service (for Ask mode) -func (h *AgentHandler) SetProviderService(svc *services.ProviderService) { - h.providerService = svc -} - -// Create creates a new agent -// POST /api/agents -func (h *AgentHandler) Create(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - var req models.CreateAgentRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if req.Name == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Name is required", - }) - } - - log.Printf("🤖 [AGENT] Creating agent '%s' for user %s", req.Name, userID) - - agent, err := h.agentService.CreateAgent(userID, req.Name, req.Description) - if err != nil { - log.Printf("❌ [AGENT] Failed to create agent: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to create agent", - }) - } - - log.Printf("✅ [AGENT] Created agent %s", agent.ID) - return c.Status(fiber.StatusCreated).JSON(agent) -} - -// List returns all agents for the authenticated user with pagination -// GET /api/agents?limit=20&offset=0 -func (h *AgentHandler) List(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - limit := c.QueryInt("limit", 20) - offset := c.QueryInt("offset", 0) - - log.Printf("📋 [AGENT] Listing agents for user %s (limit: %d, offset: %d)", userID, limit, offset) - - response, err := h.agentService.ListAgentsPaginated(userID, limit, offset) - if err != nil { - log.Printf("❌ [AGENT] Failed to list agents: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to list agents", - }) - } - - // Ensure agents array is not null - if response.Agents == nil { - response.Agents = []models.AgentListItem{} - } - - return c.JSON(response) -} - -// ListRecent returns the 10 most recent agents for the landing page -// GET /api/agents/recent -func (h *AgentHandler) ListRecent(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - log.Printf("📋 [AGENT] Getting recent agents for user %s", userID) - - response, err := h.agentService.GetRecentAgents(userID) - if err != nil { - log.Printf("❌ [AGENT] Failed to get recent agents: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get recent agents", - }) - } - - // Ensure agents array is not null - if response.Agents == nil { - response.Agents = []models.AgentListItem{} - } - - return c.JSON(response) -} - -// Get returns a single agent by ID -// GET /api/agents/:id -func (h *AgentHandler) Get(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - log.Printf("🔍 [AGENT] Getting agent %s for user %s", agentID, userID) - - agent, err := h.agentService.GetAgent(agentID, userID) - if err != nil { - if err.Error() == "agent not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Agent not found", - }) - } - log.Printf("❌ [AGENT] Failed to get agent: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get agent", - }) - } - - return c.JSON(agent) -} - -// Update updates an agent's metadata -// PUT /api/agents/:id -func (h *AgentHandler) Update(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - var req models.UpdateAgentRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - log.Printf("✏️ [AGENT] Updating agent %s for user %s", agentID, userID) - - // Check if we're deploying and need to auto-generate a description - if req.Status == "deployed" { - // Get current agent to check if description is empty or placeholder - currentAgent, err := h.agentService.GetAgent(agentID, userID) - if err == nil && currentAgent != nil { - // Auto-generate description if empty or a placeholder - if isPlaceholderDescription(currentAgent.Description) { - log.Printf("🔍 [AGENT] Agent %s has no/placeholder description, generating one on deploy", agentID) - workflow, err := h.agentService.GetWorkflow(agentID) - if err == nil && workflow != nil { - description, err := h.workflowGeneratorService.GenerateDescriptionFromWorkflow(workflow, currentAgent.Name) - if err != nil { - log.Printf("⚠️ [AGENT] Failed to generate description (non-fatal): %v", err) - } else if description != "" { - req.Description = description - log.Printf("📝 [AGENT] Auto-generated description for agent %s: %s", agentID, description) - } - } - } - } - } - - agent, err := h.agentService.UpdateAgent(agentID, userID, &req) - if err != nil { - if err.Error() == "agent not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Agent not found", - }) - } - log.Printf("❌ [AGENT] Failed to update agent: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to update agent", - }) - } - - log.Printf("✅ [AGENT] Updated agent %s", agentID) - return c.JSON(agent) -} - -// Delete deletes an agent -// DELETE /api/agents/:id -func (h *AgentHandler) Delete(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - log.Printf("🗑️ [AGENT] Deleting agent %s for user %s", agentID, userID) - - err := h.agentService.DeleteAgent(agentID, userID) - if err != nil { - if err.Error() == "agent not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Agent not found", - }) - } - log.Printf("❌ [AGENT] Failed to delete agent: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to delete agent", - }) - } - - log.Printf("✅ [AGENT] Deleted agent %s", agentID) - return c.Status(fiber.StatusNoContent).Send(nil) -} - -// SaveWorkflow saves or updates the workflow for an agent -// PUT /api/agents/:id/workflow -func (h *AgentHandler) SaveWorkflow(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - var req models.SaveWorkflowRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - log.Printf("💾 [AGENT] Saving workflow for agent %s (user: %s, blocks: %d)", agentID, userID, len(req.Blocks)) - - workflow, err := h.agentService.SaveWorkflow(agentID, userID, &req) - if err != nil { - if err.Error() == "agent not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Agent not found", - }) - } - log.Printf("❌ [AGENT] Failed to save workflow: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to save workflow", - }) - } - - log.Printf("✅ [AGENT] Saved workflow for agent %s (version: %d)", agentID, workflow.Version) - return c.JSON(workflow) -} - -// GetWorkflow returns the workflow for an agent -// GET /api/agents/:id/workflow -func (h *AgentHandler) GetWorkflow(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - // Verify agent belongs to user - _, err := h.agentService.GetAgent(agentID, userID) - if err != nil { - if err.Error() == "agent not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Agent not found", - }) - } - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to verify agent ownership", - }) - } - - workflow, err := h.agentService.GetWorkflow(agentID) - if err != nil { - if err.Error() == "workflow not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Workflow not found", - }) - } - log.Printf("❌ [AGENT] Failed to get workflow: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get workflow", - }) - } - - return c.JSON(workflow) -} - -// GenerateWorkflow generates or modifies a workflow using AI -// POST /api/agents/:id/generate-workflow -func (h *AgentHandler) GenerateWorkflow(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - // Parse request body - var req models.WorkflowGenerateRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if req.UserMessage == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "User message is required", - }) - } - - req.AgentID = agentID - - // Get or create the agent - auto-create if it doesn't exist yet - // This supports the frontend workflow where agent IDs are generated client-side - agent, err := h.agentService.GetAgent(agentID, userID) - if err != nil { - if err.Error() == "agent not found" { - // Auto-create the agent with a default name (user can rename later) - log.Printf("🆕 [WORKFLOW-GEN] Agent %s doesn't exist, creating it", agentID) - agent, err = h.agentService.CreateAgentWithID(agentID, userID, "New Agent", "") - if err != nil { - log.Printf("❌ [WORKFLOW-GEN] Failed to auto-create agent: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to create agent", - }) - } - log.Printf("✅ [WORKFLOW-GEN] Auto-created agent %s", agentID) - } else { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to verify agent ownership", - }) - } - } - _ = agent // Agent verified or created - - log.Printf("🔧 [WORKFLOW-GEN] Generating workflow for agent %s (user: %s)", agentID, userID) - - // Generate the workflow - response, err := h.workflowGeneratorService.GenerateWorkflow(&req, userID) - if err != nil { - log.Printf("❌ [WORKFLOW-GEN] Failed to generate workflow: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to generate workflow", - }) - } - - if !response.Success { - log.Printf("⚠️ [WORKFLOW-GEN] Workflow generation failed: %s", response.Error) - return c.Status(fiber.StatusUnprocessableEntity).JSON(response) - } - - // Generate suggested name and description for new workflows - // Check if agent still has default name - if so, generate a better one - shouldGenerateMetadata := response.Action == "create" || (agent != nil && agent.Name == "New Agent") - log.Printf("🔍 [WORKFLOW-GEN] Checking metadata generation: action=%s, agentName=%s, shouldGenerate=%v", response.Action, agent.Name, shouldGenerateMetadata) - if shouldGenerateMetadata { - metadata, err := h.workflowGeneratorService.GenerateAgentMetadata(req.UserMessage) - if err != nil { - log.Printf("⚠️ [WORKFLOW-GEN] Failed to generate agent metadata (non-fatal): %v", err) - } else { - response.SuggestedName = metadata.Name - response.SuggestedDescription = metadata.Description - log.Printf("📝 [WORKFLOW-GEN] Suggested agent: name=%s, desc=%s", metadata.Name, metadata.Description) - - // Immediately persist the generated name to the database - // This ensures the name is saved even if frontend fails to update - if metadata.Name != "" { - updateReq := &models.UpdateAgentRequest{ - Name: metadata.Name, - Description: metadata.Description, - } - _, updateErr := h.agentService.UpdateAgent(agentID, userID, updateReq) - if updateErr != nil { - log.Printf("⚠️ [WORKFLOW-GEN] Failed to persist agent metadata (non-fatal): %v", updateErr) - } else { - log.Printf("💾 [WORKFLOW-GEN] Persisted agent metadata to database: name=%s", metadata.Name) - } - } - } - } - - log.Printf("✅ [WORKFLOW-GEN] Generated workflow for agent %s: %d blocks", agentID, len(response.Workflow.Blocks)) - return c.JSON(response) -} - -// ============================================================================ -// Workflow Version Handlers -// ============================================================================ - -// ListWorkflowVersions returns all versions for an agent's workflow -// GET /api/agents/:id/workflow/versions -func (h *AgentHandler) ListWorkflowVersions(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - log.Printf("📜 [WORKFLOW] Listing versions for agent %s (user: %s)", agentID, userID) - - versions, err := h.agentService.ListWorkflowVersions(agentID, userID) - if err != nil { - if err.Error() == "agent not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Agent not found", - }) - } - log.Printf("❌ [WORKFLOW] Failed to list versions: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to list workflow versions", - }) - } - - return c.JSON(fiber.Map{ - "versions": versions, - "count": len(versions), - }) -} - -// GetWorkflowVersion returns a specific workflow version -// GET /api/agents/:id/workflow/versions/:version -func (h *AgentHandler) GetWorkflowVersion(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - version, err := c.ParamsInt("version") - if err != nil || version <= 0 { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Valid version number is required", - }) - } - - log.Printf("🔍 [WORKFLOW] Getting version %d for agent %s (user: %s)", version, agentID, userID) - - workflow, err := h.agentService.GetWorkflowVersion(agentID, userID, version) - if err != nil { - if err.Error() == "agent not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Agent not found", - }) - } - if err.Error() == "workflow version not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Workflow version not found", - }) - } - log.Printf("❌ [WORKFLOW] Failed to get version: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get workflow version", - }) - } - - return c.JSON(workflow) -} - -// RestoreWorkflowVersion restores a workflow to a previous version -// POST /api/agents/:id/workflow/restore/:version -func (h *AgentHandler) RestoreWorkflowVersion(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - version, err := c.ParamsInt("version") - if err != nil || version <= 0 { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Valid version number is required", - }) - } - - log.Printf("⏪ [WORKFLOW] Restoring version %d for agent %s (user: %s)", version, agentID, userID) - - workflow, err := h.agentService.RestoreWorkflowVersion(agentID, userID, version) - if err != nil { - if err.Error() == "agent not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Agent not found", - }) - } - if err.Error() == "workflow version not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Workflow version not found", - }) - } - log.Printf("❌ [WORKFLOW] Failed to restore version: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to restore workflow version", - }) - } - - log.Printf("✅ [WORKFLOW] Restored version %d for agent %s (new version: %d)", version, agentID, workflow.Version) - return c.JSON(workflow) -} - -// SyncAgent syncs a local agent to the backend on first message -// This creates/updates the agent, workflow, and conversation in one call -// POST /api/agents/:id/sync -func (h *AgentHandler) SyncAgent(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - var req models.SyncAgentRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if req.Name == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent name is required", - }) - } - - log.Printf("🔄 [AGENT] Syncing agent %s for user %s", agentID, userID) - - // Sync agent and workflow - agent, workflow, err := h.agentService.SyncAgent(agentID, userID, &req) - if err != nil { - log.Printf("❌ [AGENT] Failed to sync agent: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to sync agent", - }) - } - - // Create conversation if builder conversation service is available - var conversationID string - if h.builderConvService != nil { - conv, err := h.builderConvService.CreateConversation(c.Context(), agentID, userID, req.ModelID) - if err != nil { - log.Printf("⚠️ [AGENT] Failed to create conversation (non-fatal): %v", err) - // Continue without conversation - not fatal - } else { - conversationID = conv.ID - log.Printf("✅ [AGENT] Created conversation %s for agent %s", conversationID, agentID) - } - } - - log.Printf("✅ [AGENT] Synced agent %s (workflow v%d, conv: %s)", agentID, workflow.Version, conversationID) - - return c.JSON(&models.SyncAgentResponse{ - Agent: agent, - Workflow: workflow, - ConversationID: conversationID, - }) -} - -// GenerateWorkflowV2 generates a workflow using multi-step process with tool selection -// POST /api/agents/:id/generate-workflow-v2 -func (h *AgentHandler) GenerateWorkflowV2(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - if h.workflowGeneratorV2Service == nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Workflow generator v2 service not available", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - // Parse request body - var req services.MultiStepGenerateRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if req.UserMessage == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "User message is required", - }) - } - - req.AgentID = agentID - - // Get or create the agent - auto-create if it doesn't exist yet - agent, err := h.agentService.GetAgent(agentID, userID) - if err != nil { - if err.Error() == "agent not found" { - log.Printf("🆕 [WORKFLOW-GEN-V2] Agent %s doesn't exist, creating it", agentID) - agent, err = h.agentService.CreateAgentWithID(agentID, userID, "New Agent", "") - if err != nil { - log.Printf("❌ [WORKFLOW-GEN-V2] Failed to auto-create agent: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to create agent", - }) - } - log.Printf("✅ [WORKFLOW-GEN-V2] Auto-created agent %s", agentID) - } else { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to verify agent ownership", - }) - } - } - - log.Printf("🔧 [WORKFLOW-GEN-V2] Starting multi-step generation for agent %s (user: %s)", agentID, userID) - - // Generate the workflow using multi-step process - response, err := h.workflowGeneratorV2Service.GenerateWorkflowMultiStep(&req, userID, nil) - if err != nil { - log.Printf("❌ [WORKFLOW-GEN-V2] Failed to generate workflow: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to generate workflow", - }) - } - - if !response.Success { - log.Printf("⚠️ [WORKFLOW-GEN-V2] Workflow generation failed: %s", response.Error) - return c.Status(fiber.StatusUnprocessableEntity).JSON(response) - } - - // Generate suggested name and description for new workflows - shouldGenerateMetadata := agent != nil && agent.Name == "New Agent" - if shouldGenerateMetadata && h.workflowGeneratorService != nil { - metadata, err := h.workflowGeneratorService.GenerateAgentMetadata(req.UserMessage) - if err != nil { - log.Printf("⚠️ [WORKFLOW-GEN-V2] Failed to generate agent metadata (non-fatal): %v", err) - } else if metadata.Name != "" { - // Persist the generated name - updateReq := &models.UpdateAgentRequest{ - Name: metadata.Name, - Description: metadata.Description, - } - _, updateErr := h.agentService.UpdateAgent(agentID, userID, updateReq) - if updateErr != nil { - log.Printf("⚠️ [WORKFLOW-GEN-V2] Failed to persist agent metadata (non-fatal): %v", updateErr) - } else { - log.Printf("💾 [WORKFLOW-GEN-V2] Persisted agent metadata: name=%s", metadata.Name) - } - } - } - - log.Printf("✅ [WORKFLOW-GEN-V2] Generated workflow for agent %s: %d blocks, %d tools selected", - agentID, len(response.Workflow.Blocks), len(response.SelectedTools)) - return c.JSON(response) -} - -// GetToolRegistry returns all available tools and categories for the frontend -// GET /api/tools/registry -func (h *AgentHandler) GetToolRegistry(c *fiber.Ctx) error { - return c.JSON(fiber.Map{ - "tools": services.ToolRegistry, - "categories": services.ToolCategoryRegistry, - }) -} - -// SelectTools performs just the tool selection step (Step 1 only) -// POST /api/agents/:id/select-tools -func (h *AgentHandler) SelectTools(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - if h.workflowGeneratorV2Service == nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Workflow generator v2 service not available", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - // Parse request body - var req services.MultiStepGenerateRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if req.UserMessage == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "User message is required", - }) - } - - req.AgentID = agentID - - log.Printf("🔧 [TOOL-SELECT] Selecting tools for agent %s (user: %s)", agentID, userID) - - // Perform tool selection only - result, err := h.workflowGeneratorV2Service.Step1SelectTools(&req, userID) - if err != nil { - log.Printf("❌ [TOOL-SELECT] Failed to select tools: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to select tools", - }) - } - - log.Printf("✅ [TOOL-SELECT] Selected %d tools for agent %s", len(result.SelectedTools), agentID) - return c.JSON(result) -} - -// GenerateWithToolsRequest is the request for generating a workflow with pre-selected tools -type GenerateWithToolsRequest struct { - UserMessage string `json:"user_message"` - ModelID string `json:"model_id,omitempty"` - SelectedTools []services.SelectedTool `json:"selected_tools"` - CurrentWorkflow *models.Workflow `json:"current_workflow,omitempty"` -} - -// GenerateWithTools performs workflow generation with pre-selected tools (Step 2 only) -// POST /api/agents/:id/generate-with-tools -func (h *AgentHandler) GenerateWithTools(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - if h.workflowGeneratorV2Service == nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Workflow generator v2 service not available", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - // Parse request body - var req GenerateWithToolsRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if req.UserMessage == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "User message is required", - }) - } - - if len(req.SelectedTools) == 0 { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Selected tools are required", - }) - } - - log.Printf("🔧 [GENERATE-WITH-TOOLS] Generating workflow for agent %s with %d pre-selected tools (user: %s)", - agentID, len(req.SelectedTools), userID) - - // Build the multi-step request - multiStepReq := &services.MultiStepGenerateRequest{ - AgentID: agentID, - UserMessage: req.UserMessage, - ModelID: req.ModelID, - CurrentWorkflow: req.CurrentWorkflow, - } - - // Perform workflow generation with pre-selected tools - result, err := h.workflowGeneratorV2Service.Step2GenerateWorkflow(multiStepReq, req.SelectedTools, userID) - if err != nil { - log.Printf("❌ [GENERATE-WITH-TOOLS] Failed to generate workflow: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to generate workflow", - "details": err.Error(), - }) - } - - if !result.Success { - log.Printf("⚠️ [GENERATE-WITH-TOOLS] Workflow generation failed: %s", result.Error) - return c.Status(fiber.StatusUnprocessableEntity).JSON(result) - } - - // Generate suggested name and description for new workflows - agent, _ := h.agentService.GetAgent(agentID, userID) - shouldGenerateMetadata := agent != nil && agent.Name == "New Agent" - if shouldGenerateMetadata && h.workflowGeneratorService != nil { - metadata, err := h.workflowGeneratorService.GenerateAgentMetadata(req.UserMessage) - if err != nil { - log.Printf("⚠️ [GENERATE-WITH-TOOLS] Failed to generate agent metadata (non-fatal): %v", err) - } else if metadata.Name != "" { - result.SuggestedName = metadata.Name - result.SuggestedDescription = metadata.Description - - // Persist the generated name - updateReq := &models.UpdateAgentRequest{ - Name: metadata.Name, - Description: metadata.Description, - } - _, updateErr := h.agentService.UpdateAgent(agentID, userID, updateReq) - if updateErr != nil { - log.Printf("⚠️ [GENERATE-WITH-TOOLS] Failed to persist agent metadata (non-fatal): %v", updateErr) - } else { - log.Printf("💾 [GENERATE-WITH-TOOLS] Persisted agent metadata: name=%s", metadata.Name) - } - } - } - - log.Printf("✅ [GENERATE-WITH-TOOLS] Generated workflow for agent %s: %d blocks", - agentID, len(result.Workflow.Blocks)) - return c.JSON(result) -} - -// GenerateSampleInput uses AI to generate sample JSON input for a workflow -// POST /api/agents/:id/generate-sample-input -func (h *AgentHandler) GenerateSampleInput(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - // Parse request body - var req struct { - ModelID string `json:"model_id"` - } - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if req.ModelID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Model ID is required", - }) - } - - // Get the agent and workflow - agent, err := h.agentService.GetAgent(agentID, userID) - if err != nil { - log.Printf("❌ [SAMPLE-INPUT] Failed to get agent: %v", err) - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Agent not found", - }) - } - - if agent.Workflow == nil || len(agent.Workflow.Blocks) == 0 { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Workflow has no blocks", - }) - } - - // Generate sample input using the workflow generator service - sampleInput, err := h.workflowGeneratorService.GenerateSampleInput(agent.Workflow, req.ModelID, userID) - if err != nil { - log.Printf("❌ [SAMPLE-INPUT] Failed to generate sample input: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to generate sample input", - "details": err.Error(), - }) - } - - log.Printf("✅ [SAMPLE-INPUT] Generated sample input for agent %s", agentID) - return c.JSON(fiber.Map{ - "success": true, - "sample_input": sampleInput, - }) -} - -// Ask handles Ask mode requests - helps users understand their workflow -// POST /api/agents/ask -func (h *AgentHandler) Ask(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - var req struct { - AgentID string `json:"agent_id"` - Message string `json:"message"` - ModelID string `json:"model_id"` - Context struct { - Workflow *models.Workflow `json:"workflow"` - AvailableTools []map[string]string `json:"available_tools"` - DeploymentExample string `json:"deployment_example"` - } `json:"context"` - } - - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if req.AgentID == "" || req.Message == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "agent_id and message are required", - }) - } - - if h.providerService == nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Provider service not available", - }) - } - - // Get the agent to verify ownership - agent, err := h.agentService.GetAgent(req.AgentID, userID) - if err != nil { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Agent not found", - }) - } - - log.Printf("💬 [ASK] User %s asking about agent %s: %s", userID, agent.Name, req.Message) - - // Build context from workflow - var workflowContext string - if req.Context.Workflow != nil && len(req.Context.Workflow.Blocks) > 0 { - workflowContext = "\n\n## Current Workflow Structure\n" - for i, block := range req.Context.Workflow.Blocks { - desc := block.Description - if desc == "" { - desc = "No description" - } - workflowContext += fmt.Sprintf("%d. **%s** (%s): %s\n", i+1, block.Name, block.Type, desc) - } - } - - // Build tools context - var toolsContext string - if len(req.Context.AvailableTools) > 0 { - toolsContext = "\n\n## Available Tools\n" - for _, tool := range req.Context.AvailableTools { - toolsContext += fmt.Sprintf("- **%s**: %s (Category: %s)\n", - tool["name"], tool["description"], tool["category"]) - } - } - - // Build deployment context - var deploymentContext string - if req.Context.DeploymentExample != "" { - deploymentContext = "\n\n## Deployment API Example\n```bash\n" + req.Context.DeploymentExample + "\n```" - } - - // Build system prompt - systemPrompt := fmt.Sprintf(`You are an AI assistant helping users understand their workflow agent in ClaraVerse. - -**Agent Name**: %s -**Agent Description**: %s - -Your role is to: -1. Answer questions about the workflow structure and how it works -2. Explain what tools are available and how to use them -3. Help with deployment and API integration questions -4. Provide clear, concise explanations - -**IMPORTANT**: You are in "Ask" mode, which is for answering questions only. If the user asks you to modify the workflow (add, change, remove blocks), politely tell them to switch to "Builder" mode. - -%s%s%s - -Be helpful, clear, and concise. If you don't know something, say so.`, - agent.Name, - agent.Description, - workflowContext, - toolsContext, - deploymentContext, - ) - - // Call LLM with simple chat endpoint - modelID := req.ModelID - if modelID == "" { - modelID = "gpt-4.1" // Default model - } - - // Get provider for model - provider, err := h.providerService.GetByModelID(modelID) - if err != nil { - log.Printf("❌ [ASK] Failed to get provider for model %s: %v", modelID, err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": fmt.Sprintf("Model '%s' not found", modelID), - }) - } - - // Build OpenAI-compatible request - type Message struct { - Role string `json:"role"` - Content string `json:"content"` - } - type OpenAIRequest struct { - Model string `json:"model"` - Messages []Message `json:"messages"` - } - - reqBody, err := json.Marshal(OpenAIRequest{ - Model: modelID, - Messages: []Message{ - {Role: "system", Content: systemPrompt}, - {Role: "user", Content: req.Message}, - }, - }) - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to prepare request", - }) - } - - // Make HTTP request - httpReq, err := http.NewRequest("POST", provider.BaseURL+"/chat/completions", bytes.NewBuffer(reqBody)) - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to create HTTP request", - }) - } - - httpReq.Header.Set("Content-Type", "application/json") - httpReq.Header.Set("Authorization", "Bearer "+provider.APIKey) - - client := &http.Client{Timeout: 60 * time.Second} - resp, err := client.Do(httpReq) - if err != nil { - log.Printf("❌ [ASK] HTTP request failed: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get response from AI", - }) - } - defer resp.Body.Close() - - // Read response - body, err := io.ReadAll(resp.Body) - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to read response", - }) - } - - if resp.StatusCode != http.StatusOK { - log.Printf("⚠️ [ASK] API error: %s", string(body)) - return c.Status(resp.StatusCode).JSON(fiber.Map{ - "error": fmt.Sprintf("AI service error: %s", string(body)), - }) - } - - // Parse response - var apiResponse struct { - Choices []struct { - Message struct { - Content string `json:"content"` - } `json:"message"` - } `json:"choices"` - } - - if err := json.Unmarshal(body, &apiResponse); err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to parse AI response", - }) - } - - if len(apiResponse.Choices) == 0 { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "No response from AI", - }) - } - - responseText := apiResponse.Choices[0].Message.Content - - log.Printf("✅ [ASK] Response generated for agent %s", agent.Name) - return c.JSON(fiber.Map{ - "response": responseText, - }) -} diff --git a/backend/internal/handlers/apikey.go b/backend/internal/handlers/apikey.go deleted file mode 100644 index 7709b451..00000000 --- a/backend/internal/handlers/apikey.go +++ /dev/null @@ -1,175 +0,0 @@ -package handlers - -import ( - "claraverse/internal/models" - "claraverse/internal/services" - "log" - - "github.com/gofiber/fiber/v2" - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// APIKeyHandler handles API key management endpoints -type APIKeyHandler struct { - apiKeyService *services.APIKeyService -} - -// NewAPIKeyHandler creates a new API key handler -func NewAPIKeyHandler(apiKeyService *services.APIKeyService) *APIKeyHandler { - return &APIKeyHandler{ - apiKeyService: apiKeyService, - } -} - -// Create creates a new API key -// POST /api/keys -func (h *APIKeyHandler) Create(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - - var req models.CreateAPIKeyRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // Validate required fields - if req.Name == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Name is required", - }) - } - - if len(req.Scopes) == 0 { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "At least one scope is required", - }) - } - - result, err := h.apiKeyService.Create(c.Context(), userID, &req) - if err != nil { - log.Printf("❌ [APIKEY] Failed to create API key: %v", err) - // Check for limit error - if err.Error()[:14] == "API key limit" { - return c.Status(fiber.StatusForbidden).JSON(fiber.Map{ - "error": err.Error(), - }) - } - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": err.Error(), - }) - } - - return c.Status(fiber.StatusCreated).JSON(result) -} - -// List lists all API keys for the user -// GET /api/keys -func (h *APIKeyHandler) List(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - - keys, err := h.apiKeyService.ListByUser(c.Context(), userID) - if err != nil { - log.Printf("❌ [APIKEY] Failed to list API keys: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to list API keys", - }) - } - - if keys == nil { - keys = []*models.APIKeyListItem{} - } - - return c.JSON(fiber.Map{ - "keys": keys, - }) -} - -// Get retrieves a specific API key -// GET /api/keys/:id -func (h *APIKeyHandler) Get(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - keyIDStr := c.Params("id") - - keyID, err := primitive.ObjectIDFromHex(keyIDStr) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid key ID", - }) - } - - key, err := h.apiKeyService.GetByIDAndUser(c.Context(), keyID, userID) - if err != nil { - if err.Error() == "API key not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "API key not found", - }) - } - log.Printf("❌ [APIKEY] Failed to get API key: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get API key", - }) - } - - return c.JSON(key.ToListItem()) -} - -// Revoke revokes an API key (soft delete) -// POST /api/keys/:id/revoke -func (h *APIKeyHandler) Revoke(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - keyIDStr := c.Params("id") - - keyID, err := primitive.ObjectIDFromHex(keyIDStr) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid key ID", - }) - } - - if err := h.apiKeyService.Revoke(c.Context(), keyID, userID); err != nil { - if err.Error() == "API key not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "API key not found", - }) - } - log.Printf("❌ [APIKEY] Failed to revoke API key: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to revoke API key", - }) - } - - return c.JSON(fiber.Map{ - "message": "API key revoked successfully", - }) -} - -// Delete permanently deletes an API key -// DELETE /api/keys/:id -func (h *APIKeyHandler) Delete(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - keyIDStr := c.Params("id") - - keyID, err := primitive.ObjectIDFromHex(keyIDStr) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid key ID", - }) - } - - if err := h.apiKeyService.Delete(c.Context(), keyID, userID); err != nil { - if err.Error() == "API key not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "API key not found", - }) - } - log.Printf("❌ [APIKEY] Failed to delete API key: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to delete API key", - }) - } - - return c.JSON(fiber.Map{ - "message": "API key deleted successfully", - }) -} diff --git a/backend/internal/handlers/audio.go b/backend/internal/handlers/audio.go deleted file mode 100644 index 67e89739..00000000 --- a/backend/internal/handlers/audio.go +++ /dev/null @@ -1,94 +0,0 @@ -package handlers - -import ( - "claraverse/internal/audio" - "fmt" - "log" - "os" - "path/filepath" - - "github.com/gofiber/fiber/v2" - "github.com/google/uuid" -) - -// AudioHandler handles audio-related API requests -type AudioHandler struct{} - -// NewAudioHandler creates a new audio handler -func NewAudioHandler() *AudioHandler { - return &AudioHandler{} -} - -// Transcribe handles audio file transcription via OpenAI Whisper -func (h *AudioHandler) Transcribe(c *fiber.Ctx) error { - // Get the uploaded file - file, err := c.FormFile("file") - if err != nil { - log.Printf("❌ [AUDIO-API] No file uploaded: %v", err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "No audio file uploaded", - }) - } - - // Validate file size (max 25MB for Whisper) - if file.Size > 25*1024*1024 { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Audio file too large. Maximum size is 25MB", - }) - } - - // Get optional parameters - language := c.FormValue("language", "") - prompt := c.FormValue("prompt", "") - - // Create temp file to store the upload - tempDir := os.TempDir() - ext := filepath.Ext(file.Filename) - if ext == "" { - ext = ".webm" // Default extension for browser recordings - } - tempFile := filepath.Join(tempDir, fmt.Sprintf("audio_%s%s", uuid.New().String(), ext)) - - // Save uploaded file to temp location - if err := c.SaveFile(file, tempFile); err != nil { - log.Printf("❌ [AUDIO-API] Failed to save temp file: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to process audio file", - }) - } - defer os.Remove(tempFile) // Clean up temp file - - // Get audio service - audioService := audio.GetService() - if audioService == nil { - log.Printf("❌ [AUDIO-API] Audio service not initialized") - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "Audio transcription service not available. Please configure OpenAI provider.", - }) - } - - // Build transcription request - req := &audio.TranscribeRequest{ - AudioPath: tempFile, - Language: language, - Prompt: prompt, - } - - // Call transcription service - log.Printf("🎵 [AUDIO-API] Transcribing audio file: %s (%d bytes)", file.Filename, file.Size) - resp, err := audioService.Transcribe(req) - if err != nil { - log.Printf("❌ [AUDIO-API] Transcription failed: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": fmt.Sprintf("Transcription failed: %v", err), - }) - } - - log.Printf("✅ [AUDIO-API] Transcription complete: %d chars, language: %s", len(resp.Text), resp.Language) - - return c.JSON(fiber.Map{ - "text": resp.Text, - "language": resp.Language, - "duration": resp.Duration, - }) -} diff --git a/backend/internal/handlers/auth_local.go b/backend/internal/handlers/auth_local.go deleted file mode 100644 index be4d87d8..00000000 --- a/backend/internal/handlers/auth_local.go +++ /dev/null @@ -1,384 +0,0 @@ -package handlers - -import ( - "claraverse/internal/models" - "claraverse/internal/services" - "claraverse/pkg/auth" - "context" - "log" - "strings" - "time" - - "github.com/gofiber/fiber/v2" - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// LocalAuthHandler handles local JWT authentication endpoints -type LocalAuthHandler struct { - jwtAuth *auth.LocalJWTAuth - userService *services.UserService -} - -// NewLocalAuthHandler creates a new local auth handler -func NewLocalAuthHandler(jwtAuth *auth.LocalJWTAuth, userService *services.UserService) *LocalAuthHandler { - return &LocalAuthHandler{ - jwtAuth: jwtAuth, - userService: userService, - } -} - -// RegisterRequest is the request body for registration -type RegisterRequest struct { - Email string `json:"email"` - Password string `json:"password"` -} - -// LoginRequest is the request body for login -type LoginRequest struct { - Email string `json:"email"` - Password string `json:"password"` -} - -// RefreshTokenRequest is the request body for token refresh -type RefreshTokenRequest struct { - RefreshToken string `json:"refresh_token"` -} - -// AuthResponse is the response for successful authentication -type AuthResponse struct { - AccessToken string `json:"access_token"` - RefreshToken string `json:"refresh_token"` - User models.UserResponse `json:"user"` - ExpiresIn int `json:"expires_in"` // seconds -} - -// Register creates a new user account -// POST /api/auth/register -func (h *LocalAuthHandler) Register(c *fiber.Ctx) error { - var req RegisterRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // Validate email - req.Email = strings.TrimSpace(strings.ToLower(req.Email)) - if req.Email == "" || !strings.Contains(req.Email, "@") { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Valid email address is required", - }) - } - - // Validate password - if err := auth.ValidatePassword(req.Password); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": err.Error(), - }) - } - - ctx := context.Background() - - // Check if user already exists - existingUser, _ := h.userService.GetUserByEmail(ctx, req.Email) - if existingUser != nil { - return c.Status(fiber.StatusConflict).JSON(fiber.Map{ - "error": "User with this email already exists", - }) - } - - // Hash password - passwordHash, err := h.jwtAuth.HashPassword(req.Password) - if err != nil { - log.Printf("❌ Failed to hash password: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to create account", - }) - } - - // Check if this is the first user (first user becomes admin) - userCount, err := h.userService.GetUserCount(ctx) - if err != nil { - log.Printf("⚠️ Failed to get user count: %v", err) - userCount = 1 // Default to non-admin if count check fails - } - - // Determine user role (first user = admin, others = user) - userRole := "user" - if userCount == 0 { - userRole = "admin" - log.Printf("🎉 Creating first user as admin: %s", req.Email) - } - - // Create user - user := &models.User{ - ID: primitive.NewObjectID(), - Email: req.Email, - PasswordHash: passwordHash, - EmailVerified: true, // Auto-verify in dev mode (no SMTP) - RefreshTokenVersion: 0, - Role: userRole, - CreatedAt: time.Now(), - LastLoginAt: time.Now(), - SubscriptionTier: "pro", // Default: all users get Pro tier - SubscriptionStatus: "active", - Preferences: models.UserPreferences{ - StoreBuilderChatHistory: true, - MemoryEnabled: false, - }, - } - - // Save user to database - if err := h.userService.CreateUser(ctx, user); err != nil { - log.Printf("❌ Failed to create user: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to create account", - }) - } - - // Generate tokens - accessToken, refreshToken, err := h.jwtAuth.GenerateTokens(user.ID.Hex(), user.Email, user.Role) - if err != nil { - log.Printf("❌ Failed to generate tokens: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to generate authentication tokens", - }) - } - - // Set refresh token as httpOnly cookie - c.Cookie(&fiber.Cookie{ - Name: "refresh_token", - Value: refreshToken, - Expires: time.Now().Add(7 * 24 * time.Hour), // 7 days - HTTPOnly: true, - Secure: c.Protocol() == "https", // HTTPS only in production - SameSite: "Strict", - Path: "/api/auth", - }) - - log.Printf("✅ User registered: %s (%s)", user.Email, user.ID.Hex()) - - return c.Status(fiber.StatusCreated).JSON(AuthResponse{ - AccessToken: accessToken, - RefreshToken: refreshToken, - User: user.ToResponse(), - ExpiresIn: 15 * 60, // 15 minutes in seconds - }) -} - -// Login authenticates a user -// POST /api/auth/login -func (h *LocalAuthHandler) Login(c *fiber.Ctx) error { - var req LoginRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - req.Email = strings.TrimSpace(strings.ToLower(req.Email)) - ctx := context.Background() - - // Get user by email - user, err := h.userService.GetUserByEmail(ctx, req.Email) - if err != nil || user == nil { - // Use constant-time response to prevent email enumeration - time.Sleep(200 * time.Millisecond) - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Invalid email or password", - }) - } - - // Verify password - valid, err := h.jwtAuth.VerifyPassword(user.PasswordHash, req.Password) - if err != nil || !valid { - log.Printf("⚠️ Failed login attempt for user: %s", req.Email) - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Invalid email or password", - }) - } - - // Update last login time - user.LastLoginAt = time.Now() - if err := h.userService.UpdateUser(ctx, user); err != nil { - log.Printf("⚠️ Failed to update last login time: %v", err) - // Non-critical, continue - } - - // Generate tokens - accessToken, refreshToken, err := h.jwtAuth.GenerateTokens(user.ID.Hex(), user.Email, user.Role) - if err != nil { - log.Printf("❌ Failed to generate tokens: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to generate authentication tokens", - }) - } - - // Set refresh token as httpOnly cookie - c.Cookie(&fiber.Cookie{ - Name: "refresh_token", - Value: refreshToken, - Expires: time.Now().Add(7 * 24 * time.Hour), - HTTPOnly: true, - Secure: c.Protocol() == "https", - SameSite: "Strict", - Path: "/api/auth", - }) - - log.Printf("✅ User logged in: %s (%s)", user.Email, user.ID.Hex()) - - return c.JSON(AuthResponse{ - AccessToken: accessToken, - RefreshToken: refreshToken, - User: user.ToResponse(), - ExpiresIn: 15 * 60, - }) -} - -// RefreshToken generates a new access token from a refresh token -// POST /api/auth/refresh -func (h *LocalAuthHandler) RefreshToken(c *fiber.Ctx) error { - // Try to get refresh token from cookie first - refreshToken := c.Cookies("refresh_token") - - // Fallback to request body - if refreshToken == "" { - var req RefreshTokenRequest - if err := c.BodyParser(&req); err == nil { - refreshToken = req.RefreshToken - } - } - - if refreshToken == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Refresh token is required", - }) - } - - // Verify refresh token - claims, err := h.jwtAuth.VerifyRefreshToken(refreshToken) - if err != nil { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Invalid or expired refresh token", - }) - } - - ctx := context.Background() - - // Get user from database - userID, err := primitive.ObjectIDFromHex(claims.UserID) - if err != nil { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Invalid user ID in token", - }) - } - - user, err := h.userService.GetUserByID(ctx, userID) - if err != nil || user == nil { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "User not found", - }) - } - - // Check if refresh token version matches (for revocation) - // Note: This would require storing token version in claims - // For now, skip this check - implement later with Redis - - // Generate new access token (refresh token remains valid) - newAccessToken, _, err := h.jwtAuth.GenerateTokens(user.ID.Hex(), user.Email, user.Role) - if err != nil { - log.Printf("❌ Failed to generate new access token: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to refresh token", - }) - } - - return c.JSON(fiber.Map{ - "access_token": newAccessToken, - "expires_in": 15 * 60, - }) -} - -// Logout invalidates the refresh token -// POST /api/auth/logout -func (h *LocalAuthHandler) Logout(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - // Allow logout even if not authenticated (clear cookie) - c.ClearCookie("refresh_token") - return c.JSON(fiber.Map{ - "message": "Logged out successfully", - }) - } - - ctx := context.Background() - - // Increment refresh token version to invalidate all existing tokens - objID, err := primitive.ObjectIDFromHex(userID) - if err == nil { - // Increment token version in database - _, err = h.userService.Collection().UpdateOne(ctx, - bson.M{"_id": objID}, - bson.M{"$inc": bson.M{"refreshTokenVersion": 1}}, - ) - if err != nil { - log.Printf("⚠️ Failed to increment token version: %v", err) - // Non-critical, continue - } - } - - // Clear refresh token cookie - c.ClearCookie("refresh_token") - - log.Printf("✅ User logged out: %s", userID) - - return c.JSON(fiber.Map{ - "message": "Logged out successfully", - }) -} - -// GetCurrentUser returns the currently authenticated user -// GET /api/auth/me -func (h *LocalAuthHandler) GetCurrentUser(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - ctx := context.Background() - objID, err := primitive.ObjectIDFromHex(userID) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid user ID", - }) - } - - user, err := h.userService.GetUserByID(ctx, objID) - if err != nil || user == nil { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "User not found", - }) - } - - return c.JSON(user.ToResponse()) -} - -// GetStatus returns system status for unauthenticated users -// GET /api/auth/status -func (h *LocalAuthHandler) GetStatus(c *fiber.Ctx) error { - ctx := context.Background() - - // Check if any users exist - userCount, err := h.userService.GetUserCount(ctx) - if err != nil { - log.Printf("⚠️ Failed to get user count: %v", err) - userCount = 0 - } - - return c.JSON(fiber.Map{ - "has_users": userCount > 0, - }) -} diff --git a/backend/internal/handlers/chat_sync.go b/backend/internal/handlers/chat_sync.go deleted file mode 100644 index e6ea31b9..00000000 --- a/backend/internal/handlers/chat_sync.go +++ /dev/null @@ -1,333 +0,0 @@ -package handlers - -import ( - "claraverse/internal/models" - "claraverse/internal/services" - "log" - "strconv" - "strings" - - "github.com/gofiber/fiber/v2" -) - -// ChatSyncHandler handles HTTP requests for chat sync operations -type ChatSyncHandler struct { - service *services.ChatSyncService -} - -// NewChatSyncHandler creates a new chat sync handler -func NewChatSyncHandler(service *services.ChatSyncService) *ChatSyncHandler { - return &ChatSyncHandler{ - service: service, - } -} - -// CreateOrUpdate creates a new chat or updates an existing one -// POST /api/chats -func (h *ChatSyncHandler) CreateOrUpdate(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - var req models.CreateChatRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if req.ID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Chat ID is required", - }) - } - - chat, err := h.service.CreateOrUpdateChat(c.Context(), userID, &req) - if err != nil { - log.Printf("❌ Failed to create/update chat: %v", err) - - // Check for version conflict (use strings.Contains to avoid panic on short errors) - errMsg := err.Error() - if strings.Contains(errMsg, "version conflict") { - return c.Status(fiber.StatusConflict).JSON(fiber.Map{ - "error": "Version conflict - chat was modified by another device", - }) - } - - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to save chat", - }) - } - - log.Printf("✅ Chat %s saved for user %s (version: %d)", req.ID, userID, chat.Version) - return c.Status(fiber.StatusOK).JSON(chat) -} - -// Get retrieves a single chat with decrypted messages -// GET /api/chats/:id -func (h *ChatSyncHandler) Get(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - chatID := c.Params("id") - if chatID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Chat ID is required", - }) - } - - chat, err := h.service.GetChat(c.Context(), userID, chatID) - if err != nil { - if err.Error() == "chat not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Chat not found", - }) - } - log.Printf("❌ Failed to get chat: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get chat", - }) - } - - return c.JSON(chat) -} - -// List returns a paginated list of chats -// GET /api/chats?page=1&page_size=20&starred=true -func (h *ChatSyncHandler) List(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - page, _ := strconv.Atoi(c.Query("page", "1")) - pageSize, _ := strconv.Atoi(c.Query("page_size", "20")) - starred := c.Query("starred") == "true" - - chats, err := h.service.ListChats(c.Context(), userID, page, pageSize, starred) - if err != nil { - log.Printf("❌ Failed to list chats: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to list chats", - }) - } - - return c.JSON(chats) -} - -// Update performs a partial update on a chat -// PUT /api/chats/:id -func (h *ChatSyncHandler) Update(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - chatID := c.Params("id") - if chatID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Chat ID is required", - }) - } - - var req models.UpdateChatRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - chat, err := h.service.UpdateChat(c.Context(), userID, chatID, &req) - if err != nil { - if err.Error() == "chat not found or version conflict" { - return c.Status(fiber.StatusConflict).JSON(fiber.Map{ - "error": "Chat not found or version conflict", - }) - } - log.Printf("❌ Failed to update chat: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to update chat", - }) - } - - log.Printf("✅ Chat %s updated for user %s", chatID, userID) - return c.JSON(chat) -} - -// Delete removes a chat -// DELETE /api/chats/:id -func (h *ChatSyncHandler) Delete(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - chatID := c.Params("id") - if chatID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Chat ID is required", - }) - } - - err := h.service.DeleteChat(c.Context(), userID, chatID) - if err != nil { - if err.Error() == "chat not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Chat not found", - }) - } - log.Printf("❌ Failed to delete chat: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to delete chat", - }) - } - - log.Printf("✅ Chat %s deleted for user %s", chatID, userID) - return c.JSON(fiber.Map{ - "success": true, - "message": "Chat deleted", - }) -} - -// BulkSync uploads multiple chats at once -// POST /api/chats/sync -func (h *ChatSyncHandler) BulkSync(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - var req models.BulkSyncRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if len(req.Chats) == 0 { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "No chats provided", - }) - } - - // Limit bulk sync to prevent abuse - if len(req.Chats) > 100 { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Maximum 100 chats per bulk sync", - }) - } - - result, err := h.service.BulkSync(c.Context(), userID, &req) - if err != nil { - log.Printf("❌ Failed to bulk sync: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to bulk sync chats", - }) - } - - log.Printf("✅ Bulk sync for user %s: %d synced, %d failed", userID, result.Synced, result.Failed) - return c.JSON(result) -} - -// SyncAll returns all chats for initial sync -// GET /api/chats/sync -func (h *ChatSyncHandler) SyncAll(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - result, err := h.service.GetAllChats(c.Context(), userID) - if err != nil { - log.Printf("❌ Failed to get all chats: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get chats", - }) - } - - log.Printf("✅ Sync all for user %s: %d chats", userID, result.TotalCount) - return c.JSON(result) -} - -// AddMessage adds a single message to a chat -// POST /api/chats/:id/messages -func (h *ChatSyncHandler) AddMessage(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - chatID := c.Params("id") - if chatID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Chat ID is required", - }) - } - - var req models.ChatAddMessageRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - chat, err := h.service.AddMessage(c.Context(), userID, chatID, &req) - if err != nil { - if err.Error() == "chat not found or version conflict" || err.Error() == "version conflict during update" { - return c.Status(fiber.StatusConflict).JSON(fiber.Map{ - "error": "Version conflict - please refresh and try again", - }) - } - log.Printf("❌ Failed to add message: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to add message", - }) - } - - log.Printf("✅ Message added to chat %s for user %s", chatID, userID) - return c.JSON(chat) -} - -// DeleteAll deletes all chats for a user (GDPR compliance) -// DELETE /api/chats -func (h *ChatSyncHandler) DeleteAll(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - count, err := h.service.DeleteAllUserChats(c.Context(), userID) - if err != nil { - log.Printf("❌ Failed to delete all chats: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to delete chats", - }) - } - - log.Printf("✅ Deleted %d chats for user %s", count, userID) - return c.JSON(fiber.Map{ - "success": true, - "deleted": count, - }) -} diff --git a/backend/internal/handlers/chat_sync_test.go b/backend/internal/handlers/chat_sync_test.go deleted file mode 100644 index 8b15a294..00000000 --- a/backend/internal/handlers/chat_sync_test.go +++ /dev/null @@ -1,405 +0,0 @@ -package handlers - -import ( - "bytes" - "claraverse/internal/models" - "encoding/json" - "io" - "net/http/httptest" - "testing" - - "github.com/gofiber/fiber/v2" -) - -// Mock user middleware for testing -func mockAuthMiddleware(userID string) fiber.Handler { - return func(c *fiber.Ctx) error { - c.Locals("user_id", userID) - return c.Next() - } -} - -func TestChatSyncHandler_CreateOrUpdate_Validation(t *testing.T) { - tests := []struct { - name string - userID string - body interface{} - expectedStatus int - expectedError string - }{ - { - name: "missing user ID", - userID: "", - body: models.CreateChatRequest{ID: "chat-1", Title: "Test"}, - expectedStatus: fiber.StatusUnauthorized, - expectedError: "Authentication required", - }, - { - name: "empty chat ID", - userID: "user-123", - body: models.CreateChatRequest{ID: "", Title: "Test"}, - expectedStatus: fiber.StatusBadRequest, - expectedError: "Chat ID is required", - }, - { - name: "invalid JSON body", - userID: "user-123", - body: "not json", - expectedStatus: fiber.StatusBadRequest, - expectedError: "Invalid request body", - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - app := fiber.New() - - // Add mock auth middleware - app.Use(mockAuthMiddleware(tt.userID)) - - // Create handler with nil service (will fail on actual operations but validation should happen first) - handler := &ChatSyncHandler{service: nil} - app.Post("/chats", handler.CreateOrUpdate) - - var body []byte - var err error - if str, ok := tt.body.(string); ok { - body = []byte(str) - } else { - body, err = json.Marshal(tt.body) - if err != nil { - t.Fatalf("Failed to marshal body: %v", err) - } - } - - req := httptest.NewRequest("POST", "/chats", bytes.NewReader(body)) - req.Header.Set("Content-Type", "application/json") - - resp, err := app.Test(req, -1) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - - if resp.StatusCode != tt.expectedStatus { - t.Errorf("Expected status %d, got %d", tt.expectedStatus, resp.StatusCode) - } - - // Check error message - respBody, _ := io.ReadAll(resp.Body) - var result map[string]string - json.Unmarshal(respBody, &result) - - if result["error"] != tt.expectedError { - t.Errorf("Expected error %q, got %q", tt.expectedError, result["error"]) - } - }) - } -} - -func TestChatSyncHandler_Get_Validation(t *testing.T) { - // Test only auth validation - service calls will panic with nil service - app := fiber.New() - app.Use(mockAuthMiddleware("")) - - handler := &ChatSyncHandler{service: nil} - app.Get("/chats/:id", handler.Get) - - req := httptest.NewRequest("GET", "/chats/chat-123", nil) - resp, err := app.Test(req, -1) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - - if resp.StatusCode != fiber.StatusUnauthorized { - t.Errorf("Expected status %d, got %d", fiber.StatusUnauthorized, resp.StatusCode) - } - - respBody, _ := io.ReadAll(resp.Body) - var result map[string]string - json.Unmarshal(respBody, &result) - - if result["error"] != "Authentication required" { - t.Errorf("Expected error %q, got %q", "Authentication required", result["error"]) - } -} - -func TestChatSyncHandler_List_Validation(t *testing.T) { - // Test only auth validation - service calls will panic with nil service - app := fiber.New() - app.Use(mockAuthMiddleware("")) - - handler := &ChatSyncHandler{service: nil} - app.Get("/chats", handler.List) - - req := httptest.NewRequest("GET", "/chats", nil) - resp, err := app.Test(req, -1) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - - if resp.StatusCode != fiber.StatusUnauthorized { - t.Errorf("Expected status %d, got %d", fiber.StatusUnauthorized, resp.StatusCode) - } -} - -func TestChatSyncHandler_Update_Validation(t *testing.T) { - app := fiber.New() - app.Use(mockAuthMiddleware("")) - - handler := &ChatSyncHandler{service: nil} - app.Put("/chats/:id", handler.Update) - - req := httptest.NewRequest("PUT", "/chats/chat-123", bytes.NewReader([]byte("{}"))) - req.Header.Set("Content-Type", "application/json") - - resp, _ := app.Test(req, -1) - if resp.StatusCode != fiber.StatusUnauthorized { - t.Errorf("Expected status %d, got %d", fiber.StatusUnauthorized, resp.StatusCode) - } -} - -func TestChatSyncHandler_Delete_Validation(t *testing.T) { - app := fiber.New() - app.Use(mockAuthMiddleware("")) - - handler := &ChatSyncHandler{service: nil} - app.Delete("/chats/:id", handler.Delete) - - req := httptest.NewRequest("DELETE", "/chats/chat-123", nil) - resp, _ := app.Test(req, -1) - - if resp.StatusCode != fiber.StatusUnauthorized { - t.Errorf("Expected status %d, got %d", fiber.StatusUnauthorized, resp.StatusCode) - } -} - -func TestChatSyncHandler_BulkSync_Validation(t *testing.T) { - tests := []struct { - name string - userID string - body interface{} - expectedStatus int - expectedError string - }{ - { - name: "missing user ID", - userID: "", - body: models.BulkSyncRequest{Chats: []models.CreateChatRequest{}}, - expectedStatus: fiber.StatusUnauthorized, - expectedError: "Authentication required", - }, - { - name: "empty chats array", - userID: "user-123", - body: models.BulkSyncRequest{Chats: []models.CreateChatRequest{}}, - expectedStatus: fiber.StatusBadRequest, - expectedError: "No chats provided", - }, - { - name: "too many chats", - userID: "user-123", - body: func() models.BulkSyncRequest { - chats := make([]models.CreateChatRequest, 101) - for i := range chats { - chats[i] = models.CreateChatRequest{ID: "chat-" + string(rune(i))} - } - return models.BulkSyncRequest{Chats: chats} - }(), - expectedStatus: fiber.StatusBadRequest, - expectedError: "Maximum 100 chats per bulk sync", - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - app := fiber.New() - app.Use(mockAuthMiddleware(tt.userID)) - - handler := &ChatSyncHandler{service: nil} - app.Post("/chats/sync", handler.BulkSync) - - body, _ := json.Marshal(tt.body) - req := httptest.NewRequest("POST", "/chats/sync", bytes.NewReader(body)) - req.Header.Set("Content-Type", "application/json") - - resp, err := app.Test(req, -1) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - - if resp.StatusCode != tt.expectedStatus { - t.Errorf("Expected status %d, got %d", tt.expectedStatus, resp.StatusCode) - } - - if tt.expectedError != "" { - respBody, _ := io.ReadAll(resp.Body) - var result map[string]string - json.Unmarshal(respBody, &result) - if result["error"] != tt.expectedError { - t.Errorf("Expected error %q, got %q", tt.expectedError, result["error"]) - } - } - }) - } -} - -func TestChatSyncHandler_SyncAll_Validation(t *testing.T) { - app := fiber.New() - app.Use(mockAuthMiddleware("")) - - handler := &ChatSyncHandler{service: nil} - app.Get("/chats/sync", handler.SyncAll) - - req := httptest.NewRequest("GET", "/chats/sync", nil) - resp, _ := app.Test(req, -1) - - if resp.StatusCode != fiber.StatusUnauthorized { - t.Errorf("Expected status %d, got %d", fiber.StatusUnauthorized, resp.StatusCode) - } -} - -func TestChatSyncHandler_AddMessage_Validation(t *testing.T) { - // Test only auth validation - app := fiber.New() - app.Use(mockAuthMiddleware("")) - - handler := &ChatSyncHandler{service: nil} - app.Post("/chats/:id/messages", handler.AddMessage) - - body, _ := json.Marshal(models.ChatAddMessageRequest{}) - req := httptest.NewRequest("POST", "/chats/chat-123/messages", bytes.NewReader(body)) - req.Header.Set("Content-Type", "application/json") - - resp, err := app.Test(req, -1) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - - if resp.StatusCode != fiber.StatusUnauthorized { - t.Errorf("Expected status %d, got %d", fiber.StatusUnauthorized, resp.StatusCode) - } -} - -func TestChatSyncHandler_DeleteAll_Validation(t *testing.T) { - app := fiber.New() - app.Use(mockAuthMiddleware("")) - - handler := &ChatSyncHandler{service: nil} - app.Delete("/chats", handler.DeleteAll) - - req := httptest.NewRequest("DELETE", "/chats", nil) - resp, _ := app.Test(req, -1) - - if resp.StatusCode != fiber.StatusUnauthorized { - t.Errorf("Expected status %d, got %d", fiber.StatusUnauthorized, resp.StatusCode) - } -} - -// Test request body parsing -func TestRequestBodyParsing(t *testing.T) { - tests := []struct { - name string - input string - shouldParse bool - }{ - { - name: "valid create chat request", - input: `{"id":"chat-1","title":"Test","messages":[{"id":"msg-1","role":"user","content":"Hello","timestamp":1700000000000}]}`, - shouldParse: true, - }, - { - name: "valid update request", - input: `{"title":"New Title","version":5}`, - shouldParse: true, - }, - { - name: "valid bulk sync request", - input: `{"chats":[{"id":"chat-1","title":"Test","messages":[]}]}`, - shouldParse: true, - }, - { - name: "request with attachments", - input: `{"id":"chat-1","title":"Test","messages":[{"id":"msg-1","role":"user","content":"Hello","timestamp":1700000000000,"attachments":[{"id":"att-1","name":"file.pdf","type":"application/pdf","size":1024}]}]}`, - shouldParse: true, - }, - { - name: "request with starred", - input: `{"id":"chat-1","title":"Test","messages":[],"is_starred":true}`, - shouldParse: true, - }, - { - name: "malformed JSON", - input: `{"id":"chat-1"`, - shouldParse: false, - }, - { - name: "empty object", - input: `{}`, - shouldParse: true, - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - var req models.CreateChatRequest - err := json.Unmarshal([]byte(tt.input), &req) - - if tt.shouldParse && err != nil { - t.Errorf("Expected to parse successfully, got error: %v", err) - } - if !tt.shouldParse && err == nil { - t.Error("Expected parse error, got nil") - } - }) - } -} - -// Test response format -func TestResponseFormat(t *testing.T) { - // Test ChatResponse JSON format - response := models.ChatResponse{ - ID: "chat-123", - Title: "Test Chat", - Messages: []models.ChatMessage{}, - IsStarred: true, - Version: 1, - } - - jsonData, err := json.Marshal(response) - if err != nil { - t.Fatalf("Failed to marshal response: %v", err) - } - - var parsed map[string]interface{} - json.Unmarshal(jsonData, &parsed) - - // Check snake_case field names - if _, ok := parsed["is_starred"]; !ok { - t.Error("Expected is_starred field in JSON") - } - if _, ok := parsed["created_at"]; !ok { - t.Error("Expected created_at field in JSON") - } - - // Test ChatListResponse JSON format - listResponse := models.ChatListResponse{ - Chats: []models.ChatListItem{}, - TotalCount: 10, - Page: 1, - PageSize: 20, - HasMore: false, - } - - jsonData, _ = json.Marshal(listResponse) - json.Unmarshal(jsonData, &parsed) - - if _, ok := parsed["total_count"]; !ok { - t.Error("Expected total_count field in JSON") - } - if _, ok := parsed["page_size"]; !ok { - t.Error("Expected page_size field in JSON") - } - if _, ok := parsed["has_more"]; !ok { - t.Error("Expected has_more field in JSON") - } -} diff --git a/backend/internal/handlers/composio_auth.go b/backend/internal/handlers/composio_auth.go deleted file mode 100644 index 69cdb252..00000000 --- a/backend/internal/handlers/composio_auth.go +++ /dev/null @@ -1,627 +0,0 @@ -package handlers - -import ( - "bytes" - "claraverse/internal/models" - "claraverse/internal/security" - "claraverse/internal/services" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "net/url" - "os" - "strings" - "time" - - "github.com/gofiber/fiber/v2" -) - -// OAuth scopes required for each service -var requiredScopes = map[string][]string{ - "gmail": { - "https://www.googleapis.com/auth/gmail.send", - "https://www.googleapis.com/auth/gmail.readonly", - "https://www.googleapis.com/auth/gmail.modify", - }, - "googlesheets": { - "https://www.googleapis.com/auth/spreadsheets", - }, -} - -// ComposioAuthHandler handles Composio OAuth flow -type ComposioAuthHandler struct { - credentialService *services.CredentialService - httpClient *http.Client - stateStore *security.OAuthStateStore -} - -// NewComposioAuthHandler creates a new Composio auth handler -func NewComposioAuthHandler(credentialService *services.CredentialService) *ComposioAuthHandler { - return &ComposioAuthHandler{ - credentialService: credentialService, - httpClient: &http.Client{Timeout: 30 * time.Second}, - stateStore: security.NewOAuthStateStore(), - } -} - -// InitiateGoogleSheetsAuth initiates OAuth flow for Google Sheets via Composio -// GET /api/integrations/composio/googlesheets/authorize -func (h *ComposioAuthHandler) InitiateGoogleSheetsAuth(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - log.Printf("❌ [COMPOSIO] COMPOSIO_API_KEY not set") - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Composio integration not configured", - }) - } - - // Use ClaraVerse user ID as Composio entity ID for simplicity - entityID := userID - - // Validate and sanitize redirect URL - redirectURL := c.Query("redirect_url") - if redirectURL == "" { - // Default to frontend settings page - frontendURL := os.Getenv("FRONTEND_URL") - if frontendURL == "" { - // Only allow localhost fallback in non-production environments - if os.Getenv("ENVIRONMENT") == "production" { - log.Printf("❌ [COMPOSIO] FRONTEND_URL not set in production") - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Configuration error", - }) - } - frontendURL = "http://localhost:5173" - } - redirectURL = fmt.Sprintf("%s/settings?tab=credentials", frontendURL) - } else { - // Validate redirect URL against allowed origins - if err := validateRedirectURL(redirectURL); err != nil { - log.Printf("⚠️ [COMPOSIO] Invalid redirect URL: %s", redirectURL) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid redirect URL", - }) - } - } - - // Get auth config ID from environment - // This must be created in Composio dashboard first - authConfigID := os.Getenv("COMPOSIO_GOOGLESHEETS_AUTH_CONFIG_ID") - if authConfigID == "" { - log.Printf("❌ [COMPOSIO] COMPOSIO_GOOGLESHEETS_AUTH_CONFIG_ID not set") - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Google Sheets auth config not configured. Please set COMPOSIO_GOOGLESHEETS_AUTH_CONFIG_ID in environment.", - }) - } - - // ✅ SECURITY FIX: Generate CSRF state token - stateToken, err := h.stateStore.GenerateState(userID, "googlesheets") - if err != nil { - log.Printf("❌ [COMPOSIO] Failed to generate state token: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to initiate OAuth", - }) - } - - // Call Composio API v3 to create a link for OAuth - // v3 uses /link endpoint which returns redirect_url - payload := map[string]interface{}{ - "auth_config_id": authConfigID, - "user_id": entityID, - } - - jsonData, err := json.Marshal(payload) - if err != nil { - log.Printf("❌ [COMPOSIO] Failed to marshal request: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to initiate OAuth", - }) - } - - // Create connection link using v3 endpoint - composioURL := "https://backend.composio.dev/api/v3/connected_accounts/link" - req, err := http.NewRequest("POST", composioURL, bytes.NewBuffer(jsonData)) - if err != nil { - log.Printf("❌ [COMPOSIO] Failed to create request: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to initiate OAuth", - }) - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("x-api-key", composioAPIKey) - - resp, err := h.httpClient.Do(req) - if err != nil { - log.Printf("❌ [COMPOSIO] Failed to call Composio API: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to initiate OAuth", - }) - } - defer resp.Body.Close() - - respBody, _ := io.ReadAll(resp.Body) - - if resp.StatusCode >= 400 { - log.Printf("❌ [COMPOSIO] API error (status %d): %s", resp.StatusCode, string(respBody)) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": fmt.Sprintf("Composio API error: %s", string(respBody)), - }) - } - - // Parse response to get redirectUrl - var composioResp map[string]interface{} - if err := json.Unmarshal(respBody, &composioResp); err != nil { - log.Printf("❌ [COMPOSIO] Failed to parse response: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to parse OAuth response", - }) - } - - // v3 API returns redirect_url (snake_case) - redirectURLFromComposio, ok := composioResp["redirect_url"].(string) - if !ok { - log.Printf("❌ [COMPOSIO] No redirect_url in response: %v", composioResp) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Invalid OAuth response from Composio", - }) - } - - // ✅ SECURITY FIX: Append state token to OAuth URL for CSRF protection - parsedURL, err := url.Parse(redirectURLFromComposio) - if err != nil { - log.Printf("❌ [COMPOSIO] Failed to parse OAuth URL: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Invalid OAuth URL", - }) - } - query := parsedURL.Query() - query.Set("state", stateToken) - parsedURL.RawQuery = query.Encode() - authURLWithState := parsedURL.String() - - log.Printf("✅ [COMPOSIO] Initiated Google Sheets OAuth for user %s", userID) - - // Return the OAuth URL to frontend - return c.JSON(fiber.Map{ - "authUrl": authURLWithState, - "entityId": entityID, - "redirectUrl": redirectURL, - }) -} - -// HandleComposioCallback handles OAuth callback from Composio -// GET /api/integrations/composio/callback -func (h *ComposioAuthHandler) HandleComposioCallback(c *fiber.Ctx) error { - // Get query parameters - code := c.Query("code") - state := c.Query("state") - errorParam := c.Query("error") - - // Get frontend URL - frontendURL := os.Getenv("FRONTEND_URL") - if frontendURL == "" { - if os.Getenv("ENVIRONMENT") == "production" { - log.Printf("❌ [COMPOSIO] FRONTEND_URL not set in production") - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Configuration error", - }) - } - frontendURL = "http://localhost:5173" - } - - if errorParam != "" { - log.Printf("❌ [COMPOSIO] OAuth error: %s", errorParam) - return c.Redirect(fmt.Sprintf("%s/settings?tab=credentials&error=%s", - frontendURL, url.QueryEscape(errorParam))) - } - - if code == "" { - log.Printf("❌ [COMPOSIO] No code in callback") - return c.Redirect(fmt.Sprintf("%s/settings?tab=credentials&error=no_code", frontendURL)) - } - - // ✅ SECURITY FIX: Validate CSRF state token - if state == "" { - log.Printf("❌ [COMPOSIO] Missing state token in callback") - return c.Redirect(fmt.Sprintf("%s/settings?tab=credentials&error=invalid_state", frontendURL)) - } - - userID, service, err := h.stateStore.ValidateState(state) - if err != nil { - log.Printf("❌ [COMPOSIO] Invalid state token: %v", err) - return c.Redirect(fmt.Sprintf("%s/settings?tab=credentials&error=invalid_state", frontendURL)) - } - - log.Printf("✅ [COMPOSIO] Valid OAuth callback for user %s, service: %s", userID, service) - - // ✅ SECURITY FIX: Store code server-side instead of passing in URL - // Generate a temporary session token to pass to frontend - sessionToken, err := h.stateStore.GenerateState(userID, service+"_callback") - if err != nil { - log.Printf("❌ [COMPOSIO] Failed to generate session token: %v", err) - return c.Redirect(fmt.Sprintf("%s/settings?tab=credentials&error=session_error", frontendURL)) - } - - // Store the authorization code temporarily (reusing state store for simplicity) - // In production, you might want a separate session store - codeStoreKey := "oauth_code:" + sessionToken - _, err = h.stateStore.GenerateState(codeStoreKey, code) // Store code using state as key - if err != nil { - log.Printf("❌ [COMPOSIO] Failed to store authorization code: %v", err) - return c.Redirect(fmt.Sprintf("%s/settings?tab=credentials&error=session_error", frontendURL)) - } - - // ✅ SECURITY FIX: Redirect without exposing authorization code in URL - redirectURL := fmt.Sprintf("%s/settings?tab=credentials&composio_success=true&service=%s&session=%s", - frontendURL, url.QueryEscape(service), url.QueryEscape(sessionToken)) - - log.Printf("✅ [COMPOSIO] OAuth callback successful, redirecting user %s", userID) - return c.Redirect(redirectURL) -} - -// GetConnectedAccount retrieves Composio connected account for entity -// GET /api/integrations/composio/connected-account -func (h *ComposioAuthHandler) GetConnectedAccount(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - entityID := userID // We use user ID as entity ID - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Composio integration not configured", - }) - } - - // Get connected accounts for entity using v3 API - baseURL := "https://backend.composio.dev/api/v3/connected_accounts" - params := url.Values{} - params.Add("user_ids", entityID) - fullURL := baseURL + "?" + params.Encode() - req, err := http.NewRequest("GET", fullURL, nil) - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to fetch connected account", - }) - } - - req.Header.Set("x-api-key", composioAPIKey) - - resp, err := h.httpClient.Do(req) - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to fetch connected account", - }) - } - defer resp.Body.Close() - - respBody, _ := io.ReadAll(resp.Body) - - if resp.StatusCode >= 400 { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": string(respBody), - }) - } - - // v3 API returns {items: [...], total_pages, current_page, ...} - var response struct { - Items []map[string]interface{} `json:"items"` - } - if err := json.Unmarshal(respBody, &response); err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to parse response", - }) - } - - // Find Google Sheets connected account - // v3 API uses toolkit.slug instead of integrationId - for _, account := range response.Items { - if toolkit, ok := account["toolkit"].(map[string]interface{}); ok { - if slug, ok := toolkit["slug"].(string); ok && slug == "googlesheets" { - return c.JSON(account) - } - } - } - - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "No Google Sheets connection found", - }) -} - -// CompleteComposioSetup creates credential after OAuth success -// POST /api/integrations/composio/complete-setup -func (h *ComposioAuthHandler) CompleteComposioSetup(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - - var req struct { - Name string `json:"name"` - IntegrationType string `json:"integrationType"` - } - - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if req.Name == "" { - req.Name = "Google Sheets" - } - - if req.IntegrationType == "" { - req.IntegrationType = "composio_googlesheets" - } - - // Extract service name from integration type (e.g., "composio_gmail" -> "gmail") - serviceName := req.IntegrationType - if len(serviceName) > 9 && serviceName[:9] == "composio_" { - serviceName = serviceName[9:] // Remove "composio_" prefix - } - - // Entity ID is the same as user ID - entityID := userID - - // Verify the connection exists in Composio - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Composio integration not configured", - }) - } - - // Get connected accounts to verify using v3 API - baseURL := "https://backend.composio.dev/api/v3/connected_accounts" - params := url.Values{} - params.Add("user_ids", entityID) - fullURL := baseURL + "?" + params.Encode() - httpReq, _ := http.NewRequest("GET", fullURL, nil) - httpReq.Header.Set("x-api-key", composioAPIKey) - - resp, err := h.httpClient.Do(httpReq) - if err != nil { - log.Printf("❌ [COMPOSIO] Failed to verify connection: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to verify Composio connection", - }) - } - defer resp.Body.Close() - - respBody, _ := io.ReadAll(resp.Body) - var response struct { - Items []map[string]interface{} `json:"items"` - } - if err := json.Unmarshal(respBody, &response); err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to verify connection", - }) - } - - // Check if the service is connected (v3 uses toolkit.slug) - found := false - for _, account := range response.Items { - if toolkit, ok := account["toolkit"].(map[string]interface{}); ok { - if slug, ok := toolkit["slug"].(string); ok && slug == serviceName { - found = true - break - } - } - } - - if !found { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": fmt.Sprintf("%s not connected in Composio. Please complete OAuth first.", strings.Title(serviceName)), - }) - } - - // Create credential - credential, err := h.credentialService.Create(c.Context(), userID, &models.CreateCredentialRequest{ - Name: req.Name, - IntegrationType: req.IntegrationType, - Data: map[string]interface{}{ - "composio_entity_id": entityID, - }, - }) - - if err != nil { - log.Printf("❌ [COMPOSIO] Failed to create credential: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to save credential", - }) - } - - log.Printf("✅ [COMPOSIO] Created credential for user %s with entity_id %s", userID, entityID) - - return c.Status(fiber.StatusCreated).JSON(credential) -} - -// InitiateGmailAuth initiates OAuth flow for Gmail via Composio -// GET /api/integrations/composio/gmail/authorize -func (h *ComposioAuthHandler) InitiateGmailAuth(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - log.Printf("❌ [COMPOSIO] COMPOSIO_API_KEY not set") - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Composio integration not configured", - }) - } - - // Use ClaraVerse user ID as Composio entity ID - entityID := userID - - // Validate and sanitize redirect URL - redirectURL := c.Query("redirect_url") - if redirectURL == "" { - // Default to frontend settings page - frontendURL := os.Getenv("FRONTEND_URL") - if frontendURL == "" { - // Only allow localhost fallback in non-production environments - if os.Getenv("ENVIRONMENT") == "production" { - log.Printf("❌ [COMPOSIO] FRONTEND_URL not set in production") - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Configuration error", - }) - } - frontendURL = "http://localhost:5173" - } - redirectURL = fmt.Sprintf("%s/settings?tab=credentials", frontendURL) - } else { - // Validate redirect URL against allowed origins - if err := validateRedirectURL(redirectURL); err != nil { - log.Printf("⚠️ [COMPOSIO] Invalid redirect URL: %s", redirectURL) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid redirect URL", - }) - } - } - - // Get auth config ID from environment - authConfigID := os.Getenv("COMPOSIO_GMAIL_AUTH_CONFIG_ID") - if authConfigID == "" { - log.Printf("❌ [COMPOSIO] COMPOSIO_GMAIL_AUTH_CONFIG_ID not set") - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Gmail auth config not configured. Please set COMPOSIO_GMAIL_AUTH_CONFIG_ID in environment.", - }) - } - - // ✅ SECURITY FIX: Generate CSRF state token - stateToken, err := h.stateStore.GenerateState(userID, "gmail") - if err != nil { - log.Printf("❌ [COMPOSIO] Failed to generate state token: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to initiate OAuth", - }) - } - - // Call Composio API v3 to create a link for OAuth - payload := map[string]interface{}{ - "auth_config_id": authConfigID, - "user_id": entityID, - } - - jsonData, err := json.Marshal(payload) - if err != nil { - log.Printf("❌ [COMPOSIO] Failed to marshal request: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to initiate OAuth", - }) - } - - // Create connection link using v3 endpoint - composioURL := "https://backend.composio.dev/api/v3/connected_accounts/link" - req, err := http.NewRequest("POST", composioURL, bytes.NewBuffer(jsonData)) - if err != nil { - log.Printf("❌ [COMPOSIO] Failed to create request: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to initiate OAuth", - }) - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("x-api-key", composioAPIKey) - - resp, err := h.httpClient.Do(req) - if err != nil { - log.Printf("❌ [COMPOSIO] Failed to call Composio API: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to initiate OAuth", - }) - } - defer resp.Body.Close() - - respBody, _ := io.ReadAll(resp.Body) - - if resp.StatusCode >= 400 { - log.Printf("❌ [COMPOSIO] API error (status %d): %s", resp.StatusCode, string(respBody)) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": fmt.Sprintf("Composio API error: %s", string(respBody)), - }) - } - - // Parse response to get redirectUrl - var composioResp map[string]interface{} - if err := json.Unmarshal(respBody, &composioResp); err != nil { - log.Printf("❌ [COMPOSIO] Failed to parse response: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to parse OAuth response", - }) - } - - // v3 API returns redirect_url - redirectURLFromComposio, ok := composioResp["redirect_url"].(string) - if !ok { - log.Printf("❌ [COMPOSIO] No redirect_url in response: %v", composioResp) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Invalid OAuth response from Composio", - }) - } - - // ✅ SECURITY FIX: Append state token to OAuth URL for CSRF protection - gmailOauthURL, err := url.Parse(redirectURLFromComposio) - if err != nil { - log.Printf("❌ [COMPOSIO] Failed to parse OAuth URL: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Invalid OAuth URL", - }) - } - gmailQueryParams := gmailOauthURL.Query() - gmailQueryParams.Set("state", stateToken) - gmailOauthURL.RawQuery = gmailQueryParams.Encode() - authURLWithState := gmailOauthURL.String() - - log.Printf("✅ [COMPOSIO] Initiated Gmail OAuth for user %s", userID) - - // Return the OAuth URL to frontend - return c.JSON(fiber.Map{ - "authUrl": authURLWithState, - "entityId": entityID, - "redirectUrl": redirectURL, - }) -} - -// validateRedirectURL validates that a redirect URL is safe -func validateRedirectURL(redirectURL string) error { - parsedURL, err := url.Parse(redirectURL) - if err != nil { - return fmt.Errorf("invalid URL format") - } - - // Must use HTTPS in production - if os.Getenv("ENVIRONMENT") == "production" && parsedURL.Scheme != "https" { - return fmt.Errorf("redirect URL must use HTTPS in production") - } - - // Allow localhost for development - if parsedURL.Scheme == "http" && (parsedURL.Hostname() == "localhost" || parsedURL.Hostname() == "127.0.0.1") { - return nil - } - - // Validate against FRONTEND_URL or ALLOWED_ORIGINS - frontendURL := os.Getenv("FRONTEND_URL") - allowedOrigins := os.Getenv("ALLOWED_ORIGINS") - - if frontendURL != "" { - parsedFrontend, err := url.Parse(frontendURL) - if err == nil && parsedURL.Host == parsedFrontend.Host { - return nil - } - } - - if allowedOrigins != "" { - origins := strings.Split(allowedOrigins, ",") - for _, origin := range origins { - origin = strings.TrimSpace(origin) - parsedOrigin, err := url.Parse(origin) - if err == nil && parsedURL.Host == parsedOrigin.Host { - return nil - } - } - } - - return fmt.Errorf("redirect URL not in allowed origins") -} diff --git a/backend/internal/handlers/config.go b/backend/internal/handlers/config.go deleted file mode 100644 index 91d2fd4e..00000000 --- a/backend/internal/handlers/config.go +++ /dev/null @@ -1,38 +0,0 @@ -package handlers - -import ( - "claraverse/internal/services" - - "github.com/gofiber/fiber/v2" -) - -// ConfigHandler handles configuration API requests -type ConfigHandler struct { - configService *services.ConfigService -} - -// NewConfigHandler creates a new config handler -func NewConfigHandler() *ConfigHandler { - return &ConfigHandler{ - configService: services.GetConfigService(), - } -} - -// GetRecommendedModels returns recommended models for all providers -func (h *ConfigHandler) GetRecommendedModels(c *fiber.Ctx) error { - recommended := h.configService.GetAllRecommendedModels() - - // Convert to a frontend-friendly format - response := make(map[string]interface{}) - for providerID, models := range recommended { - response[string(rune(providerID+'0'))] = fiber.Map{ - "top": models.Top, - "medium": models.Medium, - "fastest": models.Fastest, - } - } - - return c.JSON(fiber.Map{ - "recommended_models": recommended, - }) -} diff --git a/backend/internal/handlers/conversation.go b/backend/internal/handlers/conversation.go deleted file mode 100644 index 858a59ca..00000000 --- a/backend/internal/handlers/conversation.go +++ /dev/null @@ -1,286 +0,0 @@ -package handlers - -import ( - "claraverse/internal/models" - "claraverse/internal/services" - "log" - - "github.com/gofiber/fiber/v2" -) - -// ConversationHandler handles conversation-related HTTP requests -type ConversationHandler struct { - chatService *services.ChatService - builderService *services.BuilderConversationService -} - -// NewConversationHandler creates a new conversation handler -func NewConversationHandler(chatService *services.ChatService, builderService *services.BuilderConversationService) *ConversationHandler { - return &ConversationHandler{ - chatService: chatService, - builderService: builderService, - } -} - -// GetStatus returns the status of a conversation (exists, has files, time until expiration) -// GET /api/conversations/:id/status -func (h *ConversationHandler) GetStatus(c *fiber.Ctx) error { - conversationID := c.Params("id") - - if conversationID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Conversation ID is required", - }) - } - - // Get user ID from context (set by auth middleware) - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - log.Printf("📊 [CONVERSATION] Status check for conversation %s (user: %s)", conversationID, userID) - - // Verify conversation ownership - if !h.chatService.IsConversationOwner(conversationID, userID) { - return c.Status(fiber.StatusForbidden).JSON(fiber.Map{ - "error": "Access denied to this conversation", - }) - } - - status := h.chatService.GetConversationStatus(conversationID) - - return c.JSON(status) -} - -// ListBuilderConversations returns all builder conversations for an agent -// GET /api/agents/:id/conversations -func (h *ConversationHandler) ListBuilderConversations(c *fiber.Ctx) error { - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - if h.builderService == nil { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "Builder conversation service not available", - }) - } - - conversations, err := h.builderService.GetConversationsByAgent(c.Context(), agentID, userID) - if err != nil { - log.Printf("❌ Failed to list builder conversations: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to list conversations", - }) - } - - return c.JSON(conversations) -} - -// GetBuilderConversation returns a specific builder conversation -// GET /api/agents/:id/conversations/:convId -func (h *ConversationHandler) GetBuilderConversation(c *fiber.Ctx) error { - conversationID := c.Params("convId") - if conversationID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Conversation ID is required", - }) - } - - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - if h.builderService == nil { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "Builder conversation service not available", - }) - } - - conversation, err := h.builderService.GetConversation(c.Context(), conversationID, userID) - if err != nil { - log.Printf("❌ Failed to get builder conversation: %v", err) - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Conversation not found", - }) - } - - return c.JSON(conversation) -} - -// CreateBuilderConversation creates a new builder conversation for an agent -// POST /api/agents/:id/conversations -func (h *ConversationHandler) CreateBuilderConversation(c *fiber.Ctx) error { - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - if h.builderService == nil { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "Builder conversation service not available", - }) - } - - var req struct { - ModelID string `json:"model_id"` - } - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - conversation, err := h.builderService.CreateConversation(c.Context(), agentID, userID, req.ModelID) - if err != nil { - log.Printf("❌ Failed to create builder conversation: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to create conversation", - }) - } - - log.Printf("✅ Created builder conversation %s for agent %s", conversation.ID, agentID) - return c.Status(fiber.StatusCreated).JSON(conversation) -} - -// AddBuilderMessage adds a message to a builder conversation -// POST /api/agents/:id/conversations/:convId/messages -func (h *ConversationHandler) AddBuilderMessage(c *fiber.Ctx) error { - conversationID := c.Params("convId") - if conversationID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Conversation ID is required", - }) - } - - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - if h.builderService == nil { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "Builder conversation service not available", - }) - } - - var req models.AddMessageRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if req.Role == "" || req.Content == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Role and content are required", - }) - } - - message, err := h.builderService.AddMessage(c.Context(), conversationID, userID, &req) - if err != nil { - log.Printf("❌ Failed to add message to conversation: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to add message", - }) - } - - return c.Status(fiber.StatusCreated).JSON(message) -} - -// DeleteBuilderConversation deletes a builder conversation -// DELETE /api/agents/:id/conversations/:convId -func (h *ConversationHandler) DeleteBuilderConversation(c *fiber.Ctx) error { - conversationID := c.Params("convId") - if conversationID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Conversation ID is required", - }) - } - - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - if h.builderService == nil { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "Builder conversation service not available", - }) - } - - if err := h.builderService.DeleteConversation(c.Context(), conversationID, userID); err != nil { - log.Printf("❌ Failed to delete builder conversation: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to delete conversation", - }) - } - - return c.JSON(fiber.Map{ - "message": "Conversation deleted successfully", - }) -} - -// GetOrCreateBuilderConversation gets the most recent conversation or creates a new one -// GET /api/agents/:id/conversations/current -func (h *ConversationHandler) GetOrCreateBuilderConversation(c *fiber.Ctx) error { - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - if h.builderService == nil { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "Builder conversation service not available", - }) - } - - modelID := c.Query("model_id", "") - - conversation, err := h.builderService.GetOrCreateConversation(c.Context(), agentID, userID, modelID) - if err != nil { - log.Printf("❌ Failed to get/create builder conversation: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get conversation", - }) - } - - return c.JSON(conversation) -} diff --git a/backend/internal/handlers/credential.go b/backend/internal/handlers/credential.go deleted file mode 100644 index c7313ae0..00000000 --- a/backend/internal/handlers/credential.go +++ /dev/null @@ -1,337 +0,0 @@ -package handlers - -import ( - "claraverse/internal/models" - "claraverse/internal/services" - "log" - - "github.com/gofiber/fiber/v2" - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// CredentialHandler handles credential management endpoints -type CredentialHandler struct { - credentialService *services.CredentialService - credentialTester *CredentialTester -} - -// NewCredentialHandler creates a new credential handler -func NewCredentialHandler(credentialService *services.CredentialService) *CredentialHandler { - return &CredentialHandler{ - credentialService: credentialService, - credentialTester: NewCredentialTester(credentialService), - } -} - -// Create creates a new credential -// POST /api/credentials -func (h *CredentialHandler) Create(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - - var req models.CreateCredentialRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // Validate required fields - if req.Name == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Name is required", - }) - } - - if req.IntegrationType == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Integration type is required", - }) - } - - if req.Data == nil || len(req.Data) == 0 { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Credential data is required", - }) - } - - result, err := h.credentialService.Create(c.Context(), userID, &req) - if err != nil { - log.Printf("❌ [CREDENTIAL] Failed to create credential: %v", err) - - // Check for validation error - if _, ok := err.(*models.CredentialValidationError); ok { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": err.Error(), - }) - } - - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to create credential", - }) - } - - return c.Status(fiber.StatusCreated).JSON(result) -} - -// List lists all credentials for the user -// GET /api/credentials -func (h *CredentialHandler) List(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - integrationType := c.Query("type") // Optional filter by type - - var credentials []*models.CredentialListItem - var err error - - if integrationType != "" { - credentials, err = h.credentialService.ListByUserAndType(c.Context(), userID, integrationType) - } else { - credentials, err = h.credentialService.ListByUser(c.Context(), userID) - } - - if err != nil { - log.Printf("❌ [CREDENTIAL] Failed to list credentials: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to list credentials", - }) - } - - return c.JSON(models.GetCredentialsResponse{ - Credentials: credentials, - Total: len(credentials), - }) -} - -// Get retrieves a specific credential (metadata only) -// GET /api/credentials/:id -func (h *CredentialHandler) Get(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - credIDStr := c.Params("id") - - credID, err := primitive.ObjectIDFromHex(credIDStr) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid credential ID", - }) - } - - credential, err := h.credentialService.GetByIDAndUser(c.Context(), credID, userID) - if err != nil { - if err.Error() == "credential not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Credential not found", - }) - } - log.Printf("❌ [CREDENTIAL] Failed to get credential: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get credential", - }) - } - - return c.JSON(credential.ToListItem()) -} - -// Update updates a credential -// PUT /api/credentials/:id -func (h *CredentialHandler) Update(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - credIDStr := c.Params("id") - - credID, err := primitive.ObjectIDFromHex(credIDStr) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid credential ID", - }) - } - - var req models.UpdateCredentialRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // At least one field must be provided - if req.Name == "" && req.Data == nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "At least name or data must be provided", - }) - } - - result, err := h.credentialService.Update(c.Context(), credID, userID, &req) - if err != nil { - if err.Error() == "credential not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Credential not found", - }) - } - log.Printf("❌ [CREDENTIAL] Failed to update credential: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to update credential", - }) - } - - return c.JSON(result) -} - -// Delete permanently deletes a credential -// DELETE /api/credentials/:id -func (h *CredentialHandler) Delete(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - credIDStr := c.Params("id") - - credID, err := primitive.ObjectIDFromHex(credIDStr) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid credential ID", - }) - } - - if err := h.credentialService.Delete(c.Context(), credID, userID); err != nil { - if err.Error() == "credential not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Credential not found", - }) - } - log.Printf("❌ [CREDENTIAL] Failed to delete credential: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to delete credential", - }) - } - - return c.JSON(fiber.Map{ - "message": "Credential deleted successfully", - }) -} - -// Test tests a credential by making a real API call -// POST /api/credentials/:id/test -func (h *CredentialHandler) Test(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - credIDStr := c.Params("id") - - credID, err := primitive.ObjectIDFromHex(credIDStr) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid credential ID", - }) - } - - // Get and decrypt the credential - decrypted, err := h.credentialService.GetDecrypted(c.Context(), userID, credID) - if err != nil { - if err.Error() == "credential not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Credential not found", - }) - } - log.Printf("❌ [CREDENTIAL] Failed to get credential for testing: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get credential", - }) - } - - // Test the credential - result := h.credentialTester.Test(c.Context(), decrypted) - - // Update test status - status := "failed" - if result.Success { - status = "success" - } - if err := h.credentialService.UpdateTestStatus(c.Context(), credID, userID, status, nil); err != nil { - log.Printf("⚠️ [CREDENTIAL] Failed to update test status: %v", err) - } - - return c.JSON(result) -} - -// GetIntegrations returns all available integrations -// GET /api/integrations -func (h *CredentialHandler) GetIntegrations(c *fiber.Ctx) error { - categories := models.GetIntegrationsByCategory() - return c.JSON(models.GetIntegrationsResponse{ - Categories: categories, - }) -} - -// GetIntegration returns a specific integration -// GET /api/integrations/:id -func (h *CredentialHandler) GetIntegration(c *fiber.Ctx) error { - integrationID := c.Params("id") - - integration, exists := models.GetIntegration(integrationID) - if !exists { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Integration not found", - }) - } - - return c.JSON(integration) -} - -// GetCredentialsByIntegration returns credentials grouped by integration type -// GET /api/credentials/by-integration -func (h *CredentialHandler) GetCredentialsByIntegration(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - - credentials, err := h.credentialService.ListByUser(c.Context(), userID) - if err != nil { - log.Printf("❌ [CREDENTIAL] Failed to list credentials: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to list credentials", - }) - } - - // Group by integration type - groupedMap := make(map[string]*models.CredentialsByIntegration) - for _, cred := range credentials { - if _, exists := groupedMap[cred.IntegrationType]; !exists { - integration, _ := models.GetIntegration(cred.IntegrationType) - groupedMap[cred.IntegrationType] = &models.CredentialsByIntegration{ - IntegrationType: cred.IntegrationType, - Integration: integration, - Credentials: []*models.CredentialListItem{}, - } - } - groupedMap[cred.IntegrationType].Credentials = append( - groupedMap[cred.IntegrationType].Credentials, - cred, - ) - } - - // Convert to slice - var integrations []models.CredentialsByIntegration - for _, group := range groupedMap { - integrations = append(integrations, *group) - } - - return c.JSON(models.GetCredentialsByIntegrationResponse{ - Integrations: integrations, - }) -} - -// GetCredentialReferences returns credential references for LLM context -// GET /api/credentials/references -func (h *CredentialHandler) GetCredentialReferences(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - - // Optional filter by integration types - var integrationTypes []string - if types := c.Query("types"); types != "" { - // Parse comma-separated types - // For simplicity, just pass nil to get all - // In production, you'd parse the query param - } - - refs, err := h.credentialService.GetCredentialReferences(c.Context(), userID, integrationTypes) - if err != nil { - log.Printf("❌ [CREDENTIAL] Failed to get credential references: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get credential references", - }) - } - - return c.JSON(fiber.Map{ - "credentials": refs, - }) -} diff --git a/backend/internal/handlers/credential_tester.go b/backend/internal/handlers/credential_tester.go deleted file mode 100644 index 1cf8ecc0..00000000 --- a/backend/internal/handlers/credential_tester.go +++ /dev/null @@ -1,1657 +0,0 @@ -package handlers - -import ( - "bytes" - "claraverse/internal/models" - "claraverse/internal/services" - "context" - "encoding/base64" - "encoding/json" - "fmt" - "io" - "net/http" - "os" - "strings" - "time" -) - -// CredentialTester handles testing credentials for different integrations -type CredentialTester struct { - credentialService *services.CredentialService - httpClient *http.Client -} - -// NewCredentialTester creates a new credential tester -func NewCredentialTester(credentialService *services.CredentialService) *CredentialTester { - return &CredentialTester{ - credentialService: credentialService, - httpClient: &http.Client{ - Timeout: 10 * time.Second, - }, - } -} - -// Test tests a credential by making a real API call based on integration type -func (t *CredentialTester) Test(ctx context.Context, cred *models.DecryptedCredential) *models.TestCredentialResponse { - switch cred.IntegrationType { - case "discord": - return t.testDiscord(ctx, cred.Data) - case "slack": - return t.testSlack(ctx, cred.Data) - case "telegram": - return t.testTelegram(ctx, cred.Data) - case "teams": - return t.testTeams(ctx, cred.Data) - case "notion": - return t.testNotion(ctx, cred.Data) - case "github": - return t.testGitHub(ctx, cred.Data) - case "gitlab": - return t.testGitLab(ctx, cred.Data) - case "linear": - return t.testLinear(ctx, cred.Data) - case "jira": - return t.testJira(ctx, cred.Data) - case "airtable": - return t.testAirtable(ctx, cred.Data) - case "trello": - return t.testTrello(ctx, cred.Data) - case "hubspot": - return t.testHubSpot(ctx, cred.Data) - case "sendgrid": - return t.testSendGrid(ctx, cred.Data) - case "brevo": - return t.testBrevo(ctx, cred.Data) - case "mailchimp": - return t.testMailchimp(ctx, cred.Data) - case "openai": - return t.testOpenAI(ctx, cred.Data) - case "anthropic": - return t.testAnthropic(ctx, cred.Data) - case "google_ai": - return t.testGoogleAI(ctx, cred.Data) - case "google_chat": - return t.testGoogleChat(ctx, cred.Data) - case "zoom": - return t.testZoom(ctx, cred.Data) - case "referralmonk": - return t.testReferralMonk(ctx, cred.Data) - case "composio_googlesheets": - return t.testComposioGoogleSheets(ctx, cred.Data) - case "composio_gmail": - return t.testComposioGmail(ctx, cred.Data) - case "custom_webhook": - return t.testCustomWebhook(ctx, cred.Data) - case "rest_api": - return t.testRestAPI(ctx, cred.Data) - case "mongodb": - return t.testMongoDB(ctx, cred.Data) - case "redis": - return t.testRedis(ctx, cred.Data) - default: - return &models.TestCredentialResponse{ - Success: false, - Message: "Testing not implemented for this integration type", - } - } -} - -// testDiscord tests a Discord webhook by sending a test message -func (t *CredentialTester) testDiscord(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - webhookURL, ok := data["webhook_url"].(string) - if !ok || webhookURL == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Webhook URL is required", - } - } - - payload := map[string]string{ - "content": "🔗 **ClaraVerse Test** - Webhook connection verified!", - } - body, _ := json.Marshal(payload) - - req, _ := http.NewRequestWithContext(ctx, "POST", webhookURL, bytes.NewBuffer(body)) - req.Header.Set("Content-Type", "application/json") - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Discord", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode >= 200 && resp.StatusCode < 300 { - return &models.TestCredentialResponse{ - Success: true, - Message: "Discord webhook is working! A test message was sent.", - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("Discord returned status %d", resp.StatusCode), - } -} - -// testSlack tests a Slack webhook -func (t *CredentialTester) testSlack(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - webhookURL, ok := data["webhook_url"].(string) - if !ok || webhookURL == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Webhook URL is required", - } - } - - payload := map[string]string{ - "text": "🔗 *ClaraVerse Test* - Webhook connection verified!", - } - body, _ := json.Marshal(payload) - - req, _ := http.NewRequestWithContext(ctx, "POST", webhookURL, bytes.NewBuffer(body)) - req.Header.Set("Content-Type", "application/json") - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Slack", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode == 200 { - return &models.TestCredentialResponse{ - Success: true, - Message: "Slack webhook is working! A test message was sent.", - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("Slack returned status %d", resp.StatusCode), - } -} - -// testTelegram tests a Telegram bot token -func (t *CredentialTester) testTelegram(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - botToken, ok := data["bot_token"].(string) - if !ok || botToken == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Bot token is required", - } - } - - chatID, ok := data["chat_id"].(string) - if !ok || chatID == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Chat ID is required", - } - } - - url := fmt.Sprintf("https://api.telegram.org/bot%s/sendMessage", botToken) - payload := map[string]string{ - "chat_id": chatID, - "text": "🔗 ClaraVerse Test - Bot connection verified!", - } - body, _ := json.Marshal(payload) - - req, _ := http.NewRequestWithContext(ctx, "POST", url, bytes.NewBuffer(body)) - req.Header.Set("Content-Type", "application/json") - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Telegram", - Details: err.Error(), - } - } - defer resp.Body.Close() - - var result map[string]interface{} - json.NewDecoder(resp.Body).Decode(&result) - - if ok, _ := result["ok"].(bool); ok { - return &models.TestCredentialResponse{ - Success: true, - Message: "Telegram bot is working! A test message was sent.", - } - } - - description, _ := result["description"].(string) - return &models.TestCredentialResponse{ - Success: false, - Message: "Telegram API error", - Details: description, - } -} - -// testTeams tests a Microsoft Teams webhook -func (t *CredentialTester) testTeams(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - webhookURL, ok := data["webhook_url"].(string) - if !ok || webhookURL == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Webhook URL is required", - } - } - - payload := map[string]string{ - "text": "🔗 **ClaraVerse Test** - Webhook connection verified!", - } - body, _ := json.Marshal(payload) - - req, _ := http.NewRequestWithContext(ctx, "POST", webhookURL, bytes.NewBuffer(body)) - req.Header.Set("Content-Type", "application/json") - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Teams", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode == 200 { - return &models.TestCredentialResponse{ - Success: true, - Message: "Teams webhook is working! A test message was sent.", - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("Teams returned status %d", resp.StatusCode), - } -} - -// testNotion tests a Notion API key -func (t *CredentialTester) testNotion(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - apiKey, ok := data["api_key"].(string) - if !ok || apiKey == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "API key is required", - } - } - - req, _ := http.NewRequestWithContext(ctx, "GET", "https://api.notion.com/v1/users/me", nil) - req.Header.Set("Authorization", "Bearer "+apiKey) - req.Header.Set("Notion-Version", "2022-06-28") - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Notion", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode == 200 { - var result map[string]interface{} - json.NewDecoder(resp.Body).Decode(&result) - name, _ := result["name"].(string) - return &models.TestCredentialResponse{ - Success: true, - Message: "Notion API key is valid!", - Details: fmt.Sprintf("Connected as: %s", name), - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("Notion returned status %d", resp.StatusCode), - } -} - -// testGitHub tests a GitHub personal access token -func (t *CredentialTester) testGitHub(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - token, ok := data["personal_access_token"].(string) - if !ok || token == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Personal access token is required", - } - } - - req, _ := http.NewRequestWithContext(ctx, "GET", "https://api.github.com/user", nil) - req.Header.Set("Authorization", "Bearer "+token) - req.Header.Set("Accept", "application/vnd.github.v3+json") - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to GitHub", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode == 200 { - var result map[string]interface{} - json.NewDecoder(resp.Body).Decode(&result) - login, _ := result["login"].(string) - return &models.TestCredentialResponse{ - Success: true, - Message: "GitHub token is valid!", - Details: fmt.Sprintf("Authenticated as: %s", login), - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("GitHub returned status %d", resp.StatusCode), - } -} - -// testGitLab tests a GitLab personal access token -func (t *CredentialTester) testGitLab(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - token, ok := data["personal_access_token"].(string) - if !ok || token == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Personal access token is required", - } - } - - baseURL := "https://gitlab.com" - if url, ok := data["base_url"].(string); ok && url != "" { - baseURL = strings.TrimSuffix(url, "/") - } - - req, _ := http.NewRequestWithContext(ctx, "GET", baseURL+"/api/v4/user", nil) - req.Header.Set("PRIVATE-TOKEN", token) - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to GitLab", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode == 200 { - var result map[string]interface{} - json.NewDecoder(resp.Body).Decode(&result) - username, _ := result["username"].(string) - return &models.TestCredentialResponse{ - Success: true, - Message: "GitLab token is valid!", - Details: fmt.Sprintf("Authenticated as: %s", username), - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("GitLab returned status %d", resp.StatusCode), - } -} - -// testLinear tests a Linear API key -func (t *CredentialTester) testLinear(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - apiKey, ok := data["api_key"].(string) - if !ok || apiKey == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "API key is required", - } - } - - query := `{"query": "{ viewer { id name email } }"}` - req, _ := http.NewRequestWithContext(ctx, "POST", "https://api.linear.app/graphql", strings.NewReader(query)) - req.Header.Set("Authorization", apiKey) - req.Header.Set("Content-Type", "application/json") - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Linear", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode == 200 { - var result map[string]interface{} - json.NewDecoder(resp.Body).Decode(&result) - if data, ok := result["data"].(map[string]interface{}); ok { - if viewer, ok := data["viewer"].(map[string]interface{}); ok { - name, _ := viewer["name"].(string) - return &models.TestCredentialResponse{ - Success: true, - Message: "Linear API key is valid!", - Details: fmt.Sprintf("Authenticated as: %s", name), - } - } - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("Linear returned status %d", resp.StatusCode), - } -} - -// testJira tests Jira credentials -func (t *CredentialTester) testJira(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - email, _ := data["email"].(string) - apiToken, _ := data["api_token"].(string) - domain, _ := data["domain"].(string) - - if email == "" || apiToken == "" || domain == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Email, API token, and domain are required", - } - } - - url := fmt.Sprintf("https://%s/rest/api/3/myself", domain) - req, _ := http.NewRequestWithContext(ctx, "GET", url, nil) - req.SetBasicAuth(email, apiToken) - req.Header.Set("Accept", "application/json") - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Jira", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode == 200 { - var result map[string]interface{} - json.NewDecoder(resp.Body).Decode(&result) - displayName, _ := result["displayName"].(string) - return &models.TestCredentialResponse{ - Success: true, - Message: "Jira credentials are valid!", - Details: fmt.Sprintf("Authenticated as: %s", displayName), - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("Jira returned status %d", resp.StatusCode), - } -} - -// testAirtable tests an Airtable API key -func (t *CredentialTester) testAirtable(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - apiKey, ok := data["api_key"].(string) - if !ok || apiKey == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "API key is required", - } - } - - req, _ := http.NewRequestWithContext(ctx, "GET", "https://api.airtable.com/v0/meta/whoami", nil) - req.Header.Set("Authorization", "Bearer "+apiKey) - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Airtable", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode == 200 { - return &models.TestCredentialResponse{ - Success: true, - Message: "Airtable API key is valid!", - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("Airtable returned status %d", resp.StatusCode), - } -} - -// testTrello tests Trello credentials -func (t *CredentialTester) testTrello(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - apiKey, _ := data["api_key"].(string) - token, _ := data["token"].(string) - - if apiKey == "" || token == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "API key and token are required", - } - } - - url := fmt.Sprintf("https://api.trello.com/1/members/me?key=%s&token=%s", apiKey, token) - req, _ := http.NewRequestWithContext(ctx, "GET", url, nil) - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Trello", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode == 200 { - var result map[string]interface{} - json.NewDecoder(resp.Body).Decode(&result) - username, _ := result["username"].(string) - return &models.TestCredentialResponse{ - Success: true, - Message: "Trello credentials are valid!", - Details: fmt.Sprintf("Authenticated as: %s", username), - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("Trello returned status %d", resp.StatusCode), - } -} - -// testHubSpot tests a HubSpot access token -func (t *CredentialTester) testHubSpot(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - accessToken, ok := data["access_token"].(string) - if !ok || accessToken == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Access token is required", - } - } - - req, _ := http.NewRequestWithContext(ctx, "GET", "https://api.hubapi.com/crm/v3/objects/contacts?limit=1", nil) - req.Header.Set("Authorization", "Bearer "+accessToken) - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to HubSpot", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode == 200 { - return &models.TestCredentialResponse{ - Success: true, - Message: "HubSpot access token is valid!", - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("HubSpot returned status %d", resp.StatusCode), - } -} - -// testSendGrid tests a SendGrid API key and optionally verifies sender identity -func (t *CredentialTester) testSendGrid(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - apiKey, ok := data["api_key"].(string) - if !ok || apiKey == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "API key is required", - } - } - - // First, test the API key validity - req, _ := http.NewRequestWithContext(ctx, "GET", "https://api.sendgrid.com/v3/user/profile", nil) - req.Header.Set("Authorization", "Bearer "+apiKey) - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to SendGrid", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode != 200 { - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("SendGrid returned status %d - invalid API key", resp.StatusCode), - } - } - - // API key is valid - now check verified senders if from_email is provided - fromEmail, hasFromEmail := data["from_email"].(string) - if !hasFromEmail || fromEmail == "" { - return &models.TestCredentialResponse{ - Success: true, - Message: "SendGrid API key is valid!", - Details: "Note: Add a 'Default From Email' to verify sender identity.", - } - } - - // Check verified senders - sendersReq, _ := http.NewRequestWithContext(ctx, "GET", "https://api.sendgrid.com/v3/verified_senders", nil) - sendersReq.Header.Set("Authorization", "Bearer "+apiKey) - - sendersResp, err := t.httpClient.Do(sendersReq) - if err != nil { - return &models.TestCredentialResponse{ - Success: true, - Message: "SendGrid API key is valid!", - Details: fmt.Sprintf("Could not verify sender '%s' - check SendGrid Sender Identity settings.", fromEmail), - } - } - defer sendersResp.Body.Close() - - if sendersResp.StatusCode == 200 { - var result map[string]interface{} - json.NewDecoder(sendersResp.Body).Decode(&result) - - // Check if the from_email is in the verified senders list - senderVerified := false - if results, ok := result["results"].([]interface{}); ok { - for _, sender := range results { - if s, ok := sender.(map[string]interface{}); ok { - if email, ok := s["from_email"].(string); ok { - if strings.EqualFold(email, fromEmail) { - if verified, ok := s["verified"].(bool); ok && verified { - senderVerified = true - break - } - } - } - } - } - } - - if senderVerified { - return &models.TestCredentialResponse{ - Success: true, - Message: "SendGrid API key and sender identity verified!", - Details: fmt.Sprintf("Verified sender: %s", fromEmail), - } - } - - return &models.TestCredentialResponse{ - Success: true, - Message: "SendGrid API key is valid, but sender not verified!", - Details: fmt.Sprintf("'%s' is not a verified sender. Visit https://app.sendgrid.com/settings/sender_auth to verify it.", fromEmail), - } - } - - return &models.TestCredentialResponse{ - Success: true, - Message: "SendGrid API key is valid!", - Details: fmt.Sprintf("Could not check sender verification for '%s'.", fromEmail), - } -} - -// testBrevo tests a Brevo (SendInBlue) API key -func (t *CredentialTester) testBrevo(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - apiKey, ok := data["api_key"].(string) - if !ok || apiKey == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "API key is required", - } - } - - // Validate API key format - if !strings.HasPrefix(apiKey, "xkeysib-") { - return &models.TestCredentialResponse{ - Success: false, - Message: "Invalid API key format - Brevo API keys start with 'xkeysib-'", - } - } - - // Test the API key by getting account info - req, _ := http.NewRequestWithContext(ctx, "GET", "https://api.brevo.com/v3/account", nil) - req.Header.Set("api-key", apiKey) - req.Header.Set("Accept", "application/json") - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Brevo", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode != 200 { - bodyBytes, _ := io.ReadAll(resp.Body) - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("Brevo returned status %d - invalid API key", resp.StatusCode), - Details: string(bodyBytes), - } - } - - var result map[string]interface{} - json.NewDecoder(resp.Body).Decode(&result) - - // Extract account info - email, _ := result["email"].(string) - companyName := "" - if plan, ok := result["plan"].([]interface{}); ok && len(plan) > 0 { - if planInfo, ok := plan[0].(map[string]interface{}); ok { - if name, ok := planInfo["type"].(string); ok { - companyName = name - } - } - } - - details := fmt.Sprintf("Account: %s", email) - if companyName != "" { - details += fmt.Sprintf(" (Plan: %s)", companyName) - } - - // Check if from_email is provided and verify sender - fromEmail, hasFromEmail := data["from_email"].(string) - if hasFromEmail && fromEmail != "" { - // Get senders list - sendersReq, _ := http.NewRequestWithContext(ctx, "GET", "https://api.brevo.com/v3/senders", nil) - sendersReq.Header.Set("api-key", apiKey) - sendersReq.Header.Set("Accept", "application/json") - - sendersResp, err := t.httpClient.Do(sendersReq) - if err == nil { - defer sendersResp.Body.Close() - if sendersResp.StatusCode == 200 { - var sendersResult map[string]interface{} - json.NewDecoder(sendersResp.Body).Decode(&sendersResult) - - senderFound := false - if senders, ok := sendersResult["senders"].([]interface{}); ok { - for _, sender := range senders { - if s, ok := sender.(map[string]interface{}); ok { - if senderEmail, ok := s["email"].(string); ok { - if strings.EqualFold(senderEmail, fromEmail) { - senderFound = true - if active, ok := s["active"].(bool); ok && active { - details += fmt.Sprintf("\nVerified sender: %s ✓", fromEmail) - } else { - details += fmt.Sprintf("\nSender '%s' found but not active", fromEmail) - } - break - } - } - } - } - } - if !senderFound { - details += fmt.Sprintf("\nWarning: '%s' not found in verified senders", fromEmail) - } - } - } - } - - return &models.TestCredentialResponse{ - Success: true, - Message: "Brevo API key is valid!", - Details: details, - } -} - -// testMailchimp tests a Mailchimp API key -func (t *CredentialTester) testMailchimp(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - apiKey, ok := data["api_key"].(string) - if !ok || apiKey == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "API key is required", - } - } - - // Extract datacenter from API key (format: xxx-usX) - parts := strings.Split(apiKey, "-") - if len(parts) < 2 { - return &models.TestCredentialResponse{ - Success: false, - Message: "Invalid API key format", - } - } - dc := parts[len(parts)-1] - - url := fmt.Sprintf("https://%s.api.mailchimp.com/3.0/", dc) - req, _ := http.NewRequestWithContext(ctx, "GET", url, nil) - req.SetBasicAuth("anystring", apiKey) - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Mailchimp", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode == 200 { - var result map[string]interface{} - json.NewDecoder(resp.Body).Decode(&result) - accountName, _ := result["account_name"].(string) - return &models.TestCredentialResponse{ - Success: true, - Message: "Mailchimp API key is valid!", - Details: fmt.Sprintf("Account: %s", accountName), - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("Mailchimp returned status %d", resp.StatusCode), - } -} - -// testOpenAI tests an OpenAI API key -func (t *CredentialTester) testOpenAI(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - apiKey, ok := data["api_key"].(string) - if !ok || apiKey == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "API key is required", - } - } - - req, _ := http.NewRequestWithContext(ctx, "GET", "https://api.openai.com/v1/models", nil) - req.Header.Set("Authorization", "Bearer "+apiKey) - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to OpenAI", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode == 200 { - return &models.TestCredentialResponse{ - Success: true, - Message: "OpenAI API key is valid!", - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("OpenAI returned status %d", resp.StatusCode), - } -} - -// testAnthropic tests an Anthropic API key -func (t *CredentialTester) testAnthropic(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - apiKey, ok := data["api_key"].(string) - if !ok || apiKey == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "API key is required", - } - } - - // Send a minimal message to test the API key - payload := map[string]interface{}{ - "model": "claude-3-haiku-20240307", - "max_tokens": 10, - "messages": []map[string]string{ - {"role": "user", "content": "Hi"}, - }, - } - body, _ := json.Marshal(payload) - - req, _ := http.NewRequestWithContext(ctx, "POST", "https://api.anthropic.com/v1/messages", bytes.NewBuffer(body)) - req.Header.Set("x-api-key", apiKey) - req.Header.Set("anthropic-version", "2023-06-01") - req.Header.Set("Content-Type", "application/json") - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Anthropic", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode == 200 { - return &models.TestCredentialResponse{ - Success: true, - Message: "Anthropic API key is valid!", - } - } - - // Read error response - bodyBytes, _ := io.ReadAll(resp.Body) - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("Anthropic returned status %d", resp.StatusCode), - Details: string(bodyBytes), - } -} - -// testGoogleAI tests a Google AI API key -func (t *CredentialTester) testGoogleAI(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - apiKey, ok := data["api_key"].(string) - if !ok || apiKey == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "API key is required", - } - } - - url := fmt.Sprintf("https://generativelanguage.googleapis.com/v1/models?key=%s", apiKey) - req, _ := http.NewRequestWithContext(ctx, "GET", url, nil) - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Google AI", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode == 200 { - return &models.TestCredentialResponse{ - Success: true, - Message: "Google AI API key is valid!", - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("Google AI returned status %d", resp.StatusCode), - } -} - -// testGoogleChat tests a Google Chat webhook -func (t *CredentialTester) testGoogleChat(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - webhookURL, ok := data["webhook_url"].(string) - if !ok || webhookURL == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Webhook URL is required", - } - } - - // Validate it's a Google Chat webhook URL - if !strings.Contains(webhookURL, "chat.googleapis.com") { - return &models.TestCredentialResponse{ - Success: false, - Message: "Invalid Google Chat webhook URL - must contain chat.googleapis.com", - } - } - - // Google Chat webhook payload format - payload := map[string]string{ - "text": "🔗 *ClaraVerse Test* - Webhook connection verified!", - } - body, _ := json.Marshal(payload) - - req, _ := http.NewRequestWithContext(ctx, "POST", webhookURL, bytes.NewBuffer(body)) - req.Header.Set("Content-Type", "application/json; charset=UTF-8") - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Google Chat", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode >= 200 && resp.StatusCode < 300 { - return &models.TestCredentialResponse{ - Success: true, - Message: "Google Chat webhook is working! A test message was sent.", - } - } - - // Read error response for details - bodyBytes, _ := io.ReadAll(resp.Body) - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("Google Chat returned status %d", resp.StatusCode), - Details: string(bodyBytes), - } -} - -// testZoom tests Zoom Server-to-Server OAuth credentials -func (t *CredentialTester) testZoom(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - accountID, _ := data["account_id"].(string) - clientID, _ := data["client_id"].(string) - clientSecret, _ := data["client_secret"].(string) - - if accountID == "" || clientID == "" || clientSecret == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Account ID, Client ID, and Client Secret are required", - } - } - - // Try to get an OAuth access token - tokenURL := "https://zoom.us/oauth/token" - tokenData := fmt.Sprintf("grant_type=account_credentials&account_id=%s", accountID) - - req, _ := http.NewRequestWithContext(ctx, "POST", tokenURL, strings.NewReader(tokenData)) - - // Basic auth with client_id:client_secret - auth := base64.StdEncoding.EncodeToString([]byte(clientID + ":" + clientSecret)) - req.Header.Set("Authorization", "Basic "+auth) - req.Header.Set("Content-Type", "application/x-www-form-urlencoded") - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Zoom OAuth", - Details: err.Error(), - } - } - defer resp.Body.Close() - - bodyBytes, _ := io.ReadAll(resp.Body) - - if resp.StatusCode != 200 { - var errorResp map[string]interface{} - json.Unmarshal(bodyBytes, &errorResp) - errorMsg := "Unknown error" - if reason, ok := errorResp["reason"].(string); ok { - errorMsg = reason - } else if errStr, ok := errorResp["error"].(string); ok { - errorMsg = errStr - } - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("Zoom OAuth failed: %s", errorMsg), - Details: string(bodyBytes), - } - } - - var tokenResp map[string]interface{} - json.Unmarshal(bodyBytes, &tokenResp) - - if _, ok := tokenResp["access_token"].(string); !ok { - return &models.TestCredentialResponse{ - Success: false, - Message: "Zoom OAuth returned invalid response", - Details: string(bodyBytes), - } - } - - // Token obtained successfully - now verify we can list users (basic API test) - accessToken := tokenResp["access_token"].(string) - - userReq, _ := http.NewRequestWithContext(ctx, "GET", "https://api.zoom.us/v2/users/me", nil) - userReq.Header.Set("Authorization", "Bearer "+accessToken) - - userResp, err := t.httpClient.Do(userReq) - if err != nil { - return &models.TestCredentialResponse{ - Success: true, - Message: "Zoom OAuth credentials are valid!", - Details: "Token generated successfully, but could not verify API access.", - } - } - defer userResp.Body.Close() - - if userResp.StatusCode == 200 { - var userInfo map[string]interface{} - json.NewDecoder(userResp.Body).Decode(&userInfo) - email, _ := userInfo["email"].(string) - firstName, _ := userInfo["first_name"].(string) - lastName, _ := userInfo["last_name"].(string) - - name := strings.TrimSpace(firstName + " " + lastName) - if name == "" { - name = email - } - - return &models.TestCredentialResponse{ - Success: true, - Message: "Zoom credentials verified!", - Details: fmt.Sprintf("Connected as: %s (%s)", name, email), - } - } - - return &models.TestCredentialResponse{ - Success: true, - Message: "Zoom OAuth credentials are valid!", - Details: "Token generated successfully.", - } -} - -// testCustomWebhook tests a custom webhook -func (t *CredentialTester) testCustomWebhook(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - url, ok := data["url"].(string) - if !ok || url == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "URL is required", - } - } - - method := "POST" - if m, ok := data["method"].(string); ok && m != "" { - method = m - } - - payload := map[string]string{ - "test": "true", - "message": "ClaraVerse webhook test", - } - body, _ := json.Marshal(payload) - - req, _ := http.NewRequestWithContext(ctx, method, url, bytes.NewBuffer(body)) - req.Header.Set("Content-Type", "application/json") - - // Add authentication if configured - if authType, ok := data["auth_type"].(string); ok && authType != "none" { - authValue, _ := data["auth_value"].(string) - switch authType { - case "bearer": - req.Header.Set("Authorization", "Bearer "+authValue) - case "basic": - // authValue should be "user:pass" - parts := strings.SplitN(authValue, ":", 2) - if len(parts) == 2 { - req.SetBasicAuth(parts[0], parts[1]) - } - case "api_key": - req.Header.Set("X-API-Key", authValue) - } - } - - // Add custom headers - if headers, ok := data["headers"].(string); ok && headers != "" { - var headerMap map[string]string - if err := json.Unmarshal([]byte(headers), &headerMap); err == nil { - for k, v := range headerMap { - req.Header.Set(k, v) - } - } - } - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to webhook", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode >= 200 && resp.StatusCode < 300 { - return &models.TestCredentialResponse{ - Success: true, - Message: fmt.Sprintf("Webhook responded with status %d", resp.StatusCode), - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("Webhook returned status %d", resp.StatusCode), - } -} - -// testRestAPI tests a REST API endpoint -func (t *CredentialTester) testRestAPI(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - baseURL, ok := data["base_url"].(string) - if !ok || baseURL == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Base URL is required", - } - } - - req, _ := http.NewRequestWithContext(ctx, "GET", baseURL, nil) - - // Add authentication if configured - if authType, ok := data["auth_type"].(string); ok && authType != "none" { - authValue, _ := data["auth_value"].(string) - switch authType { - case "bearer": - req.Header.Set("Authorization", "Bearer "+authValue) - case "basic": - parts := strings.SplitN(authValue, ":", 2) - if len(parts) == 2 { - req.SetBasicAuth(parts[0], parts[1]) - } - case "api_key_header": - headerName := "X-API-Key" - if name, ok := data["auth_header_name"].(string); ok && name != "" { - headerName = name - } - req.Header.Set(headerName, authValue) - case "api_key_query": - q := req.URL.Query() - q.Add("api_key", authValue) - req.URL.RawQuery = q.Encode() - } - } - - // Add default headers - if headers, ok := data["headers"].(string); ok && headers != "" { - var headerMap map[string]string - if err := json.Unmarshal([]byte(headers), &headerMap); err == nil { - for k, v := range headerMap { - req.Header.Set(k, v) - } - } - } - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to API", - Details: err.Error(), - } - } - defer resp.Body.Close() - - if resp.StatusCode >= 200 && resp.StatusCode < 300 { - return &models.TestCredentialResponse{ - Success: true, - Message: fmt.Sprintf("API responded with status %d", resp.StatusCode), - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: fmt.Sprintf("API returned status %d", resp.StatusCode), - } -} - -// testMongoDB tests MongoDB connection credentials -func (t *CredentialTester) testMongoDB(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - connectionString, ok := data["connection_string"].(string) - if !ok || connectionString == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Connection string is required", - } - } - - database, _ := data["database"].(string) - if database == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Database name is required", - } - } - - // Import MongoDB driver dynamically to test connection - // We'll use a simple HTTP-based approach to avoid adding heavy dependencies - // For production, we use the actual MongoDB driver in the tools - - // Validate connection string format - if !strings.HasPrefix(connectionString, "mongodb://") && !strings.HasPrefix(connectionString, "mongodb+srv://") { - return &models.TestCredentialResponse{ - Success: false, - Message: "Invalid connection string format", - Details: "Connection string must start with 'mongodb://' or 'mongodb+srv://'", - } - } - - // Use MongoDB Go driver for actual connection test - // Import is done at runtime via the tool execution - return t.testMongoDBConnection(ctx, connectionString, database) -} - -// testMongoDBConnection performs the actual MongoDB connection test -func (t *CredentialTester) testMongoDBConnection(ctx context.Context, connectionString, database string) *models.TestCredentialResponse { - // We need to use the MongoDB driver here - // Import: go.mongodb.org/mongo-driver/mongo - // Since this is a handler and we want to keep it lightweight, - // we'll call a helper that uses the actual driver - - // For now, return a placeholder that will be replaced with actual implementation - // Using the MongoDB driver directly here - - // Create a context with timeout for the connection test - testCtx, cancel := context.WithTimeout(ctx, 10*time.Second) - defer cancel() - - // Attempt to connect using the MongoDB driver - // Note: The actual implementation requires importing the MongoDB driver - // which is already a dependency in this project (used in mongodb_tool.go) - - // We'll use a goroutine-based approach to test the connection - // without blocking the handler for too long - - resultChan := make(chan *models.TestCredentialResponse, 1) - - go func() { - result := testMongoDBWithDriver(testCtx, connectionString, database) - resultChan <- result - }() - - select { - case result := <-resultChan: - return result - case <-testCtx.Done(): - return &models.TestCredentialResponse{ - Success: false, - Message: "Connection test timed out", - Details: "MongoDB server did not respond within 10 seconds", - } - } -} - -// testRedis tests Redis connection credentials -func (t *CredentialTester) testRedis(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - host, _ := data["host"].(string) - if host == "" { - host = "localhost" - } - - port, _ := data["port"].(string) - if port == "" { - port = "6379" - } - - password, _ := data["password"].(string) - dbNum, _ := data["database"].(string) - if dbNum == "" { - dbNum = "0" - } - - // Create a context with timeout - testCtx, cancel := context.WithTimeout(ctx, 10*time.Second) - defer cancel() - - resultChan := make(chan *models.TestCredentialResponse, 1) - - go func() { - result := testRedisWithDriver(testCtx, host, port, password, dbNum) - resultChan <- result - }() - - select { - case result := <-resultChan: - return result - case <-testCtx.Done(): - return &models.TestCredentialResponse{ - Success: false, - Message: "Connection test timed out", - Details: "Redis server did not respond within 10 seconds", - } - } -} - -// testReferralMonk tests ReferralMonk API credentials by calling their API -func (t *CredentialTester) testReferralMonk(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - apiToken, ok := data["api_token"].(string) - if !ok || apiToken == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "API Token is required", - } - } - - apiSecret, ok := data["api_secret"].(string) - if !ok || apiSecret == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "API Secret is required", - } - } - - // Make a simple API call to verify credentials - using a test endpoint if available - // For now, we'll verify the credentials are properly formatted - url := "https://ahaguru.referralmonk.com/api/campaign" - - // Create a minimal test payload that won't actually send a message - // Note: This is a validation check - we're verifying the API responds to our credentials - testPayload := map[string]interface{}{ - "template_name": "test_validation", - "channel": "whatsapp", - "recipients": []map[string]interface{}{}, - } - - body, _ := json.Marshal(testPayload) - req, err := http.NewRequestWithContext(ctx, "POST", url, bytes.NewBuffer(body)) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to create request", - Details: err.Error(), - } - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Api-Token", apiToken) - req.Header.Set("Api-Secret", apiSecret) - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to ReferralMonk API", - Details: err.Error(), - } - } - defer resp.Body.Close() - - bodyBytes, _ := io.ReadAll(resp.Body) - - // Check for authentication success (even if template doesn't exist, auth should work) - // 401/403 = bad credentials - // 400/404 = credentials work but request issue (which is expected for test) - // 200/201 = success - if resp.StatusCode == 401 || resp.StatusCode == 403 { - return &models.TestCredentialResponse{ - Success: false, - Message: "Invalid API credentials", - Details: fmt.Sprintf("Authentication failed: %s", string(bodyBytes)), - } - } - - // If we get 200, 201, 400, or 404, credentials are valid (just the request format might be off) - if resp.StatusCode < 500 { - return &models.TestCredentialResponse{ - Success: true, - Message: "ReferralMonk API credentials verified successfully", - Details: "API token and secret are valid", - } - } - - // 500+ errors indicate server issues - return &models.TestCredentialResponse{ - Success: false, - Message: "ReferralMonk API server error", - Details: string(bodyBytes), - } -} - -// testComposioGoogleSheets tests Composio Google Sheets connection -func (t *CredentialTester) testComposioGoogleSheets(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - entityID, ok := data["composio_entity_id"].(string) - if !ok || entityID == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Entity ID is required", - } - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Composio integration not configured", - Details: "COMPOSIO_API_KEY environment variable not set", - } - } - - // Check if the entity has a connected Google Sheets account using v3 API - url := fmt.Sprintf("https://backend.composio.dev/api/v3/connected_accounts?user_ids=%s", entityID) - req, err := http.NewRequestWithContext(ctx, "GET", url, nil) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to create test request", - } - } - - req.Header.Set("x-api-key", composioAPIKey) - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Composio API", - Details: err.Error(), - } - } - defer resp.Body.Close() - - bodyBytes, _ := io.ReadAll(resp.Body) - - if resp.StatusCode >= 400 { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to verify Composio connection", - Details: string(bodyBytes), - } - } - - // Parse response to check for Google Sheets connection - // v3 API returns {items: [...]} - var response struct { - Items []map[string]interface{} `json:"items"` - } - if err := json.Unmarshal(bodyBytes, &response); err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to parse Composio response", - } - } - - // Check if Google Sheets is connected (v3 uses toolkit.slug) - for _, account := range response.Items { - if toolkit, ok := account["toolkit"].(map[string]interface{}); ok { - if slug, ok := toolkit["slug"].(string); ok && slug == "googlesheets" { - return &models.TestCredentialResponse{ - Success: true, - Message: "Google Sheets connected successfully via Composio", - Details: fmt.Sprintf("Entity ID: %s", entityID), - } - } - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: "No Google Sheets connection found", - Details: "Please reconnect your Google account", - } -} - -// testComposioGmail tests Composio Gmail connection -func (t *CredentialTester) testComposioGmail(ctx context.Context, data map[string]interface{}) *models.TestCredentialResponse { - entityID, ok := data["composio_entity_id"].(string) - if !ok || entityID == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Entity ID is required", - } - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return &models.TestCredentialResponse{ - Success: false, - Message: "Composio integration not configured", - Details: "COMPOSIO_API_KEY environment variable not set", - } - } - - // Check if the entity has a connected Gmail account using v3 API - url := fmt.Sprintf("https://backend.composio.dev/api/v3/connected_accounts?user_ids=%s", entityID) - req, err := http.NewRequestWithContext(ctx, "GET", url, nil) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to create test request", - } - } - - req.Header.Set("x-api-key", composioAPIKey) - - resp, err := t.httpClient.Do(req) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Composio API", - Details: err.Error(), - } - } - defer resp.Body.Close() - - bodyBytes, _ := io.ReadAll(resp.Body) - - if resp.StatusCode >= 400 { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to verify Composio connection", - Details: string(bodyBytes), - } - } - - // Parse response to check for Gmail connection - // v3 API returns {items: [...]} - var response struct { - Items []map[string]interface{} `json:"items"` - } - if err := json.Unmarshal(bodyBytes, &response); err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to parse Composio response", - } - } - - // Check if Gmail is connected (v3 uses toolkit.slug) - for _, account := range response.Items { - if toolkit, ok := account["toolkit"].(map[string]interface{}); ok { - if slug, ok := toolkit["slug"].(string); ok && slug == "gmail" { - return &models.TestCredentialResponse{ - Success: true, - Message: "Gmail connected successfully via Composio", - Details: fmt.Sprintf("Entity ID: %s", entityID), - } - } - } - } - - return &models.TestCredentialResponse{ - Success: false, - Message: "No Gmail connection found", - Details: "Please reconnect your Gmail account", - } -} diff --git a/backend/internal/handlers/database_testers.go b/backend/internal/handlers/database_testers.go deleted file mode 100644 index d7dfecca..00000000 --- a/backend/internal/handlers/database_testers.go +++ /dev/null @@ -1,246 +0,0 @@ -package handlers - -import ( - "claraverse/internal/models" - "context" - "fmt" - "strconv" - "time" - - "github.com/redis/go-redis/v9" - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/mongo" - "go.mongodb.org/mongo-driver/mongo/options" - "go.mongodb.org/mongo-driver/mongo/readpref" -) - -// testMongoDBWithDriver tests MongoDB connection using the official driver -func testMongoDBWithDriver(ctx context.Context, connectionString, database string) *models.TestCredentialResponse { - // Set client options with timeout - clientOptions := options.Client(). - ApplyURI(connectionString). - SetConnectTimeout(10 * time.Second). - SetServerSelectionTimeout(10 * time.Second) - - // Connect to MongoDB - client, err := mongo.Connect(ctx, clientOptions) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to create MongoDB client", - Details: err.Error(), - } - } - defer func() { - disconnectCtx, cancel := context.WithTimeout(context.Background(), 5*time.Second) - defer cancel() - client.Disconnect(disconnectCtx) - }() - - // Ping the database to verify connection - if err := client.Ping(ctx, readpref.Primary()); err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to MongoDB", - Details: err.Error(), - } - } - - // Try to access the specified database and list collections - db := client.Database(database) - collections, err := db.ListCollectionNames(ctx, bson.M{}) - if err != nil { - return &models.TestCredentialResponse{ - Success: false, - Message: "Connected but failed to access database", - Details: fmt.Sprintf("Database '%s': %s", database, err.Error()), - } - } - - // Get server info - var serverStatus bson.M - err = db.RunCommand(ctx, bson.D{{Key: "serverStatus", Value: 1}}).Decode(&serverStatus) - - details := fmt.Sprintf("Database: %s\nCollections: %d", database, len(collections)) - - if err == nil { - if version, ok := serverStatus["version"].(string); ok { - details = fmt.Sprintf("Server version: %s\n%s", version, details) - } - } - - if len(collections) > 0 && len(collections) <= 5 { - details += fmt.Sprintf("\nCollections: %v", collections) - } else if len(collections) > 5 { - details += fmt.Sprintf("\nCollections (first 5): %v...", collections[:5]) - } - - return &models.TestCredentialResponse{ - Success: true, - Message: "MongoDB connection successful!", - Details: details, - } -} - -// testRedisWithDriver tests Redis connection using the official driver -func testRedisWithDriver(ctx context.Context, host, port, password, dbNum string) *models.TestCredentialResponse { - // Parse database number - db, err := strconv.Atoi(dbNum) - if err != nil { - db = 0 - } - - // Create Redis client - addr := fmt.Sprintf("%s:%s", host, port) - client := redis.NewClient(&redis.Options{ - Addr: addr, - Password: password, - DB: db, - DialTimeout: 10 * time.Second, - ReadTimeout: 10 * time.Second, - WriteTimeout: 10 * time.Second, - }) - defer client.Close() - - // Test connection with PING - pong, err := client.Ping(ctx).Result() - if err != nil { - // Provide more helpful error messages - errMsg := err.Error() - if password == "" && (contains(errMsg, "NOAUTH") || contains(errMsg, "AUTH")) { - return &models.TestCredentialResponse{ - Success: false, - Message: "Redis requires authentication", - Details: "This Redis server requires a password. Please provide the password in credentials.", - } - } - if contains(errMsg, "connection refused") { - return &models.TestCredentialResponse{ - Success: false, - Message: "Connection refused", - Details: fmt.Sprintf("Could not connect to Redis at %s. Please verify the host and port are correct.", addr), - } - } - if contains(errMsg, "timeout") || contains(errMsg, "deadline") { - return &models.TestCredentialResponse{ - Success: false, - Message: "Connection timed out", - Details: fmt.Sprintf("Redis server at %s did not respond in time.", addr), - } - } - return &models.TestCredentialResponse{ - Success: false, - Message: "Failed to connect to Redis", - Details: err.Error(), - } - } - - // Get server info - info, err := client.Info(ctx, "server").Result() - details := fmt.Sprintf("Address: %s\nDatabase: %d\nPing response: %s", addr, db, pong) - - if err == nil { - // Parse server info for version - version := parseRedisInfoField(info, "redis_version") - if version != "" { - details = fmt.Sprintf("Redis version: %s\n%s", version, details) - } - - // Get memory info - memInfo, memErr := client.Info(ctx, "memory").Result() - if memErr == nil { - usedMemory := parseRedisInfoField(memInfo, "used_memory_human") - if usedMemory != "" { - details += fmt.Sprintf("\nMemory used: %s", usedMemory) - } - } - } - - // Get key count for current database - dbSize, err := client.DBSize(ctx).Result() - if err == nil { - details += fmt.Sprintf("\nKeys in DB %d: %d", db, dbSize) - } - - return &models.TestCredentialResponse{ - Success: true, - Message: "Redis connection successful!", - Details: details, - } -} - -// contains checks if a string contains a substring (case-insensitive) -func contains(s, substr string) bool { - return len(s) >= len(substr) && (s == substr || len(s) > 0 && containsIgnoreCase(s, substr)) -} - -func containsIgnoreCase(s, substr string) bool { - for i := 0; i <= len(s)-len(substr); i++ { - if equalFoldSlice(s[i:i+len(substr)], substr) { - return true - } - } - return false -} - -func equalFoldSlice(s1, s2 string) bool { - if len(s1) != len(s2) { - return false - } - for i := 0; i < len(s1); i++ { - c1, c2 := s1[i], s2[i] - if c1 >= 'A' && c1 <= 'Z' { - c1 += 'a' - 'A' - } - if c2 >= 'A' && c2 <= 'Z' { - c2 += 'a' - 'A' - } - if c1 != c2 { - return false - } - } - return true -} - -// parseRedisInfoField extracts a field value from Redis INFO output -func parseRedisInfoField(info, field string) string { - lines := splitLines(info) - prefix := field + ":" - for _, line := range lines { - if len(line) > len(prefix) && line[:len(prefix)] == prefix { - return trimSpace(line[len(prefix):]) - } - } - return "" -} - -func splitLines(s string) []string { - var lines []string - start := 0 - for i := 0; i < len(s); i++ { - if s[i] == '\n' { - line := s[start:i] - if len(line) > 0 && line[len(line)-1] == '\r' { - line = line[:len(line)-1] - } - lines = append(lines, line) - start = i + 1 - } - } - if start < len(s) { - lines = append(lines, s[start:]) - } - return lines -} - -func trimSpace(s string) string { - start := 0 - end := len(s) - for start < end && (s[start] == ' ' || s[start] == '\t' || s[start] == '\r' || s[start] == '\n') { - start++ - } - for end > start && (s[end-1] == ' ' || s[end-1] == '\t' || s[end-1] == '\r' || s[end-1] == '\n') { - end-- - } - return s[start:end] -} diff --git a/backend/internal/handlers/download.go b/backend/internal/handlers/download.go deleted file mode 100644 index 839570b5..00000000 --- a/backend/internal/handlers/download.go +++ /dev/null @@ -1,80 +0,0 @@ -package handlers - -import ( - "claraverse/internal/document" - "log" - - "github.com/gofiber/fiber/v2" -) - -// DownloadHandler handles file download requests -type DownloadHandler struct { - documentService *document.Service -} - -// NewDownloadHandler creates a new download handler -func NewDownloadHandler() *DownloadHandler { - return &DownloadHandler{ - documentService: document.GetService(), - } -} - -// Download serves a generated document and marks it for deletion -func (h *DownloadHandler) Download(c *fiber.Ctx) error { - documentID := c.Params("id") - - // Get user ID from auth middleware - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" || userID == "anonymous" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required to download documents", - }) - } - - // Get document - doc, exists := h.documentService.GetDocument(documentID) - if !exists { - log.Printf("⚠️ [DOWNLOAD] Document not found: %s (user: %s)", documentID, userID) - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Document not found or already deleted", - }) - } - - // Verify ownership - if doc.UserID != userID { - log.Printf("🚫 [SECURITY] User %s denied access to document %s (owned by %s)", - userID, documentID, doc.UserID) - return c.Status(fiber.StatusForbidden).JSON(fiber.Map{ - "error": "Access denied to this document", - }) - } - - log.Printf("📥 [DOWNLOAD] Serving document: %s (user: %s, size: %d bytes)", - doc.Filename, doc.UserID, doc.Size) - - // Determine content type - contentType := doc.ContentType - if contentType == "" { - contentType = "application/octet-stream" // Fallback for unknown types - } - - // Set headers for download - c.Set("Content-Disposition", "attachment; filename=\""+doc.Filename+"\"") - c.Set("Content-Type", contentType) - - // Send file - err := c.SendFile(doc.FilePath) - if err != nil { - log.Printf("❌ [DOWNLOAD] Failed to send file: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to download file", - }) - } - - // Mark as downloaded (will be deleted in 5 minutes by cleanup job) - h.documentService.MarkDownloaded(documentID) - - log.Printf("✅ [DOWNLOAD] Document downloaded: %s (user: %s)", doc.Filename, userID) - - return nil -} diff --git a/backend/internal/handlers/execution.go b/backend/internal/handlers/execution.go deleted file mode 100644 index d408645a..00000000 --- a/backend/internal/handlers/execution.go +++ /dev/null @@ -1,126 +0,0 @@ -package handlers - -import ( - "claraverse/internal/services" - "log" - "strconv" - - "github.com/gofiber/fiber/v2" - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// ExecutionHandler handles execution-related HTTP requests -type ExecutionHandler struct { - executionService *services.ExecutionService -} - -// NewExecutionHandler creates a new execution handler -func NewExecutionHandler(executionService *services.ExecutionService) *ExecutionHandler { - return &ExecutionHandler{ - executionService: executionService, - } -} - -// ListByAgent returns paginated executions for a specific agent -// GET /api/agents/:id/executions -func (h *ExecutionHandler) ListByAgent(c *fiber.Ctx) error { - agentID := c.Params("id") - userID := c.Locals("user_id").(string) - - opts := h.parseListOptions(c) - - result, err := h.executionService.ListByAgent(c.Context(), agentID, userID, opts) - if err != nil { - log.Printf("❌ [EXECUTION] Failed to list agent executions: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to list executions", - }) - } - - return c.JSON(result) -} - -// ListAll returns paginated executions for the current user -// GET /api/executions -func (h *ExecutionHandler) ListAll(c *fiber.Ctx) error { - userID := c.Locals("user_id").(string) - - opts := h.parseListOptions(c) - - result, err := h.executionService.ListByUser(c.Context(), userID, opts) - if err != nil { - log.Printf("❌ [EXECUTION] Failed to list user executions: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to list executions", - }) - } - - return c.JSON(result) -} - -// GetByID returns a specific execution -// GET /api/executions/:id -func (h *ExecutionHandler) GetByID(c *fiber.Ctx) error { - executionIDStr := c.Params("id") - userID := c.Locals("user_id").(string) - - executionID, err := primitive.ObjectIDFromHex(executionIDStr) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid execution ID", - }) - } - - execution, err := h.executionService.GetByIDAndUser(c.Context(), executionID, userID) - if err != nil { - if err.Error() == "execution not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Execution not found", - }) - } - log.Printf("❌ [EXECUTION] Failed to get execution: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get execution", - }) - } - - return c.JSON(execution) -} - -// GetStats returns execution statistics for an agent -// GET /api/agents/:id/executions/stats -func (h *ExecutionHandler) GetStats(c *fiber.Ctx) error { - agentID := c.Params("id") - userID := c.Locals("user_id").(string) - - stats, err := h.executionService.GetStats(c.Context(), agentID, userID) - if err != nil { - log.Printf("❌ [EXECUTION] Failed to get stats: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get execution stats", - }) - } - - return c.JSON(stats) -} - -// parseListOptions extracts pagination and filter options from query params -func (h *ExecutionHandler) parseListOptions(c *fiber.Ctx) *services.ListExecutionsOptions { - opts := &services.ListExecutionsOptions{ - Page: 1, - Limit: 20, - Status: c.Query("status"), - TriggerType: c.Query("trigger_type"), - AgentID: c.Query("agent_id"), - } - - if page, err := strconv.Atoi(c.Query("page")); err == nil && page > 0 { - opts.Page = page - } - - if limit, err := strconv.Atoi(c.Query("limit")); err == nil && limit > 0 && limit <= 100 { - opts.Limit = limit - } - - return opts -} diff --git a/backend/internal/handlers/handlers_test.go b/backend/internal/handlers/handlers_test.go deleted file mode 100644 index 8de69f91..00000000 --- a/backend/internal/handlers/handlers_test.go +++ /dev/null @@ -1,418 +0,0 @@ -package handlers - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "claraverse/internal/services" - "encoding/json" - "io" - "net/http/httptest" - "os" - "testing" - "time" - - "github.com/gofiber/fiber/v2" -) - -func setupTestApp(t *testing.T) (*fiber.App, *database.DB, func()) { - tmpFile := "test_handlers.db" - db, err := database.New(tmpFile) - if err != nil { - t.Fatalf("Failed to create test database: %v", err) - } - - if err := db.Initialize(); err != nil { - t.Fatalf("Failed to initialize test database: %v", err) - } - - app := fiber.New() - - cleanup := func() { - db.Close() - os.Remove(tmpFile) - } - - return app, db, cleanup -} - -func createTestProvider(t *testing.T, db *database.DB) *models.Provider { - providerService := services.NewProviderService(db) - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: "https://api.test.com/v1", - APIKey: "test-key", - Enabled: true, - } - - provider, err := providerService.Create(config) - if err != nil { - t.Fatalf("Failed to create test provider: %v", err) - } - - return provider -} - -func insertTestModel(t *testing.T, db *database.DB, model *models.Model) { - _, err := db.Exec(` - INSERT OR REPLACE INTO models - (id, provider_id, name, display_name, is_visible, fetched_at) - VALUES (?, ?, ?, ?, ?, ?) - `, model.ID, model.ProviderID, model.Name, model.DisplayName, model.IsVisible, time.Now()) - - if err != nil { - t.Fatalf("Failed to insert test model: %v", err) - } -} - -// TestHealthHandler tests the health check endpoint -func TestHealthHandler(t *testing.T) { - app, _, cleanup := setupTestApp(t) - defer cleanup() - - connManager := services.NewConnectionManager() - handler := NewHealthHandler(connManager) - - app.Get("/health", handler.Handle) - - req := httptest.NewRequest("GET", "/health", nil) - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Failed to send request: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != fiber.StatusOK { - t.Errorf("Expected status 200, got %d", resp.StatusCode) - } - - body, err := io.ReadAll(resp.Body) - if err != nil { - t.Fatalf("Failed to read response: %v", err) - } - - var result map[string]interface{} - if err := json.Unmarshal(body, &result); err != nil { - t.Fatalf("Failed to parse JSON: %v", err) - } - - if result["status"] != "healthy" { - t.Errorf("Expected status 'healthy', got %v", result["status"]) - } - - if result["connections"] == nil { - t.Error("Expected 'connections' field in response") - } - - if result["timestamp"] == nil { - t.Error("Expected 'timestamp' field in response") - } -} - -// TestModelHandler_List tests listing all models -func TestModelHandler_List(t *testing.T) { - app, db, cleanup := setupTestApp(t) - defer cleanup() - - modelService := services.NewModelService(db) - handler := NewModelHandler(modelService) - - app.Get("/api/models", handler.List) - - // Create test provider and models - provider := createTestProvider(t, db) - testModels := []models.Model{ - {ID: "model-1", ProviderID: provider.ID, Name: "Model 1", IsVisible: true}, - {ID: "model-2", ProviderID: provider.ID, Name: "Model 2", IsVisible: true}, - {ID: "model-3", ProviderID: provider.ID, Name: "Model 3", IsVisible: false}, - } - - for i := range testModels { - insertTestModel(t, db, &testModels[i]) - } - - // Test with default (visible only) - req := httptest.NewRequest("GET", "/api/models", nil) - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Failed to send request: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != fiber.StatusOK { - t.Errorf("Expected status 200, got %d", resp.StatusCode) - } - - body, err := io.ReadAll(resp.Body) - if err != nil { - t.Fatalf("Failed to read response: %v", err) - } - - var result map[string]interface{} - if err := json.Unmarshal(body, &result); err != nil { - t.Fatalf("Failed to parse JSON: %v", err) - } - - models, ok := result["models"].([]interface{}) - if !ok { - t.Fatal("Expected 'models' to be an array") - } - - // Should only return visible models - if len(models) != 2 { - t.Errorf("Expected 2 visible models, got %d", len(models)) - } - - count, ok := result["count"].(float64) - if !ok { - t.Fatal("Expected 'count' to be a number") - } - - if int(count) != 2 { - t.Errorf("Expected count 2, got %d", int(count)) - } -} - -// TestModelHandler_List_AllModels tests listing all models including hidden -func TestModelHandler_List_AllModels(t *testing.T) { - app, db, cleanup := setupTestApp(t) - defer cleanup() - - modelService := services.NewModelService(db) - handler := NewModelHandler(modelService) - - app.Get("/api/models", handler.List) - - // Create test provider and models - provider := createTestProvider(t, db) - testModels := []models.Model{ - {ID: "model-1", ProviderID: provider.ID, Name: "Model 1", IsVisible: true}, - {ID: "model-2", ProviderID: provider.ID, Name: "Model 2", IsVisible: false}, - } - - for i := range testModels { - insertTestModel(t, db, &testModels[i]) - } - - // Test with visible_only=false - req := httptest.NewRequest("GET", "/api/models?visible_only=false", nil) - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Failed to send request: %v", err) - } - defer resp.Body.Close() - - body, err := io.ReadAll(resp.Body) - if err != nil { - t.Fatalf("Failed to read response: %v", err) - } - - var result map[string]interface{} - if err := json.Unmarshal(body, &result); err != nil { - t.Fatalf("Failed to parse JSON: %v", err) - } - - models, ok := result["models"].([]interface{}) - if !ok { - t.Fatal("Expected 'models' to be an array") - } - - // Should return all models - if len(models) != 2 { - t.Errorf("Expected 2 models, got %d", len(models)) - } -} - -// TestModelHandler_ListByProvider tests listing models for a specific provider -func TestModelHandler_ListByProvider(t *testing.T) { - app, db, cleanup := setupTestApp(t) - defer cleanup() - - modelService := services.NewModelService(db) - handler := NewModelHandler(modelService) - - app.Get("/api/providers/:id/models", handler.ListByProvider) - - // Create test providers and models - provider1 := createTestProvider(t, db) - - providerService := services.NewProviderService(db) - provider2Config := models.ProviderConfig{ - Name: "Provider 2", - BaseURL: "https://api.provider2.com/v1", - APIKey: "test-key-2", - Enabled: true, - } - provider2, _ := providerService.Create(provider2Config) - - testModels := []models.Model{ - {ID: "model-1", ProviderID: provider1.ID, Name: "Model 1", IsVisible: true}, - {ID: "model-2", ProviderID: provider1.ID, Name: "Model 2", IsVisible: true}, - {ID: "model-3", ProviderID: provider2.ID, Name: "Model 3", IsVisible: true}, - } - - for i := range testModels { - insertTestModel(t, db, &testModels[i]) - } - - // Test provider 1 models - req := httptest.NewRequest("GET", "/api/providers/1/models", nil) - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Failed to send request: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != fiber.StatusOK { - t.Errorf("Expected status 200, got %d", resp.StatusCode) - } - - body, err := io.ReadAll(resp.Body) - if err != nil { - t.Fatalf("Failed to read response: %v", err) - } - - var result map[string]interface{} - if err := json.Unmarshal(body, &result); err != nil { - t.Fatalf("Failed to parse JSON: %v", err) - } - - models, ok := result["models"].([]interface{}) - if !ok { - t.Fatal("Expected 'models' to be an array") - } - - if len(models) != 2 { - t.Errorf("Expected 2 models for provider 1, got %d", len(models)) - } -} - -// TestModelHandler_ListByProvider_InvalidID tests with invalid provider ID -func TestModelHandler_ListByProvider_InvalidID(t *testing.T) { - app, db, cleanup := setupTestApp(t) - defer cleanup() - - modelService := services.NewModelService(db) - handler := NewModelHandler(modelService) - - app.Get("/api/providers/:id/models", handler.ListByProvider) - - // Test with invalid ID - req := httptest.NewRequest("GET", "/api/providers/invalid/models", nil) - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Failed to send request: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != fiber.StatusBadRequest { - t.Errorf("Expected status 400, got %d", resp.StatusCode) - } - - body, err := io.ReadAll(resp.Body) - if err != nil { - t.Fatalf("Failed to read response: %v", err) - } - - var result map[string]interface{} - if err := json.Unmarshal(body, &result); err != nil { - t.Fatalf("Failed to parse JSON: %v", err) - } - - if result["error"] == nil { - t.Error("Expected error message in response") - } -} - -// TestProviderHandler_List tests listing all providers -func TestProviderHandler_List(t *testing.T) { - app, db, cleanup := setupTestApp(t) - defer cleanup() - - providerService := services.NewProviderService(db) - handler := NewProviderHandler(providerService) - - app.Get("/api/providers", handler.List) - - // Create test providers - configs := []models.ProviderConfig{ - {Name: "Provider A", BaseURL: "https://a.com", APIKey: "key-a", Enabled: true}, - {Name: "Provider B", BaseURL: "https://b.com", APIKey: "key-b", Enabled: true}, - {Name: "Provider C", BaseURL: "https://c.com", APIKey: "key-c", Enabled: false}, - } - - for _, config := range configs { - providerService.Create(config) - } - - req := httptest.NewRequest("GET", "/api/providers", nil) - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Failed to send request: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != fiber.StatusOK { - t.Errorf("Expected status 200, got %d", resp.StatusCode) - } - - body, err := io.ReadAll(resp.Body) - if err != nil { - t.Fatalf("Failed to read response: %v", err) - } - - var result map[string]interface{} - if err := json.Unmarshal(body, &result); err != nil { - t.Fatalf("Failed to parse JSON: %v", err) - } - - providers, ok := result["providers"].([]interface{}) - if !ok { - t.Fatal("Expected 'providers' to be an array") - } - - // Should only return enabled providers - if len(providers) != 2 { - t.Errorf("Expected 2 enabled providers, got %d", len(providers)) - } -} - -// TestHealthHandler_WithConnections tests health endpoint with active connections -func TestHealthHandler_WithConnections(t *testing.T) { - app, _, cleanup := setupTestApp(t) - defer cleanup() - - connManager := services.NewConnectionManager() - handler := NewHealthHandler(connManager) - - app.Get("/health", handler.Handle) - - // Simulate adding connections (we can't easily test WebSocket connections here, - // so we'll just verify the endpoint works) - - req := httptest.NewRequest("GET", "/health", nil) - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Failed to send request: %v", err) - } - defer resp.Body.Close() - - body, err := io.ReadAll(resp.Body) - if err != nil { - t.Fatalf("Failed to read response: %v", err) - } - - var result map[string]interface{} - if err := json.Unmarshal(body, &result); err != nil { - t.Fatalf("Failed to parse JSON: %v", err) - } - - connections, ok := result["connections"].(float64) - if !ok { - t.Fatal("Expected 'connections' to be a number") - } - - // Should be 0 since we haven't added any - if int(connections) != 0 { - t.Errorf("Expected 0 connections, got %d", int(connections)) - } -} diff --git a/backend/internal/handlers/health.go b/backend/internal/handlers/health.go deleted file mode 100644 index e47567a4..00000000 --- a/backend/internal/handlers/health.go +++ /dev/null @@ -1,27 +0,0 @@ -package handlers - -import ( - "claraverse/internal/services" - "time" - - "github.com/gofiber/fiber/v2" -) - -// HealthHandler handles health check requests -type HealthHandler struct { - connManager *services.ConnectionManager -} - -// NewHealthHandler creates a new health handler -func NewHealthHandler(connManager *services.ConnectionManager) *HealthHandler { - return &HealthHandler{connManager: connManager} -} - -// Handle responds with server health status -func (h *HealthHandler) Handle(c *fiber.Ctx) error { - return c.JSON(fiber.Map{ - "status": "healthy", - "connections": h.connManager.Count(), - "timestamp": time.Now().Format(time.RFC3339), - }) -} diff --git a/backend/internal/handlers/image_proxy.go b/backend/internal/handlers/image_proxy.go deleted file mode 100644 index 426d47b3..00000000 --- a/backend/internal/handlers/image_proxy.go +++ /dev/null @@ -1,283 +0,0 @@ -package handlers - -import ( - "claraverse/internal/security" - "io" - "log" - "net/http" - "net/url" - "strings" - "sync" - "time" - - "github.com/gofiber/fiber/v2" -) - -// ImageProxyHandler handles image proxy requests -type ImageProxyHandler struct { - client *http.Client - cache *imageCache -} - -// imageCache provides in-memory caching for proxied images -type imageCache struct { - mu sync.RWMutex - cache map[string]*cachedImage - maxSize int64 // Max total cache size in bytes - currSize int64 // Current cache size -} - -type cachedImage struct { - data []byte - contentType string - timestamp time.Time - size int64 -} - -const ( - maxImageSize = 10 * 1024 * 1024 // 10MB max per image - maxCacheSize = 50 * 1024 * 1024 // 50MB total cache - cacheTTL = 10 * time.Minute - requestTimeout = 15 * time.Second -) - -// NewImageProxyHandler creates a new image proxy handler -func NewImageProxyHandler() *ImageProxyHandler { - return &ImageProxyHandler{ - client: &http.Client{ - Timeout: requestTimeout, - // Don't follow redirects automatically - we'll handle them - CheckRedirect: func(req *http.Request, via []*http.Request) error { - if len(via) >= 3 { - return http.ErrUseLastResponse - } - return nil - }, - }, - cache: &imageCache{ - cache: make(map[string]*cachedImage), - maxSize: maxCacheSize, - }, - } -} - -// ProxyImage handles GET /api/proxy/image?url={encoded_url} -func (h *ImageProxyHandler) ProxyImage(c *fiber.Ctx) error { - // Get URL parameter - imageURL := c.Query("url") - if imageURL == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "url parameter is required", - }) - } - - // Validate URL - parsedURL, err := url.Parse(imageURL) - if err != nil { - log.Printf("⚠️ [IMAGE-PROXY] Invalid URL: %s - %v", imageURL, err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "invalid URL format", - }) - } - - // Only allow http/https schemes - if parsedURL.Scheme != "http" && parsedURL.Scheme != "https" { - log.Printf("⚠️ [IMAGE-PROXY] Blocked non-http URL: %s", imageURL) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "only http and https URLs are allowed", - }) - } - - // SSRF protection: block requests to internal/private networks - if err := security.ValidateURLForSSRF(imageURL); err != nil { - log.Printf("🚫 [IMAGE-PROXY] SSRF blocked: %s - %v", truncateURL(imageURL), err) - return c.Status(fiber.StatusForbidden).JSON(fiber.Map{ - "error": "access to internal resources is not allowed", - }) - } - - // Check cache first - if cached := h.cache.get(imageURL); cached != nil { - log.Printf("✅ [IMAGE-PROXY] Cache hit for: %s", truncateURL(imageURL)) - c.Set("Content-Type", cached.contentType) - c.Set("Cache-Control", "public, max-age=3600") - c.Set("X-Cache", "HIT") - return c.Send(cached.data) - } - - // Fetch the image - log.Printf("🖼️ [IMAGE-PROXY] Fetching: %s", truncateURL(imageURL)) - - req, err := http.NewRequest("GET", imageURL, nil) - if err != nil { - log.Printf("❌ [IMAGE-PROXY] Failed to create request: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "failed to create request", - }) - } - - // Set headers to look like a browser - req.Header.Set("User-Agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36") - req.Header.Set("Accept", "image/avif,image/webp,image/apng,image/svg+xml,image/*,*/*;q=0.8") - req.Header.Set("Accept-Language", "en-US,en;q=0.9") - - // Set appropriate referer based on host (some sites like Bing require specific referers) - referer := parsedURL.Scheme + "://" + parsedURL.Host + "/" - host := strings.ToLower(parsedURL.Host) - if strings.Contains(host, "bing.net") || strings.Contains(host, "bing.com") { - referer = "https://www.bing.com/" - } else if strings.Contains(host, "google") { - referer = "https://www.google.com/" - } else if strings.Contains(host, "duckduckgo") { - referer = "https://duckduckgo.com/" - } - req.Header.Set("Referer", referer) - - resp, err := h.client.Do(req) - if err != nil { - log.Printf("❌ [IMAGE-PROXY] Fetch failed: %v", err) - return c.Status(fiber.StatusBadGateway).JSON(fiber.Map{ - "error": "failed to fetch image", - }) - } - defer resp.Body.Close() - - // Check response status - if resp.StatusCode != http.StatusOK { - log.Printf("⚠️ [IMAGE-PROXY] Upstream returned %d for: %s", resp.StatusCode, truncateURL(imageURL)) - return c.Status(fiber.StatusBadGateway).JSON(fiber.Map{ - "error": "upstream server returned error", - }) - } - - // Validate content type - contentType := resp.Header.Get("Content-Type") - if !isValidImageContentType(contentType) { - log.Printf("⚠️ [IMAGE-PROXY] Invalid content type: %s for: %s", contentType, truncateURL(imageURL)) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "URL does not point to a valid image", - }) - } - - // Read body with size limit - limitedReader := io.LimitReader(resp.Body, maxImageSize+1) - data, err := io.ReadAll(limitedReader) - if err != nil { - log.Printf("❌ [IMAGE-PROXY] Failed to read response: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "failed to read image data", - }) - } - - // Check if image exceeds size limit - if int64(len(data)) > maxImageSize { - log.Printf("⚠️ [IMAGE-PROXY] Image too large: %d bytes for: %s", len(data), truncateURL(imageURL)) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "image exceeds maximum allowed size (10MB)", - }) - } - - // Cache the image - h.cache.set(imageURL, data, contentType) - - log.Printf("✅ [IMAGE-PROXY] Served: %s (%d bytes)", truncateURL(imageURL), len(data)) - - // Send response - c.Set("Content-Type", contentType) - c.Set("Cache-Control", "public, max-age=3600") - c.Set("X-Cache", "MISS") - return c.Send(data) -} - -// isValidImageContentType checks if the content type is a valid image type -func isValidImageContentType(contentType string) bool { - contentType = strings.ToLower(contentType) - validTypes := []string{ - "image/jpeg", - "image/jpg", - "image/png", - "image/gif", - "image/webp", - "image/svg+xml", - "image/avif", - "image/bmp", - "image/tiff", - } - - for _, vt := range validTypes { - if strings.HasPrefix(contentType, vt) { - return true - } - } - return false -} - -// truncateURL truncates URL for logging -func truncateURL(u string) string { - if len(u) > 80 { - return u[:77] + "..." - } - return u -} - -// Cache methods - -func (c *imageCache) get(key string) *cachedImage { - c.mu.RLock() - defer c.mu.RUnlock() - - cached, exists := c.cache[key] - if !exists { - return nil - } - - // Check TTL - if time.Since(cached.timestamp) > cacheTTL { - return nil - } - - return cached -} - -func (c *imageCache) set(key string, data []byte, contentType string) { - c.mu.Lock() - defer c.mu.Unlock() - - size := int64(len(data)) - - // If this single image is too large, don't cache it - if size > c.maxSize/2 { - return - } - - // Evict old entries if needed - for c.currSize+size > c.maxSize && len(c.cache) > 0 { - c.evictOldest() - } - - // Store in cache - c.cache[key] = &cachedImage{ - data: data, - contentType: contentType, - timestamp: time.Now(), - size: size, - } - c.currSize += size -} - -func (c *imageCache) evictOldest() { - var oldestKey string - var oldestTime time.Time - - for k, v := range c.cache { - if oldestKey == "" || v.timestamp.Before(oldestTime) { - oldestKey = k - oldestTime = v.timestamp - } - } - - if oldestKey != "" { - c.currSize -= c.cache[oldestKey].size - delete(c.cache, oldestKey) - } -} diff --git a/backend/internal/handlers/mcp_websocket.go b/backend/internal/handlers/mcp_websocket.go deleted file mode 100644 index 6dbbf28a..00000000 --- a/backend/internal/handlers/mcp_websocket.go +++ /dev/null @@ -1,201 +0,0 @@ -package handlers - -import ( - "encoding/json" - "fmt" - "log" - "time" - - "claraverse/internal/models" - "claraverse/internal/services" - "github.com/gofiber/contrib/websocket" - "github.com/gofiber/fiber/v2" -) - -// MCPWebSocketHandler handles MCP client WebSocket connections -type MCPWebSocketHandler struct { - mcpService *services.MCPBridgeService -} - -// NewMCPWebSocketHandler creates a new MCP WebSocket handler -func NewMCPWebSocketHandler(mcpService *services.MCPBridgeService) *MCPWebSocketHandler { - return &MCPWebSocketHandler{ - mcpService: mcpService, - } -} - -// HandleConnection handles incoming MCP client WebSocket connections -func (h *MCPWebSocketHandler) HandleConnection(c *websocket.Conn) { - // Get user from fiber context (set by auth middleware) - userID := c.Locals("user_id").(string) - - if userID == "" || userID == "anonymous" { - log.Printf("❌ MCP connection rejected: no authenticated user") - c.WriteJSON(fiber.Map{ - "type": "error", - "payload": map[string]interface{}{ - "message": "Authentication required", - }, - }) - c.Close() - return - } - - log.Printf("🔌 MCP client connecting: user=%s", userID) - - var mcpConn *models.MCPConnection - var clientID string - - // Read loop - for { - var msg models.MCPClientMessage - err := c.ReadJSON(&msg) - if err != nil { - if mcpConn != nil { - log.Printf("MCP client disconnected: %v", err) - h.mcpService.DisconnectClient(clientID) - } - break - } - - switch msg.Type { - case "register_tools": - // Parse registration payload - regData, err := json.Marshal(msg.Payload) - if err != nil { - log.Printf("Failed to marshal registration payload: %v", err) - continue - } - - var registration models.MCPToolRegistration - err = json.Unmarshal(regData, ®istration) - if err != nil { - log.Printf("Failed to unmarshal registration: %v", err) - c.WriteJSON(models.MCPServerMessage{ - Type: "error", - Payload: map[string]interface{}{ - "message": "Invalid registration format", - }, - }) - continue - } - - // Register client - conn, err := h.mcpService.RegisterClient(userID, ®istration) - if err != nil { - log.Printf("Failed to register MCP client: %v", err) - c.WriteJSON(models.MCPServerMessage{ - Type: "error", - Payload: map[string]interface{}{ - "message": fmt.Sprintf("Registration failed: %v", err), - }, - }) - continue - } - - mcpConn = conn - clientID = registration.ClientID - - // Start write loop - go h.writeLoop(c, conn) - - log.Printf("✅ MCP client registered successfully: user=%s, client=%s", userID, clientID) - - case "tool_result": - // Handle tool execution result - resultData, err := json.Marshal(msg.Payload) - if err != nil { - log.Printf("Failed to marshal tool result: %v", err) - continue - } - - var result models.MCPToolResult - err = json.Unmarshal(resultData, &result) - if err != nil { - log.Printf("Failed to unmarshal tool result: %v", err) - continue - } - - // Log execution for audit - execTime := 0 // We don't track this yet, but could add it - h.mcpService.LogToolExecution(userID, "", "", execTime, result.Success, result.Error) - - log.Printf("Tool result received: call_id=%s, success=%v", result.CallID, result.Success) - - // Forward result to pending result channel - if conn, exists := h.mcpService.GetConnection(clientID); exists { - if resultChan, pending := conn.PendingResults[result.CallID]; pending { - // Non-blocking send to result channel - select { - case resultChan <- result: - log.Printf("✅ Tool result forwarded to waiting channel: %s", result.CallID) - default: - log.Printf("⚠️ Result channel full or closed for call_id: %s", result.CallID) - } - } else { - log.Printf("⚠️ No pending result channel for call_id: %s", result.CallID) - } - } - - case "heartbeat": - // Update heartbeat - if clientID != "" { - err := h.mcpService.UpdateHeartbeat(clientID) - if err != nil { - log.Printf("Failed to update heartbeat: %v", err) - } - } - - case "disconnect": - // Client is gracefully disconnecting - if clientID != "" { - h.mcpService.DisconnectClient(clientID) - } - c.Close() - return - - default: - log.Printf("Unknown message type from MCP client: %s", msg.Type) - c.WriteJSON(models.MCPServerMessage{ - Type: "error", - Payload: map[string]interface{}{ - "message": "Unknown message type", - }, - }) - } - } -} - -// writeLoop handles outgoing messages to the MCP client -func (h *MCPWebSocketHandler) writeLoop(c *websocket.Conn, conn *models.MCPConnection) { - ticker := time.NewTicker(30 * time.Second) - defer ticker.Stop() - - for { - select { - case msg, ok := <-conn.WriteChan: - if !ok { - // Channel closed - return - } - - err := c.WriteJSON(msg) - if err != nil { - log.Printf("Failed to write message to MCP client: %v", err) - return - } - - case <-conn.StopChan: - // Stop signal received - return - - case <-ticker.C: - // Send ping to keep connection alive - err := c.WriteMessage(websocket.PingMessage, []byte{}) - if err != nil { - log.Printf("Failed to send ping to MCP client: %v", err) - return - } - } - } -} diff --git a/backend/internal/handlers/memory_handler.go b/backend/internal/handlers/memory_handler.go deleted file mode 100644 index 217953ba..00000000 --- a/backend/internal/handlers/memory_handler.go +++ /dev/null @@ -1,578 +0,0 @@ -package handlers - -import ( - "context" - "log" - "regexp" - "strconv" - "strings" - "time" - - "claraverse/internal/models" - "claraverse/internal/services" - - "github.com/gofiber/fiber/v2" - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// MemoryHandler handles memory-related API endpoints -type MemoryHandler struct { - memoryStorageService *services.MemoryStorageService - memoryExtractionService *services.MemoryExtractionService - chatService *services.ChatService -} - -// NewMemoryHandler creates a new memory handler -func NewMemoryHandler( - memoryStorageService *services.MemoryStorageService, - memoryExtractionService *services.MemoryExtractionService, - chatService *services.ChatService, -) *MemoryHandler { - return &MemoryHandler{ - memoryStorageService: memoryStorageService, - memoryExtractionService: memoryExtractionService, - chatService: chatService, - } -} - -// ListMemories returns paginated list of memories with optional filters -// GET /api/v1/memories?category=preferences&tags=ui,theme&includeArchived=false&page=1&pageSize=20 -func (h *MemoryHandler) ListMemories(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - // Parse query parameters - category := c.Query("category", "") - tagsParam := c.Query("tags", "") - includeArchived := c.Query("includeArchived", "false") == "true" - page, _ := strconv.Atoi(c.Query("page", "1")) - pageSize, _ := strconv.Atoi(c.Query("pageSize", "20")) - - // Validate pagination - if page < 1 { - page = 1 - } - if pageSize < 1 || pageSize > 100 { - pageSize = 20 - } - - // Parse and sanitize tags (SECURITY: prevent NoSQL injection) - var tags []string - if tagsParam != "" { - // Split by comma - rawTags := strings.Split(tagsParam, ",") - for _, rawTag := range rawTags { - sanitizedTag := sanitizeTag(strings.TrimSpace(rawTag)) - if sanitizedTag != "" && len(sanitizedTag) <= 50 { - tags = append(tags, sanitizedTag) - } - } - - // Limit number of tags to prevent abuse - if len(tags) > 20 { - tags = tags[:20] - } - } - - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - // Get memories - memories, total, err := h.memoryStorageService.ListMemories( - ctx, - userID, - category, - tags, - includeArchived, - page, - pageSize, - ) - if err != nil { - log.Printf("❌ [MEMORY-API] Failed to list memories: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to retrieve memories", - }) - } - - // Build response - memoryResponses := make([]fiber.Map, len(memories)) - for i, mem := range memories { - memoryResponses[i] = buildMemoryResponse(mem) - } - - return c.JSON(fiber.Map{ - "memories": memoryResponses, - "pagination": fiber.Map{ - "page": page, - "page_size": pageSize, - "total": total, - "total_pages": (total + int64(pageSize) - 1) / int64(pageSize), - }, - }) -} - -// GetMemory returns a single memory by ID -// GET /api/v1/memories/:id -func (h *MemoryHandler) GetMemory(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - memoryIDParam := c.Params("id") - - memoryID, err := primitive.ObjectIDFromHex(memoryIDParam) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid memory ID", - }) - } - - ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) - defer cancel() - - memory, err := h.memoryStorageService.GetMemory(ctx, userID, memoryID) - if err != nil { - log.Printf("❌ [MEMORY-API] Failed to get memory: %v", err) - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Memory not found", - }) - } - - return c.JSON(buildMemoryResponse(*memory)) -} - -// CreateMemory creates a new manual memory -// POST /api/v1/memories -type CreateMemoryRequest struct { - Content string `json:"content"` - Category string `json:"category"` - Tags []string `json:"tags"` -} - -func (h *MemoryHandler) CreateMemory(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - var req CreateMemoryRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // Validate request (SECURITY: input validation) - const MaxMemoryContentLength = 10000 // 10KB per memory - const MaxTagCount = 20 - const MaxTagLength = 50 - - if req.Content == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Content is required", - }) - } - - if len(req.Content) > MaxMemoryContentLength { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Content must be less than 10,000 characters", - }) - } - - if req.Category == "" { - req.Category = "context" - } - - // Sanitize and validate tags - if len(req.Tags) > MaxTagCount { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Maximum 20 tags allowed", - }) - } - - sanitizedTags := make([]string, 0, len(req.Tags)) - for _, tag := range req.Tags { - sanitized := sanitizeTag(strings.TrimSpace(tag)) - if sanitized != "" && len(sanitized) <= MaxTagLength { - sanitizedTags = append(sanitizedTags, sanitized) - } - } - req.Tags = sanitizedTags - - // Validate category - validCategories := map[string]bool{ - "personal_info": true, - "preferences": true, - "context": true, - "fact": true, - "instruction": true, - } - if !validCategories[req.Category] { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid category. Must be one of: personal_info, preferences, context, fact, instruction", - }) - } - - if req.Tags == nil { - req.Tags = []string{} - } - - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - // Create memory with default engagement score of 0.5 (manually created) - memory, err := h.memoryStorageService.CreateMemory( - ctx, - userID, - req.Content, - req.Category, - req.Tags, - 0.5, // Default engagement for manual memories - "", // No conversation ID for manual memories - ) - if err != nil { - log.Printf("❌ [MEMORY-API] Failed to create memory: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to create memory", - }) - } - - // Decrypt for response - decryptedMemory, err := h.memoryStorageService.GetMemory(ctx, userID, memory.ID) - if err != nil { - log.Printf("❌ [MEMORY-API] Failed to decrypt memory: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Memory created but failed to retrieve", - }) - } - - return c.Status(fiber.StatusCreated).JSON(buildMemoryResponse(*decryptedMemory)) -} - -// UpdateMemory updates an existing memory -// PUT /api/v1/memories/:id -type UpdateMemoryRequest struct { - Content *string `json:"content,omitempty"` - Category *string `json:"category,omitempty"` - Tags *[]string `json:"tags,omitempty"` -} - -func (h *MemoryHandler) UpdateMemory(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - memoryIDParam := c.Params("id") - - memoryID, err := primitive.ObjectIDFromHex(memoryIDParam) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid memory ID", - }) - } - - var req UpdateMemoryRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - // Get existing memory to retrieve current values - existingMemory, err := h.memoryStorageService.GetMemory(ctx, userID, memoryID) - if err != nil { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Memory not found", - }) - } - - // Validate and sanitize input - const MaxMemoryContentLength = 10000 - const MaxTagCount = 20 - const MaxTagLength = 50 - - // Update fields - content := existingMemory.DecryptedContent - if req.Content != nil { - if len(*req.Content) > MaxMemoryContentLength { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Content must be less than 10,000 characters", - }) - } - content = *req.Content - } - - category := existingMemory.Category - if req.Category != nil { - category = *req.Category - // Validate category - validCategories := map[string]bool{ - "personal_info": true, - "preferences": true, - "context": true, - "fact": true, - "instruction": true, - } - if !validCategories[category] { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid category", - }) - } - } - - tags := existingMemory.Tags - if req.Tags != nil { - if len(*req.Tags) > MaxTagCount { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Maximum 20 tags allowed", - }) - } - - // Sanitize tags - sanitizedTags := make([]string, 0, len(*req.Tags)) - for _, tag := range *req.Tags { - sanitized := sanitizeTag(strings.TrimSpace(tag)) - if sanitized != "" && len(sanitized) <= MaxTagLength { - sanitizedTags = append(sanitizedTags, sanitized) - } - } - tags = sanitizedTags - } - - // SECURITY FIX: Use atomic update instead of delete-create to prevent race conditions - updatedMemory, err := h.memoryStorageService.UpdateMemoryInPlace( - ctx, - userID, - memoryID, - content, - category, - tags, - existingMemory.SourceEngagement, - existingMemory.ConversationID, - ) - if err != nil { - log.Printf("❌ [MEMORY-API] Failed to update memory: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to update memory", - }) - } - - // Get decrypted version for response - decryptedMemory, err := h.memoryStorageService.GetMemory(ctx, userID, updatedMemory.ID) - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Memory updated but failed to retrieve", - }) - } - - return c.JSON(buildMemoryResponse(*decryptedMemory)) -} - -// DeleteMemory permanently deletes a memory -// DELETE /api/v1/memories/:id -func (h *MemoryHandler) DeleteMemory(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - memoryIDParam := c.Params("id") - - memoryID, err := primitive.ObjectIDFromHex(memoryIDParam) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid memory ID", - }) - } - - ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) - defer cancel() - - err = h.memoryStorageService.DeleteMemory(ctx, userID, memoryID) - if err != nil { - log.Printf("❌ [MEMORY-API] Failed to delete memory: %v", err) - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Memory not found or already deleted", - }) - } - - return c.JSON(fiber.Map{ - "success": true, - "message": "Memory deleted successfully", - }) -} - -// ArchiveMemory archives a memory -// POST /api/v1/memories/:id/archive -func (h *MemoryHandler) ArchiveMemory(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - memoryIDParam := c.Params("id") - - memoryID, err := primitive.ObjectIDFromHex(memoryIDParam) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid memory ID", - }) - } - - ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) - defer cancel() - - err = h.memoryStorageService.ArchiveMemory(ctx, userID, memoryID) - if err != nil { - log.Printf("❌ [MEMORY-API] Failed to archive memory: %v", err) - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Memory not found", - }) - } - - return c.JSON(fiber.Map{ - "success": true, - "message": "Memory archived successfully", - }) -} - -// UnarchiveMemory restores an archived memory -// POST /api/v1/memories/:id/unarchive -func (h *MemoryHandler) UnarchiveMemory(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - memoryIDParam := c.Params("id") - - memoryID, err := primitive.ObjectIDFromHex(memoryIDParam) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid memory ID", - }) - } - - ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) - defer cancel() - - err = h.memoryStorageService.UnarchiveMemory(ctx, userID, memoryID) - if err != nil { - log.Printf("❌ [MEMORY-API] Failed to unarchive memory: %v", err) - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Memory not found", - }) - } - - return c.JSON(fiber.Map{ - "success": true, - "message": "Memory restored successfully", - }) -} - -// GetMemoryStats returns statistics about user's memories -// GET /api/v1/memories/stats -func (h *MemoryHandler) GetMemoryStats(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) - defer cancel() - - stats, err := h.memoryStorageService.GetMemoryStats(ctx, userID) - if err != nil { - log.Printf("❌ [MEMORY-API] Failed to get memory stats: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to retrieve memory statistics", - }) - } - - return c.JSON(stats) -} - -// TriggerMemoryExtraction manually triggers memory extraction for a conversation -// POST /api/v1/conversations/:id/extract-memories -func (h *MemoryHandler) TriggerMemoryExtraction(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - conversationID := c.Params("id") - - ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second) - defer cancel() - - // Get conversation messages - messages := h.chatService.GetConversationMessages(conversationID) - if len(messages) == 0 { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Conversation not found or has no messages", - }) - } - - // Enqueue extraction job - err := h.memoryExtractionService.EnqueueExtraction(ctx, userID, conversationID, messages) - if err != nil { - log.Printf("❌ [MEMORY-API] Failed to trigger extraction: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to trigger memory extraction", - }) - } - - return c.JSON(fiber.Map{ - "success": true, - "message": "Memory extraction queued successfully", - }) -} - -// buildMemoryResponse creates a response object from a decrypted memory -func buildMemoryResponse(mem models.DecryptedMemory) fiber.Map { - return fiber.Map{ - "id": mem.ID.Hex(), - "content": mem.DecryptedContent, - "category": mem.Category, - "tags": mem.Tags, - "score": mem.Score, - "access_count": mem.AccessCount, - "last_accessed_at": mem.LastAccessedAt, - "is_archived": mem.IsArchived, - "archived_at": mem.ArchivedAt, - "source_engagement": mem.SourceEngagement, - "conversation_id": mem.ConversationID, - "created_at": mem.CreatedAt, - "updated_at": mem.UpdatedAt, - "version": mem.Version, - } -} - -// sanitizeTag removes potentially dangerous characters from tags -// SECURITY: Prevents NoSQL injection via tag parameters -func sanitizeTag(tag string) string { - // Only allow alphanumeric characters, hyphens, and underscores - // This prevents MongoDB operators like $where, $regex, etc. - reg := regexp.MustCompile(`[^a-zA-Z0-9_-]`) - sanitized := reg.ReplaceAllString(tag, "") - return sanitized -} diff --git a/backend/internal/handlers/model.go b/backend/internal/handlers/model.go deleted file mode 100644 index 6adb9135..00000000 --- a/backend/internal/handlers/model.go +++ /dev/null @@ -1,313 +0,0 @@ -package handlers - -import ( - "claraverse/internal/models" - "claraverse/internal/services" - "fmt" - "log" - "strconv" - "strings" - - "github.com/gofiber/fiber/v2" -) - -// aliasKey creates a composite key for tracking aliased models by provider+name -func aliasKey(providerID int, modelName string) string { - return fmt.Sprintf("%d:%s", providerID, strings.ToLower(modelName)) -} - -// ModelHandler handles model-related requests -type ModelHandler struct { - modelService *services.ModelService -} - -// NewModelHandler creates a new model handler -func NewModelHandler(modelService *services.ModelService) *ModelHandler { - return &ModelHandler{modelService: modelService} -} - -// List returns all available models -func (h *ModelHandler) List(c *fiber.Ctx) error { - // Check if we should only return visible models - visibleOnly := c.Query("visible_only", "true") == "true" - - modelsList, err := h.modelService.GetAll(visibleOnly) - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to fetch models", - }) - } - - // Get config service for alias information - configService := services.GetConfigService() - - // Build a map of model alias -> global recommendation tier with full tier info - // Query global tiers from recommended_models table joined with tier_labels - type TierInfo struct { - Tier string - Label string - Description string - Icon string - } - modelRecommendationTier := make(map[string]TierInfo) // model_alias -> tier info - - rows, err := h.modelService.GetDB().Query(` - SELECT r.tier, r.model_alias, r.provider_id, - t.label, t.description, t.icon - FROM recommended_models r - JOIN tier_labels t ON r.tier = t.tier - `) - if err != nil { - log.Printf("⚠️ [MODEL-HANDLER] Failed to load global tiers: %v", err) - } else { - defer rows.Close() - for rows.Next() { - var tier, modelAlias, label, description, icon string - var providerID int - if err := rows.Scan(&tier, &modelAlias, &providerID, &label, &description, &icon); err != nil { - log.Printf("⚠️ [MODEL-HANDLER] Failed to scan tier row: %v", err) - continue - } - // Use lowercase alias for case-insensitive matching - key := fmt.Sprintf("%d:%s", providerID, strings.ToLower(modelAlias)) - modelRecommendationTier[key] = TierInfo{ - Tier: tier, - Label: label, - Description: description, - Icon: icon, - } - log.Printf("🎯 [MODEL-HANDLER] Loaded global tier: %s -> %s (%s - %s) (provider %d)", modelAlias, tier, label, icon, providerID) - } - } - - // Build enriched models with alias structure - // If a model has an alias, we expose ONLY the alias to the frontend - enrichedModels := make([]interface{}, 0) - aliasedModels := make(map[string]bool) // Track which original models have been aliased (key: providerID:lowercase_name) - - // First pass: Add all aliased models - allAliases := configService.GetAllModelAliases() - for providerID, aliases := range allAliases { - for aliasName, aliasInfo := range aliases { - // Find the original model to get its capabilities - // Use case-insensitive comparison for model name OR ID matching - var foundModel *models.Model - actualModelLower := strings.ToLower(aliasInfo.ActualModel) - for i := range modelsList { - if modelsList[i].ProviderID == providerID && - (strings.ToLower(modelsList[i].Name) == actualModelLower || - strings.ToLower(modelsList[i].ID) == actualModelLower) { - foundModel = &modelsList[i] - // Use composite key (providerID:lowercase_name) to track aliased models - aliasedModels[aliasKey(providerID, modelsList[i].Name)] = true - aliasedModels[aliasKey(providerID, modelsList[i].ID)] = true - break - } - } - - if foundModel == nil { - log.Printf("⚠️ [MODEL-ALIAS] Could not find model '%s' for alias '%s' (provider %d)", aliasInfo.ActualModel, aliasName, providerID) - continue - } - - // Determine supports_vision: use alias override if set, otherwise use model's value - supportsVision := foundModel.SupportsVision - if aliasInfo.SupportsVision != nil { - supportsVision = *aliasInfo.SupportsVision - } - - // Determine agents_enabled: use alias Agents flag if set, otherwise default to true - agentsEnabled := true // Default to true (all models available for agents) - if aliasInfo.Agents != nil { - agentsEnabled = *aliasInfo.Agents - } - - // Get provider security status - isProviderSecure := configService.IsProviderSecure(providerID) - - // Create model entry using alias as the ID - modelMap := map[string]interface{}{ - "id": aliasName, // Alias name becomes the ID - "provider_id": providerID, - "provider_name": foundModel.ProviderName, - "name": aliasInfo.DisplayName, // Use alias display name - "display_name": aliasInfo.DisplayName, // Use alias display name - "supports_tools": foundModel.SupportsTools, - "supports_streaming": foundModel.SupportsStreaming, - "supports_vision": supportsVision, - "agents_enabled": agentsEnabled, - "provider_secure": isProviderSecure, - "is_visible": foundModel.IsVisible, - "fetched_at": foundModel.FetchedAt, - } - - // Check if this model (by alias name) is in the recommendation tier - recommendationKey := fmt.Sprintf("%d:%s", providerID, strings.ToLower(aliasName)) - var tierDescription string - if tierInfo, exists := modelRecommendationTier[recommendationKey]; exists { - modelMap["recommendation_tier"] = map[string]interface{}{ - "tier": tierInfo.Tier, - "label": tierInfo.Label, - "description": tierInfo.Description, - "icon": tierInfo.Icon, - } - tierDescription = tierInfo.Description - log.Printf("✅ [MODEL-HANDLER] Added tier '%s' (%s %s) to alias '%s'", tierInfo.Tier, tierInfo.Icon, tierInfo.Label, aliasName) - } - - // Add description - use tier description as fallback if model description is empty - if aliasInfo.Description != "" { - modelMap["description"] = aliasInfo.Description - } else if tierDescription != "" { - modelMap["description"] = tierDescription - } - - if foundModel.ProviderFavicon != "" { - modelMap["provider_favicon"] = foundModel.ProviderFavicon - } - if aliasInfo.StructuredOutputSupport != "" { - modelMap["structured_output_support"] = aliasInfo.StructuredOutputSupport - } - if aliasInfo.StructuredOutputCompliance != nil { - modelMap["structured_output_compliance"] = *aliasInfo.StructuredOutputCompliance - } - if aliasInfo.StructuredOutputWarning != "" { - modelMap["structured_output_warning"] = aliasInfo.StructuredOutputWarning - } - if aliasInfo.StructuredOutputSpeedMs != nil { - modelMap["structured_output_speed_ms"] = *aliasInfo.StructuredOutputSpeedMs - } - if aliasInfo.StructuredOutputBadge != "" { - modelMap["structured_output_badge"] = aliasInfo.StructuredOutputBadge - } - - enrichedModels = append(enrichedModels, modelMap) - } - } - - // Second pass: Add non-aliased models - for _, model := range modelsList { - // Use composite key (providerID:lowercase_name) to check if model is aliased - if !aliasedModels[aliasKey(model.ProviderID, model.Name)] { - // Get provider security status - isProviderSecure := configService.IsProviderSecure(model.ProviderID) - - modelMap := map[string]interface{}{ - "id": model.ID, - "provider_id": model.ProviderID, - "provider_name": model.ProviderName, - "name": model.Name, - "display_name": model.DisplayName, - "supports_tools": model.SupportsTools, - "supports_streaming": model.SupportsStreaming, - "supports_vision": model.SupportsVision, - "agents_enabled": model.AgentsEnabled, // Use model's AgentsEnabled field (defaults to false for non-aliased) - "provider_secure": isProviderSecure, - "is_visible": model.IsVisible, - "fetched_at": model.FetchedAt, - } - // Check if this model is in the recommendation tier - recommendationKey := fmt.Sprintf("%d:%s", model.ProviderID, strings.ToLower(model.ID)) - var tierDescription string - if tierInfo, exists := modelRecommendationTier[recommendationKey]; exists { - modelMap["recommendation_tier"] = map[string]interface{}{ - "tier": tierInfo.Tier, - "label": tierInfo.Label, - "description": tierInfo.Description, - "icon": tierInfo.Icon, - } - tierDescription = tierInfo.Description - log.Printf("✅ [MODEL-HANDLER] Added tier '%s' (%s %s) to model '%s'", tierInfo.Tier, tierInfo.Icon, tierInfo.Label, model.ID) - } - - // Add description - use tier description as fallback if model description is empty - if tierDescription != "" { - modelMap["description"] = tierDescription - } - - // Add optional provider favicon if present - if model.ProviderFavicon != "" { - modelMap["provider_favicon"] = model.ProviderFavicon - } - - enrichedModels = append(enrichedModels, modelMap) - } - } - - // Check user authentication status - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - userID = "anonymous" - } - - // Filter models based on authentication - if userID == "anonymous" { - // Anonymous users only get free tier models - var filteredModels []interface{} - for _, model := range enrichedModels { - if modelMap, ok := model.(map[string]interface{}); ok { - modelID, _ := modelMap["id"].(string) - - // Check if this model is marked as free tier - if h.modelService.IsFreeTier(modelID) { - filteredModels = append(filteredModels, model) - } - } - } - - log.Printf("🔒 Anonymous user - filtered to %d free tier models", len(filteredModels)) - return c.JSON(fiber.Map{ - "models": filteredModels, - "count": len(filteredModels), - "tier": "anonymous", - }) - } - - // Authenticated users get all models - log.Printf("✅ Authenticated user (%s) - showing all %d models", userID, len(enrichedModels)) - return c.JSON(fiber.Map{ - "models": enrichedModels, - "count": len(enrichedModels), - "tier": "authenticated", - }) -} - -// ListByProvider returns models for a specific provider -func (h *ModelHandler) ListByProvider(c *fiber.Ctx) error { - providerID, err := strconv.Atoi(c.Params("id")) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid provider ID", - }) - } - - visibleOnly := c.Query("visible_only", "true") == "true" - - models, err := h.modelService.GetByProvider(providerID, visibleOnly) - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to fetch models", - }) - } - - return c.JSON(fiber.Map{ - "models": models, - "count": len(models), - }) -} - -// ListToolPredictorModels returns only models that can be used as tool predictors -// GET /api/models/tool-predictors -func (h *ModelHandler) ListToolPredictorModels(c *fiber.Ctx) error { - models, err := h.modelService.GetToolPredictorModels() - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to fetch tool predictor models", - }) - } - - return c.JSON(fiber.Map{ - "models": models, - "count": len(models), - }) -} diff --git a/backend/internal/handlers/model_management.go b/backend/internal/handlers/model_management.go deleted file mode 100644 index 0a83c9c9..00000000 --- a/backend/internal/handlers/model_management.go +++ /dev/null @@ -1,787 +0,0 @@ -package handlers - -import ( - "claraverse/internal/services" - "fmt" - "log" - "net/url" - "strconv" - - "github.com/gofiber/fiber/v2" -) - -// ModelManagementHandler handles model management operations for admin -type ModelManagementHandler struct { - modelMgmtService *services.ModelManagementService - modelService *services.ModelService - providerService *services.ProviderService -} - -// Helper function to decode URL-encoded model IDs (handles slashes and special characters) -func decodeModelID(encodedID string) (string, error) { - return url.QueryUnescape(encodedID) -} - -// NewModelManagementHandler creates a new model management handler -func NewModelManagementHandler( - modelMgmtService *services.ModelManagementService, - modelService *services.ModelService, - providerService *services.ProviderService, -) *ModelManagementHandler { - return &ModelManagementHandler{ - modelMgmtService: modelMgmtService, - modelService: modelService, - providerService: providerService, - } -} - -// ================== MODEL CRUD ENDPOINTS ================== - -// GetAllModels returns all models with metadata -// GET /api/admin/models -func (h *ModelManagementHandler) GetAllModels(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - log.Printf("🔍 Admin %s fetching all models", adminUserID) - - // Get all models (including hidden ones) - models, err := h.modelService.GetAll(false) - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to fetch models", - }) - } - - return c.JSON(models) -} - -// CreateModel creates a new model manually -// POST /api/admin/models -func (h *ModelManagementHandler) CreateModel(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - - var req services.CreateModelRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // Validate required fields - if req.ModelID == "" || req.ProviderID == 0 || req.Name == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "model_id, provider_id, and name are required", - }) - } - - log.Printf("📝 Admin %s creating model: %s (provider %d)", adminUserID, req.ModelID, req.ProviderID) - - model, err := h.modelMgmtService.CreateModel(c.Context(), &req) - if err != nil { - log.Printf("❌ Failed to create model: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to create model: " + err.Error(), - }) - } - - return c.Status(fiber.StatusCreated).JSON(model) -} - -// UpdateModel updates an existing model's metadata -// PUT /api/admin/models/:modelId -func (h *ModelManagementHandler) UpdateModel(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - encodedModelID := c.Params("modelId") - - if encodedModelID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Model ID is required", - }) - } - - // Decode URL-encoded model ID - modelID, err := decodeModelID(encodedModelID) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid model ID encoding", - }) - } - - var req services.UpdateModelRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // WORKAROUND: Fiber's BodyParser doesn't handle *bool correctly when value is false - // Manually parse boolean fields from raw JSON if present - var rawBody map[string]interface{} - if err := c.BodyParser(&rawBody); err == nil { - log.Printf("[DEBUG] Raw request body: %+v", rawBody) - - if val, exists := rawBody["is_visible"]; exists { - if boolVal, ok := val.(bool); ok { - req.IsVisible = &boolVal - log.Printf("[DEBUG] Manually parsed is_visible: %v", boolVal) - } - } - - if val, exists := rawBody["smart_tool_router"]; exists { - if boolVal, ok := val.(bool); ok { - req.SmartToolRouter = &boolVal - log.Printf("[DEBUG] Manually parsed smart_tool_router: %v", boolVal) - } - } - - if val, exists := rawBody["supports_tools"]; exists { - if boolVal, ok := val.(bool); ok { - req.SupportsTools = &boolVal - log.Printf("[DEBUG] Manually parsed supports_tools: %v", boolVal) - } - } - - if val, exists := rawBody["supports_vision"]; exists { - if boolVal, ok := val.(bool); ok { - req.SupportsVision = &boolVal - log.Printf("[DEBUG] Manually parsed supports_vision: %v", boolVal) - } - } - - if val, exists := rawBody["supports_streaming"]; exists { - if boolVal, ok := val.(bool); ok { - req.SupportsStreaming = &boolVal - log.Printf("[DEBUG] Manually parsed supports_streaming: %v", boolVal) - } - } - - if val, exists := rawBody["free_tier"]; exists { - if boolVal, ok := val.(bool); ok { - req.FreeTier = &boolVal - log.Printf("[DEBUG] Manually parsed free_tier: %v", boolVal) - } - } - } - - log.Printf("📝 Admin %s updating model: %s", adminUserID, modelID) - - model, err := h.modelMgmtService.UpdateModel(c.Context(), modelID, &req) - if err != nil { - log.Printf("❌ Failed to update model: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to update model: " + err.Error(), - }) - } - - return c.JSON(model) -} - -// DeleteModel deletes a model -// DELETE /api/admin/models/:modelId -func (h *ModelManagementHandler) DeleteModel(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - encodedModelID := c.Params("modelId") - - if encodedModelID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Model ID is required", - }) - } - - // Decode URL-encoded model ID - modelID, err := decodeModelID(encodedModelID) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid model ID encoding", - }) - } - - log.Printf("🗑️ Admin %s deleting model: %s", adminUserID, modelID) - - if err := h.modelMgmtService.DeleteModel(c.Context(), modelID); err != nil { - log.Printf("❌ Failed to delete model: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to delete model: " + err.Error(), - }) - } - - return c.JSON(fiber.Map{ - "success": true, - "message": "Model deleted successfully", - }) -} - -// ================== MODEL FETCHING ENDPOINTS ================== - -// FetchModelsFromProvider fetches models from a provider's API -// POST /api/admin/providers/:providerId/fetch -func (h *ModelManagementHandler) FetchModelsFromProvider(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - providerIDStr := c.Params("providerId") - - providerID, err := strconv.Atoi(providerIDStr) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid provider ID", - }) - } - - log.Printf("🔄 Admin %s fetching models from provider %d", adminUserID, providerID) - - count, err := h.modelMgmtService.FetchModelsFromProvider(c.Context(), providerID) - if err != nil { - log.Printf("❌ Failed to fetch models: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to fetch models: " + err.Error(), - }) - } - - return c.JSON(fiber.Map{ - "success": true, - "models_fetched": count, - "message": "Models fetched and stored successfully", - }) -} - -// SyncProviderToJSON forces sync of a provider's models to providers.json -// POST /api/admin/providers/:providerId/sync -func (h *ModelManagementHandler) SyncProviderToJSON(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - providerIDStr := c.Params("providerId") - - providerID, err := strconv.Atoi(providerIDStr) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid provider ID", - }) - } - - log.Printf("🔄 Admin %s syncing provider %d to JSON", adminUserID, providerID) - - // The sync is handled automatically by the service, just trigger it - _, err = h.modelMgmtService.FetchModelsFromProvider(c.Context(), providerID) - if err != nil { - log.Printf("❌ Failed to sync provider: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to sync provider: " + err.Error(), - }) - } - - return c.JSON(fiber.Map{ - "success": true, - "message": "Provider synced to providers.json successfully", - }) -} - -// ================== MODEL TESTING ENDPOINTS ================== - -// TestModelConnection tests basic connection to a model -// POST /api/admin/models/:modelId/test/connection -func (h *ModelManagementHandler) TestModelConnection(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - encodedModelID := c.Params("modelId") - - if encodedModelID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Model ID is required", - }) - } - - // Decode URL-encoded model ID - modelID, err := decodeModelID(encodedModelID) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid model ID encoding", - }) - } - - log.Printf("🔌 Admin %s testing connection for model: %s", adminUserID, modelID) - - result, err := h.modelMgmtService.TestModelConnection(c.Context(), modelID) - if err != nil { - log.Printf("❌ Connection test failed: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Connection test failed: " + err.Error(), - }) - } - - if result.Passed { - return c.JSON(fiber.Map{ - "success": true, - "passed": result.Passed, - "latency_ms": result.LatencyMs, - "message": "Connection test passed", - }) - } - - return c.JSON(fiber.Map{ - "success": false, - "passed": result.Passed, - "latency_ms": result.LatencyMs, - "error": result.Error, - "message": "Connection test failed", - }) -} - -// TestModelCapability tests model capabilities (tools, vision, streaming) -// POST /api/admin/models/:modelId/test/capability -func (h *ModelManagementHandler) TestModelCapability(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - encodedModelID := c.Params("modelId") - - if encodedModelID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Model ID is required", - }) - } - - // Decode URL-encoded model ID - modelID, err := decodeModelID(encodedModelID) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid model ID encoding", - }) - } - - log.Printf("🧪 Admin %s testing capabilities for model: %s", adminUserID, modelID) - - // TODO: Implement capability testing - // This would test tools, vision, streaming support - - return c.JSON(fiber.Map{ - "message": "Capability testing not yet implemented", - }) -} - -// RunModelBenchmark runs comprehensive benchmark suite on a model -// POST /api/admin/models/:modelId/benchmark -func (h *ModelManagementHandler) RunModelBenchmark(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - encodedModelID := c.Params("modelId") - - if encodedModelID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Model ID is required", - }) - } - - // Decode URL-encoded model ID - modelID, err := decodeModelID(encodedModelID) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid model ID encoding", - }) - } - - // URL decode the model ID (handles IDs with slashes) - decodedModelID, err := url.QueryUnescape(modelID) - if err != nil { - log.Printf("❌ Failed to decode model ID: %v", err) - decodedModelID = modelID // Fallback to original - } - - log.Printf("📊 Admin %s running benchmark for model: %s (decoded: %s)", adminUserID, modelID, decodedModelID) - - results, err := h.modelMgmtService.RunBenchmark(c.Context(), decodedModelID) - if err != nil { - log.Printf("❌ Benchmark failed for model %s: %v", decodedModelID, err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Benchmark failed: " + err.Error(), - }) - } - - log.Printf("✅ Benchmark completed for model %s. Results: connection=%v, structured=%v, performance=%v", - decodedModelID, - results.ConnectionTest != nil, - results.StructuredOutput != nil, - results.Performance != nil) - - return c.JSON(results) -} - -// GetModelTestResults retrieves latest test results for a model -// GET /api/admin/models/:modelId/test-results -func (h *ModelManagementHandler) GetModelTestResults(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - encodedModelID := c.Params("modelId") - - if encodedModelID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Model ID is required", - }) - } - - // Decode URL-encoded model ID - modelID, err := decodeModelID(encodedModelID) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid model ID encoding", - }) - } - - log.Printf("🔍 Admin %s fetching test results for model: %s", adminUserID, modelID) - - // TODO: Query model_capabilities table for test results - - return c.JSON(fiber.Map{ - "message": "Test results retrieval not yet implemented", - }) -} - -// ================== ALIAS MANAGEMENT ENDPOINTS ================== - -// GetModelAliases retrieves all aliases for a model -// GET /api/admin/models/:modelId/aliases -func (h *ModelManagementHandler) GetModelAliases(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - encodedModelID := c.Params("modelId") - - if encodedModelID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Model ID is required", - }) - } - - // Decode URL-encoded model ID - modelID, err := decodeModelID(encodedModelID) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid model ID encoding", - }) - } - - // URL decode the model ID (handles IDs with slashes) - decodedModelID, err := url.QueryUnescape(modelID) - if err != nil { - log.Printf("❌ Failed to decode model ID: %v", err) - decodedModelID = modelID // Fallback to original - } - - log.Printf("🔍 Admin %s fetching aliases for model: %s (decoded: %s)", adminUserID, modelID, decodedModelID) - - aliases, err := h.modelMgmtService.GetAliases(c.Context(), decodedModelID) - if err != nil { - log.Printf("❌ Failed to get aliases: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get aliases: " + err.Error(), - }) - } - - log.Printf("✅ Returning %d aliases for model %s", len(aliases), decodedModelID) - return c.JSON(aliases) -} - -// CreateModelAlias creates a new alias for a model -// POST /api/admin/models/:modelId/aliases -func (h *ModelManagementHandler) CreateModelAlias(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - encodedModelID := c.Params("modelId") - - if encodedModelID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Model ID is required", - }) - } - - // Decode URL-encoded model ID - modelID, err := decodeModelID(encodedModelID) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid model ID encoding", - }) - } - - // URL decode the model ID (handles IDs with slashes) - decodedModelID, err := url.QueryUnescape(modelID) - if err != nil { - log.Printf("❌ Failed to decode model ID: %v", err) - decodedModelID = modelID // Fallback to original - } - - var req services.CreateAliasRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // Set model ID from URL parameter - req.ModelID = decodedModelID - - // Validate required fields - if req.AliasName == "" || req.ProviderID == 0 { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "alias_name and provider_id are required", - }) - } - - log.Printf("📝 Admin %s creating alias %s for model: %s", adminUserID, req.AliasName, modelID) - - if err := h.modelMgmtService.CreateAlias(c.Context(), &req); err != nil { - log.Printf("❌ Failed to create alias: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to create alias: " + err.Error(), - }) - } - - return c.Status(fiber.StatusCreated).JSON(fiber.Map{ - "success": true, - "message": "Alias created successfully", - }) -} - -// UpdateModelAlias updates an existing alias -// PUT /api/admin/models/:modelId/aliases/:alias -func (h *ModelManagementHandler) UpdateModelAlias(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - modelID := c.Params("modelId") - aliasName := c.Params("alias") - - if modelID == "" || aliasName == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Model ID and alias name are required", - }) - } - - log.Printf("📝 Admin %s updating alias %s for model: %s", adminUserID, aliasName, modelID) - - // TODO: Implement alias update - - return c.JSON(fiber.Map{ - "message": "Alias update not yet implemented", - }) -} - -// DeleteModelAlias deletes an alias -// DELETE /api/admin/models/:modelId/aliases/:alias -func (h *ModelManagementHandler) DeleteModelAlias(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - modelID := c.Params("modelId") - aliasName := c.Params("alias") - - if modelID == "" || aliasName == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Model ID and alias name are required", - }) - } - - // Get provider ID from query parameter or request body - providerIDStr := c.Query("provider_id") - if providerIDStr == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "provider_id query parameter is required", - }) - } - - providerID, err := strconv.Atoi(providerIDStr) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid provider ID", - }) - } - - log.Printf("🗑️ Admin %s deleting alias %s for model: %s", adminUserID, aliasName, modelID) - - if err := h.modelMgmtService.DeleteAlias(c.Context(), aliasName, providerID); err != nil { - log.Printf("❌ Failed to delete alias: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to delete alias: " + err.Error(), - }) - } - - return c.JSON(fiber.Map{ - "success": true, - "message": "Alias deleted successfully", - }) -} - -// ImportAliasesFromJSON imports all aliases from providers.json into the database -// POST /api/admin/models/import-aliases -func (h *ModelManagementHandler) ImportAliasesFromJSON(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - - log.Printf("📥 Admin %s triggering alias import from providers.json", adminUserID) - - if err := h.modelMgmtService.ImportAliasesFromJSON(c.Context()); err != nil { - log.Printf("❌ Failed to import aliases: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to import aliases: " + err.Error(), - }) - } - - return c.JSON(fiber.Map{ - "success": true, - "message": "Aliases imported successfully from providers.json", - }) -} - -// ================== BULK OPERATIONS ENDPOINTS ================== - -// BulkUpdateAgentsEnabled bulk enables/disables models for agent builder -// PUT /api/admin/models/bulk/agents-enabled -func (h *ModelManagementHandler) BulkUpdateAgentsEnabled(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - - var req struct { - ModelIDs []string `json:"model_ids"` - Enabled bool `json:"enabled"` - } - - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if len(req.ModelIDs) == 0 { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "model_ids array is required", - }) - } - - log.Printf("📝 Admin %s bulk updating agents_enabled for %d models", adminUserID, len(req.ModelIDs)) - - if err := h.modelMgmtService.BulkUpdateAgentsEnabled(req.ModelIDs, req.Enabled); err != nil { - log.Printf("❌ Failed to bulk update agents_enabled: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": fmt.Sprintf("Failed to update models: %v", err), - }) - } - - return c.JSON(fiber.Map{ - "message": fmt.Sprintf("Updated agents_enabled=%v for %d models", req.Enabled, len(req.ModelIDs)), - }) -} - -// BulkUpdateVisibility bulk shows/hides models -// PUT /api/admin/models/bulk/visibility -func (h *ModelManagementHandler) BulkUpdateVisibility(c *fiber.Ctx) error { - adminUserID := c.Locals("user_id").(string) - - var req struct { - ModelIDs []string `json:"model_ids"` - Visible bool `json:"visible"` - } - - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if len(req.ModelIDs) == 0 { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "model_ids array is required", - }) - } - - log.Printf("📝 Admin %s bulk updating visibility for %d models", adminUserID, len(req.ModelIDs)) - - if err := h.modelMgmtService.BulkUpdateVisibility(req.ModelIDs, req.Visible); err != nil { - log.Printf("❌ Failed to bulk update visibility: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": fmt.Sprintf("Failed to update models: %v", err), - }) - } - - return c.JSON(fiber.Map{ - "message": fmt.Sprintf("Updated is_visible=%v for %d models", req.Visible, len(req.ModelIDs)), - }) -} - -// ================== GLOBAL TIER MANAGEMENT ================== - -// SetModelTier assigns a model to a global tier (tier1-tier5) -// POST /api/admin/models/:modelId/tier -func (h *ModelManagementHandler) SetModelTier(c *fiber.Ctx) error { - modelID := c.Params("modelId") - if modelID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Model ID is required", - }) - } - - // URL decode the model ID (handles slashes and other special characters) - decodedModelID, err := url.QueryUnescape(modelID) - if err != nil { - log.Printf("❌ Failed to decode model ID '%s': %v", modelID, err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid model ID encoding", - }) - } - modelID = decodedModelID - - var req struct { - ProviderID int `json:"provider_id"` - Tier string `json:"tier"` // "tier1", "tier2", "tier3", "tier4", "tier5" - } - - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if req.Tier == "" || req.ProviderID == 0 { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "tier and provider_id are required", - }) - } - - if err := h.modelMgmtService.SetGlobalTier(modelID, req.ProviderID, req.Tier); err != nil { - log.Printf("❌ Failed to set tier: %v", err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": fmt.Sprintf("Failed to set tier: %v", err), - }) - } - - return c.JSON(fiber.Map{ - "message": fmt.Sprintf("Model assigned to %s", req.Tier), - }) -} - -// ClearModelTier removes a model from its tier -// DELETE /api/admin/models/:modelId/tier -func (h *ModelManagementHandler) ClearModelTier(c *fiber.Ctx) error { - var req struct { - Tier string `json:"tier"` - } - - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if req.Tier == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "tier is required", - }) - } - - if err := h.modelMgmtService.ClearTier(req.Tier); err != nil { - log.Printf("❌ Failed to clear tier: %v", err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": fmt.Sprintf("Failed to clear tier: %v", err), - }) - } - - return c.JSON(fiber.Map{ - "message": fmt.Sprintf("Tier %s cleared", req.Tier), - }) -} - -// GetTiers retrieves all global tier assignments -// GET /api/admin/tiers -func (h *ModelManagementHandler) GetTiers(c *fiber.Ctx) error { - tiers, err := h.modelMgmtService.GetGlobalTiers() - if err != nil { - log.Printf("❌ Failed to get tiers: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to retrieve tiers", - }) - } - - return c.JSON(fiber.Map{ - "tiers": tiers, - }) -} diff --git a/backend/internal/handlers/provider.go b/backend/internal/handlers/provider.go deleted file mode 100644 index 998025c4..00000000 --- a/backend/internal/handlers/provider.go +++ /dev/null @@ -1,67 +0,0 @@ -package handlers - -import ( - "claraverse/internal/services" - "strconv" - - "github.com/gofiber/fiber/v2" -) - -// ProviderHandler handles provider-related requests -type ProviderHandler struct { - providerService *services.ProviderService -} - -// NewProviderHandler creates a new provider handler -func NewProviderHandler(providerService *services.ProviderService) *ProviderHandler { - return &ProviderHandler{providerService: providerService} -} - -// List returns all enabled providers (names only, no credentials) -func (h *ProviderHandler) List(c *fiber.Ctx) error { - providers, err := h.providerService.GetAll() - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to fetch providers", - }) - } - - // Hide sensitive information (API keys, base URLs) - type PublicProvider struct { - ID int `json:"id"` - Name string `json:"name"` - } - - publicProviders := make([]PublicProvider, len(providers)) - for i, p := range providers { - publicProviders[i] = PublicProvider{ - ID: p.ID, - Name: p.Name, - } - } - - return c.JSON(fiber.Map{ - "providers": publicProviders, - "count": len(publicProviders), - }) -} - -// GetModels returns models for a specific provider -func (h *ProviderHandler) GetModels(c *fiber.Ctx) error { - providerID, err := strconv.Atoi(c.Params("id")) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid provider ID", - }) - } - - // Get provider to verify it exists - _, err = h.providerService.GetByID(providerID) - if err != nil { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Provider not found", - }) - } - - return c.Next() -} diff --git a/backend/internal/handlers/schedule.go b/backend/internal/handlers/schedule.go deleted file mode 100644 index 52db26ad..00000000 --- a/backend/internal/handlers/schedule.go +++ /dev/null @@ -1,298 +0,0 @@ -package handlers - -import ( - "claraverse/internal/models" - "claraverse/internal/services" - "log" - - "github.com/gofiber/fiber/v2" -) - -// ScheduleHandler handles schedule-related HTTP requests -type ScheduleHandler struct { - schedulerService *services.SchedulerService - agentService *services.AgentService -} - -// NewScheduleHandler creates a new schedule handler -func NewScheduleHandler(schedulerService *services.SchedulerService, agentService *services.AgentService) *ScheduleHandler { - return &ScheduleHandler{ - schedulerService: schedulerService, - agentService: agentService, - } -} - -// Create creates a new schedule for an agent -// POST /api/agents/:id/schedule -func (h *ScheduleHandler) Create(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - // Verify agent exists and belongs to user - agent, err := h.agentService.GetAgent(agentID, userID) - if err != nil { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Agent not found", - }) - } - - // Check if workflow has file inputs (which expire in 30 minutes) - if hasFileInputs(agent.Workflow) { - log.Printf("🚫 [SCHEDULE] Cannot schedule agent %s: workflow has file inputs", agentID) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Cannot schedule workflows with file inputs", - "reason": "Uploaded files expire after 30 minutes and won't be available at scheduled execution time", - "suggestion": "Use the API trigger endpoint instead: POST /api/trigger/" + agentID, - }) - } - - var req models.CreateScheduleRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // Validate required fields - if req.CronExpression == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "cronExpression is required", - }) - } - if req.Timezone == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "timezone is required", - }) - } - - log.Printf("📅 [SCHEDULE] Creating schedule for agent %s (user: %s, cron: %s)", agentID, userID, req.CronExpression) - - schedule, err := h.schedulerService.CreateSchedule(c.Context(), agentID, userID, &req) - if err != nil { - log.Printf("❌ [SCHEDULE] Failed to create schedule: %v", err) - - // Check for specific errors - if err.Error() == "agent already has a schedule" { - return c.Status(fiber.StatusConflict).JSON(fiber.Map{ - "error": err.Error(), - }) - } - if err.Error()[:14] == "schedule limit" { - return c.Status(fiber.StatusForbidden).JSON(fiber.Map{ - "error": err.Error(), - }) - } - - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": err.Error(), - }) - } - - log.Printf("✅ [SCHEDULE] Created schedule %s for agent %s", schedule.ID.Hex(), agentID) - return c.Status(fiber.StatusCreated).JSON(schedule.ToResponse()) -} - -// Get retrieves the schedule for an agent -// GET /api/agents/:id/schedule -func (h *ScheduleHandler) Get(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - schedule, err := h.schedulerService.GetScheduleByAgentID(c.Context(), agentID, userID) - if err != nil { - if err.Error() == "schedule not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "No schedule found for this agent", - }) - } - log.Printf("❌ [SCHEDULE] Failed to get schedule: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get schedule", - }) - } - - return c.JSON(schedule.ToResponse()) -} - -// Update updates the schedule for an agent -// PUT /api/agents/:id/schedule -func (h *ScheduleHandler) Update(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - // Get existing schedule - existingSchedule, err := h.schedulerService.GetScheduleByAgentID(c.Context(), agentID, userID) - if err != nil { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "No schedule found for this agent", - }) - } - - var req models.UpdateScheduleRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - log.Printf("📝 [SCHEDULE] Updating schedule %s for agent %s", existingSchedule.ID.Hex(), agentID) - - schedule, err := h.schedulerService.UpdateSchedule(c.Context(), existingSchedule.ID.Hex(), userID, &req) - if err != nil { - log.Printf("❌ [SCHEDULE] Failed to update schedule: %v", err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": err.Error(), - }) - } - - log.Printf("✅ [SCHEDULE] Updated schedule %s", schedule.ID.Hex()) - return c.JSON(schedule.ToResponse()) -} - -// Delete deletes the schedule for an agent -// DELETE /api/agents/:id/schedule -func (h *ScheduleHandler) Delete(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - // Get existing schedule - existingSchedule, err := h.schedulerService.GetScheduleByAgentID(c.Context(), agentID, userID) - if err != nil { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "No schedule found for this agent", - }) - } - - log.Printf("🗑️ [SCHEDULE] Deleting schedule %s for agent %s", existingSchedule.ID.Hex(), agentID) - - if err := h.schedulerService.DeleteSchedule(c.Context(), existingSchedule.ID.Hex(), userID); err != nil { - log.Printf("❌ [SCHEDULE] Failed to delete schedule: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to delete schedule", - }) - } - - log.Printf("✅ [SCHEDULE] Deleted schedule for agent %s", agentID) - return c.Status(fiber.StatusNoContent).Send(nil) -} - -// TriggerNow triggers an immediate execution of the schedule -// POST /api/agents/:id/schedule/run -func (h *ScheduleHandler) TriggerNow(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - agentID := c.Params("id") - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent ID is required", - }) - } - - // Get existing schedule - existingSchedule, err := h.schedulerService.GetScheduleByAgentID(c.Context(), agentID, userID) - if err != nil { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "No schedule found for this agent", - }) - } - - log.Printf("▶️ [SCHEDULE] Triggering immediate run for schedule %s (agent: %s)", existingSchedule.ID.Hex(), agentID) - - if err := h.schedulerService.TriggerNow(c.Context(), existingSchedule.ID.Hex(), userID); err != nil { - log.Printf("❌ [SCHEDULE] Failed to trigger schedule: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to trigger schedule", - }) - } - - return c.JSON(fiber.Map{ - "message": "Schedule triggered successfully", - }) -} - -// GetUsage returns the user's schedule usage stats -// GET /api/schedules/usage -func (h *ScheduleHandler) GetUsage(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - usage, err := h.schedulerService.GetScheduleUsage(c.Context(), userID) - if err != nil { - log.Printf("❌ [SCHEDULE] Failed to get schedule usage: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get schedule usage", - }) - } - - return c.JSON(usage) -} - -// hasFileInputs checks if a workflow has any variable blocks with file input type -// File inputs cannot be scheduled because uploaded files expire after 30 minutes -func hasFileInputs(workflow *models.Workflow) bool { - if workflow == nil { - return false - } - for _, block := range workflow.Blocks { - if block.Type == "variable" { - // Config is map[string]any - if inputType, exists := block.Config["inputType"]; exists && inputType == "file" { - return true - } - } - } - return false -} diff --git a/backend/internal/handlers/secure_download.go b/backend/internal/handlers/secure_download.go deleted file mode 100644 index 2b10ada3..00000000 --- a/backend/internal/handlers/secure_download.go +++ /dev/null @@ -1,181 +0,0 @@ -package handlers - -import ( - "claraverse/internal/securefile" - "fmt" - "log" - - "github.com/gofiber/fiber/v2" -) - -// SecureDownloadHandler handles secure file downloads with access codes -type SecureDownloadHandler struct { - secureFileService *securefile.Service -} - -// NewSecureDownloadHandler creates a new secure download handler -func NewSecureDownloadHandler() *SecureDownloadHandler { - return &SecureDownloadHandler{ - secureFileService: securefile.GetService(), - } -} - -// Download handles file downloads with access code validation -// GET /api/files/:id?code=ACCESS_CODE -func (h *SecureDownloadHandler) Download(c *fiber.Ctx) error { - fileID := c.Params("id") - accessCode := c.Query("code") - - if fileID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "file_id is required", - }) - } - - if accessCode == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "access code is required", - }) - } - - log.Printf("📥 [SECURE-DOWNLOAD] Download request for file %s", fileID) - - // Get file with access code verification - file, content, err := h.secureFileService.GetFile(fileID, accessCode) - if err != nil { - log.Printf("❌ [SECURE-DOWNLOAD] Failed to get file %s: %v", fileID, err) - - if err.Error() == "invalid access code" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "invalid access code", - }) - } - - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "file not found or expired", - }) - } - - // Set headers for download - c.Set("Content-Type", file.MimeType) - c.Set("Content-Disposition", fmt.Sprintf("attachment; filename=\"%s\"", file.Filename)) - c.Set("Content-Length", fmt.Sprintf("%d", file.Size)) - c.Set("X-File-ID", file.ID) - c.Set("X-Expires-At", file.ExpiresAt.Format("2006-01-02T15:04:05Z07:00")) - - log.Printf("✅ [SECURE-DOWNLOAD] Serving file %s (%s, %d bytes)", file.ID, file.Filename, file.Size) - - return c.Send(content) -} - -// GetInfo returns file metadata without downloading -// GET /api/files/:id/info?code=ACCESS_CODE -func (h *SecureDownloadHandler) GetInfo(c *fiber.Ctx) error { - fileID := c.Params("id") - accessCode := c.Query("code") - - if fileID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "file_id is required", - }) - } - - if accessCode == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "access code is required", - }) - } - - file, err := h.secureFileService.GetFileInfo(fileID, accessCode) - if err != nil { - if err.Error() == "invalid access code" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "invalid access code", - }) - } - - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "file not found or expired", - }) - } - - return c.JSON(fiber.Map{ - "id": file.ID, - "filename": file.Filename, - "mime_type": file.MimeType, - "size": file.Size, - "created_at": file.CreatedAt, - "expires_at": file.ExpiresAt, - }) -} - -// Delete removes a file (requires authentication and ownership) -// DELETE /api/files/:id -func (h *SecureDownloadHandler) Delete(c *fiber.Ctx) error { - fileID := c.Params("id") - userID := c.Locals("user_id") - - if fileID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "file_id is required", - }) - } - - if userID == nil { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "authentication required", - }) - } - - err := h.secureFileService.DeleteFile(fileID, userID.(string)) - if err != nil { - if err.Error() == "access denied" { - return c.Status(fiber.StatusForbidden).JSON(fiber.Map{ - "error": "you don't have permission to delete this file", - }) - } - - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "file not found", - }) - } - - log.Printf("✅ [SECURE-DOWNLOAD] File %s deleted by user %s", fileID, userID) - - return c.JSON(fiber.Map{ - "success": true, - "message": "file deleted", - }) -} - -// ListUserFiles returns all files for the authenticated user -// GET /api/files -func (h *SecureDownloadHandler) ListUserFiles(c *fiber.Ctx) error { - userID := c.Locals("user_id") - - if userID == nil { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "authentication required", - }) - } - - files := h.secureFileService.ListUserFiles(userID.(string)) - - // Convert to response format (without sensitive data) - response := make([]fiber.Map, 0, len(files)) - for _, file := range files { - response = append(response, fiber.Map{ - "id": file.ID, - "filename": file.Filename, - "mime_type": file.MimeType, - "size": file.Size, - "created_at": file.CreatedAt, - "expires_at": file.ExpiresAt, - }) - } - - return c.JSON(fiber.Map{ - "files": response, - "count": len(response), - }) -} diff --git a/backend/internal/handlers/subscription.go b/backend/internal/handlers/subscription.go deleted file mode 100644 index 6d6e2620..00000000 --- a/backend/internal/handlers/subscription.go +++ /dev/null @@ -1,363 +0,0 @@ -package handlers - -import ( - "claraverse/internal/services" - "context" - "log" - - "github.com/gofiber/fiber/v2" -) - -// SubscriptionHandler handles subscription-related endpoints -type SubscriptionHandler struct { - paymentService *services.PaymentService - userService *services.UserService -} - -// NewSubscriptionHandler creates a new subscription handler -func NewSubscriptionHandler(paymentService *services.PaymentService, userService *services.UserService) *SubscriptionHandler { - return &SubscriptionHandler{ - paymentService: paymentService, - userService: userService, - } -} - -// ListPlans returns all available subscription plans -// GET /api/subscriptions/plans -func (h *SubscriptionHandler) ListPlans(c *fiber.Ctx) error { - plans := h.paymentService.GetAvailablePlans() - return c.JSON(fiber.Map{ - "plans": plans, - }) -} - -// GetCurrent returns the user's current subscription -// GET /api/subscriptions/current -func (h *SubscriptionHandler) GetCurrent(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - // Get email from auth middleware - email, _ := c.Locals("user_email").(string) - - ctx := context.Background() - - // Sync user from Supabase (creates user if not exists, applies promo) - user, err := h.userService.SyncUserFromSupabase(ctx, userID, email) - if err != nil { - log.Printf("⚠️ Failed to sync user %s: %v", userID, err) - } - - sub, err := h.paymentService.GetCurrentSubscription(ctx, userID) - if err != nil { - log.Printf("⚠️ Failed to get subscription for user %s: %v", userID, err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get subscription", - }) - } - - // Get user data for promo detection and welcome popup status - isPromoUser := false - hasSeenWelcomePopup := false - - if user != nil { - hasSeenWelcomePopup = user.HasSeenWelcomePopup - // Promo user = PRO tier + has expiration + no Dodo subscription - isPromoUser = user.SubscriptionTier == "pro" && - user.SubscriptionExpiresAt != nil && - user.DodoSubscriptionID == "" - } - - // Build complete subscription response - response := fiber.Map{ - "id": sub.ID.Hex(), - "user_id": sub.UserID, - "tier": sub.Tier, - "status": sub.Status, - "current_period_start": sub.CurrentPeriodStart, - "current_period_end": sub.CurrentPeriodEnd, - "cancel_at_period_end": sub.CancelAtPeriodEnd, - "is_promo_user": isPromoUser, - "has_seen_welcome_popup": hasSeenWelcomePopup, - "created_at": sub.CreatedAt, - "updated_at": sub.UpdatedAt, - } - - // Add optional Dodo fields if present - if sub.DodoSubscriptionID != "" { - response["dodo_subscription_id"] = sub.DodoSubscriptionID - } - if sub.DodoCustomerID != "" { - response["dodo_customer_id"] = sub.DodoCustomerID - } - - // Add subscription expiration (for promo users) - if user != nil && user.SubscriptionExpiresAt != nil { - response["subscription_expires_at"] = user.SubscriptionExpiresAt.Format("2006-01-02T15:04:05Z07:00") - } - - // Add scheduled change info if exists - if sub.HasScheduledChange() { - response["scheduled_tier"] = sub.ScheduledTier - response["scheduled_change_at"] = sub.ScheduledChangeAt - } - - // Add cancelled_at if present - if sub.CancelledAt != nil && !sub.CancelledAt.IsZero() { - response["cancelled_at"] = sub.CancelledAt - } - - return c.JSON(response) -} - -// CreateCheckoutRequest represents a checkout creation request -type CreateCheckoutRequest struct { - PlanID string `json:"plan_id" validate:"required"` -} - -// CreateCheckout creates a checkout session for a subscription -// POST /api/subscriptions/checkout -func (h *SubscriptionHandler) CreateCheckout(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - // Get email from auth context for syncing new users - userEmail, _ := c.Locals("user_email").(string) - - var req CreateCheckoutRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if req.PlanID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "plan_id is required", - }) - } - - ctx := context.Background() - checkout, err := h.paymentService.CreateCheckoutSession(ctx, userID, userEmail, req.PlanID) - if err != nil { - log.Printf("⚠️ Failed to create checkout for user %s: %v", userID, err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": err.Error(), - }) - } - - return c.JSON(checkout) -} - -// ChangePlanRequest represents a plan change request -type ChangePlanRequest struct { - PlanID string `json:"plan_id" validate:"required"` -} - -// ChangePlan changes the user's subscription plan -// POST /api/subscriptions/change-plan -func (h *SubscriptionHandler) ChangePlan(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - var req ChangePlanRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - if req.PlanID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "plan_id is required", - }) - } - - ctx := context.Background() - result, err := h.paymentService.ChangePlan(ctx, userID, req.PlanID) - if err != nil { - log.Printf("⚠️ Failed to change plan for user %s: %v", userID, err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": err.Error(), - }) - } - - return c.JSON(result) -} - -// PreviewPlanChange previews a plan change -// GET /api/subscriptions/change-plan/preview?plan_id=pro -func (h *SubscriptionHandler) PreviewPlanChange(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - planID := c.Query("plan_id") - if planID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "plan_id query parameter is required", - }) - } - - ctx := context.Background() - preview, err := h.paymentService.PreviewPlanChange(ctx, userID, planID) - if err != nil { - log.Printf("⚠️ Failed to preview plan change for user %s: %v", userID, err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": err.Error(), - }) - } - - return c.JSON(preview) -} - -// Cancel cancels the user's subscription -// POST /api/subscriptions/cancel -func (h *SubscriptionHandler) Cancel(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - ctx := context.Background() - err := h.paymentService.CancelSubscription(ctx, userID) - if err != nil { - log.Printf("⚠️ Failed to cancel subscription for user %s: %v", userID, err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": err.Error(), - }) - } - - // Get updated subscription to return cancel date - sub, _ := h.paymentService.GetCurrentSubscription(ctx, userID) - return c.JSON(fiber.Map{ - "status": "pending_cancel", - "cancel_at": sub.CurrentPeriodEnd, - "message": "Your subscription will be cancelled at the end of the billing period. You'll retain access until then.", - }) -} - -// Reactivate reactivates a cancelled subscription -// POST /api/subscriptions/reactivate -func (h *SubscriptionHandler) Reactivate(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - ctx := context.Background() - err := h.paymentService.ReactivateSubscription(ctx, userID) - if err != nil { - log.Printf("⚠️ Failed to reactivate subscription for user %s: %v", userID, err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": err.Error(), - }) - } - - return c.JSON(fiber.Map{ - "status": "active", - "message": "Subscription reactivated successfully", - }) -} - -// GetPortalURL returns the DodoPayments customer portal URL -// GET /api/subscriptions/portal -func (h *SubscriptionHandler) GetPortalURL(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - ctx := context.Background() - url, err := h.paymentService.GetCustomerPortalURL(ctx, userID) - if err != nil { - log.Printf("⚠️ Failed to get portal URL for user %s: %v", userID, err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": err.Error(), - }) - } - - return c.JSON(fiber.Map{ - "portal_url": url, - }) -} - -// ListInvoices returns invoice history (placeholder - requires DodoPayments API) -// GET /api/subscriptions/invoices -func (h *SubscriptionHandler) ListInvoices(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - // TODO: Implement invoice listing when DodoPayments API is available - return c.JSON(fiber.Map{ - "invoices": []interface{}{}, - "message": "Invoice history coming soon", - }) -} - -// SyncSubscription manually syncs subscription data from DodoPayments -// POST /api/subscriptions/sync -func (h *SubscriptionHandler) SyncSubscription(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - result, err := h.paymentService.SyncSubscriptionFromDodo(c.Context(), userID) - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": err.Error(), - }) - } - - return c.JSON(result) -} - -// GetUsageStats returns current usage statistics for the user -// GET /api/subscriptions/usage -func (h *SubscriptionHandler) GetUsageStats(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - ctx := c.Context() - stats, err := h.paymentService.GetUsageStats(ctx, userID) - if err != nil { - log.Printf("⚠️ Failed to get usage stats for user %s: %v", userID, err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get usage statistics", - }) - } - - return c.JSON(stats) -} diff --git a/backend/internal/handlers/subscription_handler_test.go b/backend/internal/handlers/subscription_handler_test.go deleted file mode 100644 index 2d472e6c..00000000 --- a/backend/internal/handlers/subscription_handler_test.go +++ /dev/null @@ -1,297 +0,0 @@ -package handlers - -import ( - "bytes" - "claraverse/internal/models" - "claraverse/internal/services" - "encoding/json" - "io" - "net/http/httptest" - "testing" - - "github.com/gofiber/fiber/v2" -) - -func setupSubscriptionTestApp(t *testing.T) (*fiber.App, *services.PaymentService) { - app := fiber.New() - paymentService := services.NewPaymentService("test", "secret", "biz", nil, nil, nil) - return app, paymentService -} - -func mockSubscriptionAuthMiddleware(userID string) fiber.Handler { - return func(c *fiber.Ctx) error { - c.Locals("user_id", userID) - c.Locals("user_email", "test@example.com") - return c.Next() - } -} - -func TestSubscriptionHandler_ListPlans(t *testing.T) { - app, paymentService := setupSubscriptionTestApp(t) - handler := NewSubscriptionHandler(paymentService) - - app.Get("/api/subscriptions/plans", handler.ListPlans) - - req := httptest.NewRequest("GET", "/api/subscriptions/plans", nil) - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != fiber.StatusOK { - t.Errorf("Expected 200, got %d", resp.StatusCode) - } - - body, _ := io.ReadAll(resp.Body) - var result struct { - Plans []models.Plan `json:"plans"` - } - json.Unmarshal(body, &result) - - if len(result.Plans) < 4 { - t.Errorf("Expected at least 4 plans, got %d", len(result.Plans)) - } -} - -func TestSubscriptionHandler_GetCurrent_Unauthenticated(t *testing.T) { - app, paymentService := setupSubscriptionTestApp(t) - handler := NewSubscriptionHandler(paymentService) - - app.Get("/api/subscriptions/current", handler.GetCurrent) - - req := httptest.NewRequest("GET", "/api/subscriptions/current", nil) - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - - if resp.StatusCode != fiber.StatusUnauthorized { - t.Errorf("Expected 401, got %d", resp.StatusCode) - } -} - -func TestSubscriptionHandler_GetCurrent_Authenticated(t *testing.T) { - app, paymentService := setupSubscriptionTestApp(t) - handler := NewSubscriptionHandler(paymentService) - - app.Use(mockAuthMiddleware("user-123")) - app.Get("/api/subscriptions/current", handler.GetCurrent) - - req := httptest.NewRequest("GET", "/api/subscriptions/current", nil) - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - - // Without MongoDB, expect default free tier response - if resp.StatusCode != fiber.StatusOK { - t.Errorf("Expected 200, got %d", resp.StatusCode) - } - - body, _ := io.ReadAll(resp.Body) - var result struct { - Tier string `json:"tier"` - Status string `json:"status"` - } - json.Unmarshal(body, &result) - - if result.Tier != models.TierFree { - t.Errorf("Expected free tier, got %s", result.Tier) - } -} - -func TestSubscriptionHandler_CreateCheckout_InvalidPlan(t *testing.T) { - app, paymentService := setupSubscriptionTestApp(t) - handler := NewSubscriptionHandler(paymentService) - - app.Use(mockAuthMiddleware("user-123")) - app.Post("/api/subscriptions/checkout", handler.CreateCheckout) - - reqBody := bytes.NewBuffer([]byte(`{"plan_id": "invalid_plan"}`)) - req := httptest.NewRequest("POST", "/api/subscriptions/checkout", reqBody) - req.Header.Set("Content-Type", "application/json") - - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - - if resp.StatusCode != fiber.StatusBadRequest { - t.Errorf("Expected 400 for invalid plan, got %d", resp.StatusCode) - } -} - -func TestSubscriptionHandler_CreateCheckout_FreePlan(t *testing.T) { - app, paymentService := setupSubscriptionTestApp(t) - handler := NewSubscriptionHandler(paymentService) - - app.Use(mockAuthMiddleware("user-123")) - app.Post("/api/subscriptions/checkout", handler.CreateCheckout) - - reqBody := bytes.NewBuffer([]byte(`{"plan_id": "free"}`)) - req := httptest.NewRequest("POST", "/api/subscriptions/checkout", reqBody) - req.Header.Set("Content-Type", "application/json") - - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - - // Should reject - can't checkout for free plan - if resp.StatusCode != fiber.StatusBadRequest { - t.Errorf("Expected 400 for free plan checkout, got %d", resp.StatusCode) - } -} - -func TestSubscriptionHandler_CreateCheckout_EnterprisePlan(t *testing.T) { - app, paymentService := setupSubscriptionTestApp(t) - handler := NewSubscriptionHandler(paymentService) - - app.Use(mockAuthMiddleware("user-123")) - app.Post("/api/subscriptions/checkout", handler.CreateCheckout) - - reqBody := bytes.NewBuffer([]byte(`{"plan_id": "enterprise"}`)) - req := httptest.NewRequest("POST", "/api/subscriptions/checkout", reqBody) - req.Header.Set("Content-Type", "application/json") - - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - - // Should reject - enterprise requires contact sales - if resp.StatusCode != fiber.StatusBadRequest { - t.Errorf("Expected 400 for enterprise checkout, got %d", resp.StatusCode) - } -} - -func TestSubscriptionHandler_CreateCheckout_MissingPlanID(t *testing.T) { - app, paymentService := setupSubscriptionTestApp(t) - handler := NewSubscriptionHandler(paymentService) - - app.Use(mockAuthMiddleware("user-123")) - app.Post("/api/subscriptions/checkout", handler.CreateCheckout) - - reqBody := bytes.NewBuffer([]byte(`{}`)) - req := httptest.NewRequest("POST", "/api/subscriptions/checkout", reqBody) - req.Header.Set("Content-Type", "application/json") - - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - - if resp.StatusCode != fiber.StatusBadRequest { - t.Errorf("Expected 400 for missing plan_id, got %d", resp.StatusCode) - } -} - -func TestSubscriptionHandler_PreviewPlanChange(t *testing.T) { - app, paymentService := setupSubscriptionTestApp(t) - handler := NewSubscriptionHandler(paymentService) - - app.Use(mockAuthMiddleware("user-123")) - app.Get("/api/subscriptions/change-plan/preview", handler.PreviewPlanChange) - - tests := []struct { - name string - planID string - expectCode int - }{ - {"valid upgrade", "pro", fiber.StatusOK}, - {"invalid plan", "invalid", fiber.StatusBadRequest}, - {"missing plan", "", fiber.StatusBadRequest}, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - url := "/api/subscriptions/change-plan/preview" - if tt.planID != "" { - url += "?plan_id=" + tt.planID - } - - req := httptest.NewRequest("GET", url, nil) - resp, _ := app.Test(req) - - if resp.StatusCode != tt.expectCode { - t.Errorf("Expected %d, got %d", tt.expectCode, resp.StatusCode) - } - }) - } -} - -func TestSubscriptionHandler_Cancel(t *testing.T) { - app, paymentService := setupSubscriptionTestApp(t) - handler := NewSubscriptionHandler(paymentService) - - app.Use(mockAuthMiddleware("user-123")) - app.Post("/api/subscriptions/cancel", handler.Cancel) - - req := httptest.NewRequest("POST", "/api/subscriptions/cancel", nil) - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - - // Without active subscription, should handle gracefully - // Could be 200 (already free) or 400 (nothing to cancel) - if resp.StatusCode != fiber.StatusOK && resp.StatusCode != fiber.StatusBadRequest { - t.Errorf("Unexpected status: %d", resp.StatusCode) - } -} - -func TestSubscriptionHandler_Reactivate_NoActiveCancellation(t *testing.T) { - app, paymentService := setupSubscriptionTestApp(t) - handler := NewSubscriptionHandler(paymentService) - - app.Use(mockAuthMiddleware("user-123")) - app.Post("/api/subscriptions/reactivate", handler.Reactivate) - - req := httptest.NewRequest("POST", "/api/subscriptions/reactivate", nil) - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - - // Without pending cancellation, should fail - if resp.StatusCode != fiber.StatusBadRequest { - t.Errorf("Expected 400, got %d", resp.StatusCode) - } -} - -func TestSubscriptionHandler_GetPortalURL_Unauthenticated(t *testing.T) { - app, paymentService := setupSubscriptionTestApp(t) - handler := NewSubscriptionHandler(paymentService) - - app.Get("/api/subscriptions/portal", handler.GetPortalURL) - - req := httptest.NewRequest("GET", "/api/subscriptions/portal", nil) - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - - if resp.StatusCode != fiber.StatusUnauthorized { - t.Errorf("Expected 401, got %d", resp.StatusCode) - } -} - -func TestSubscriptionHandler_ListInvoices(t *testing.T) { - app, paymentService := setupSubscriptionTestApp(t) - handler := NewSubscriptionHandler(paymentService) - - app.Use(mockAuthMiddleware("user-123")) - app.Get("/api/subscriptions/invoices", handler.ListInvoices) - - req := httptest.NewRequest("GET", "/api/subscriptions/invoices", nil) - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - - if resp.StatusCode != fiber.StatusOK { - t.Errorf("Expected 200, got %d", resp.StatusCode) - } -} - diff --git a/backend/internal/handlers/tools.go b/backend/internal/handlers/tools.go deleted file mode 100644 index 85d95649..00000000 --- a/backend/internal/handlers/tools.go +++ /dev/null @@ -1,381 +0,0 @@ -package handlers - -import ( - "claraverse/internal/services" - "claraverse/internal/tools" - "sort" - "strings" - - "github.com/gofiber/fiber/v2" -) - -// ToolsHandler handles tool-related requests -type ToolsHandler struct { - registry *tools.Registry - toolService *services.ToolService -} - -// NewToolsHandler creates a new tools handler -func NewToolsHandler(registry *tools.Registry, toolService *services.ToolService) *ToolsHandler { - return &ToolsHandler{ - registry: registry, - toolService: toolService, - } -} - -// ToolResponse represents a tool in the API response -type ToolResponse struct { - Name string `json:"name"` - DisplayName string `json:"display_name"` - Description string `json:"description"` - Icon string `json:"icon"` - Category string `json:"category"` - Keywords []string `json:"keywords"` - Source string `json:"source"` -} - -// CategoryResponse represents a category with its tools -type CategoryResponse struct { - Name string `json:"name"` - Count int `json:"count"` - Tools []ToolResponse `json:"tools"` -} - -// ListTools returns all tools available to the authenticated user, grouped by category -func (h *ToolsHandler) ListTools(c *fiber.Ctx) error { - // Extract user ID from auth middleware - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "User not authenticated", - }) - } - - // Get all tools for the user (built-in + MCP) - toolsList := h.registry.GetUserTools(userID) - - // Group tools by category - categoryMap := make(map[string][]ToolResponse) - - for _, toolDef := range toolsList { - function, ok := toolDef["function"].(map[string]interface{}) - if !ok { - continue - } - - name, _ := function["name"].(string) - description, _ := function["description"].(string) - - // Get the actual tool to extract metadata - tool, exists := h.registry.GetUserTool(userID, name) - if !exists { - continue - } - - toolResponse := ToolResponse{ - Name: tool.Name, - DisplayName: tool.DisplayName, - Description: description, - Icon: tool.Icon, - Category: tool.Category, - Keywords: tool.Keywords, - Source: string(tool.Source), - } - - // Group by category (default to "other" if no category) - category := tool.Category - if category == "" { - category = "other" - } - - categoryMap[category] = append(categoryMap[category], toolResponse) - } - - // Convert map to array of CategoryResponse - categories := make([]CategoryResponse, 0, len(categoryMap)) - for categoryName, categoryTools := range categoryMap { - categories = append(categories, CategoryResponse{ - Name: categoryName, - Count: len(categoryTools), - Tools: categoryTools, - }) - } - - // Sort categories alphabetically - sort.Slice(categories, func(i, j int) bool { - return categories[i].Name < categories[j].Name - }) - - return c.JSON(fiber.Map{ - "categories": categories, - "total": h.registry.CountUserTools(userID), - }) -} - -// AvailableToolResponse represents a tool with credential metadata -type AvailableToolResponse struct { - Name string `json:"name"` - DisplayName string `json:"display_name"` - Description string `json:"description"` - Icon string `json:"icon"` - Category string `json:"category"` - Keywords []string `json:"keywords"` - Source string `json:"source"` - RequiresCredential bool `json:"requires_credential"` - IntegrationType string `json:"integration_type,omitempty"` -} - -// GetAvailableTools returns tools filtered by user's credentials -// Only tools that don't require credentials OR tools where user has configured credentials are returned -func (h *ToolsHandler) GetAvailableTools(c *fiber.Ctx) error { - // Extract user ID from auth middleware - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "User not authenticated", - }) - } - - // If no tool service available, fall back to all tools - if h.toolService == nil { - return h.ListTools(c) - } - - // Get filtered tools from tool service - filteredTools := h.toolService.GetAvailableTools(c.Context(), userID) - - // Build response with metadata - toolResponses := make([]AvailableToolResponse, 0, len(filteredTools)) - categoryMap := make(map[string][]AvailableToolResponse) - - for _, toolDef := range filteredTools { - function, ok := toolDef["function"].(map[string]interface{}) - if !ok { - continue - } - - name, _ := function["name"].(string) - description, _ := function["description"].(string) - - // Get the actual tool to extract metadata - tool, exists := h.registry.GetUserTool(userID, name) - if !exists { - continue - } - - // Check if tool requires credentials - integrationType := tools.GetIntegrationTypeForTool(name) - requiresCredential := integrationType != "" - - toolResponse := AvailableToolResponse{ - Name: tool.Name, - DisplayName: tool.DisplayName, - Description: description, - Icon: tool.Icon, - Category: tool.Category, - Keywords: tool.Keywords, - Source: string(tool.Source), - RequiresCredential: requiresCredential, - IntegrationType: integrationType, - } - - toolResponses = append(toolResponses, toolResponse) - - // Group by category - category := tool.Category - if category == "" { - category = "other" - } - categoryMap[category] = append(categoryMap[category], toolResponse) - } - - // Convert to category response format - categories := make([]struct { - Name string `json:"name"` - Count int `json:"count"` - Tools []AvailableToolResponse `json:"tools"` - }, 0, len(categoryMap)) - - for categoryName, categoryTools := range categoryMap { - categories = append(categories, struct { - Name string `json:"name"` - Count int `json:"count"` - Tools []AvailableToolResponse `json:"tools"` - }{ - Name: categoryName, - Count: len(categoryTools), - Tools: categoryTools, - }) - } - - // Sort categories alphabetically - sort.Slice(categories, func(i, j int) bool { - return categories[i].Name < categories[j].Name - }) - - // Get total count for comparison - allToolsCount := h.registry.CountUserTools(userID) - filteredCount := allToolsCount - len(filteredTools) - - return c.JSON(fiber.Map{ - "categories": categories, - "total": len(filteredTools), - "filtered_count": filteredCount, // Number of tools filtered out due to missing credentials - }) -} - -// RecommendToolsRequest represents the request body for tool recommendations -type RecommendToolsRequest struct { - BlockName string `json:"block_name"` - BlockDescription string `json:"block_description"` - BlockType string `json:"block_type"` -} - -// ToolRecommendation represents a recommended tool with a score -type ToolRecommendation struct { - Name string `json:"name"` - DisplayName string `json:"display_name"` - Description string `json:"description"` - Icon string `json:"icon"` - Category string `json:"category"` - Keywords []string `json:"keywords"` - Source string `json:"source"` - Score int `json:"score"` - Reason string `json:"reason"` -} - -// RecommendTools returns scored and ranked tool recommendations based on block context -func (h *ToolsHandler) RecommendTools(c *fiber.Ctx) error { - // Extract user ID from auth middleware - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "User not authenticated", - }) - } - - // Parse request body - var req RecommendToolsRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // Tokenize block context (name + description) - context := strings.ToLower(req.BlockName + " " + req.BlockDescription) - contextTokens := tokenize(context) - - // Get all tools for the user - toolsList := h.registry.GetUserTools(userID) - - // Score each tool based on keyword matching - recommendations := []ToolRecommendation{} - - for _, toolDef := range toolsList { - function, ok := toolDef["function"].(map[string]interface{}) - if !ok { - continue - } - - name, _ := function["name"].(string) - - // Get the actual tool to extract metadata - tool, exists := h.registry.GetUserTool(userID, name) - if !exists { - continue - } - - // Calculate match score - score, matchedKeywords := calculateMatchScore(contextTokens, tool.Keywords) - - // Only include tools with a score > 0 - if score > 0 { - reason := "Matches: " + strings.Join(matchedKeywords, ", ") - - recommendations = append(recommendations, ToolRecommendation{ - Name: tool.Name, - DisplayName: tool.DisplayName, - Description: tool.Description, - Icon: tool.Icon, - Category: tool.Category, - Keywords: tool.Keywords, - Source: string(tool.Source), - Score: score, - Reason: reason, - }) - } - } - - // Sort recommendations by score (descending) - sort.Slice(recommendations, func(i, j int) bool { - return recommendations[i].Score > recommendations[j].Score - }) - - // Limit to top 10 recommendations - if len(recommendations) > 10 { - recommendations = recommendations[:10] - } - - return c.JSON(fiber.Map{ - "recommendations": recommendations, - "count": len(recommendations), - }) -} - -// tokenize splits a string into lowercase tokens -func tokenize(text string) []string { - // Replace common separators with spaces - text = strings.ReplaceAll(text, "-", " ") - text = strings.ReplaceAll(text, "_", " ") - text = strings.ReplaceAll(text, "/", " ") - - // Split by whitespace - tokens := strings.Fields(text) - - // Deduplicate tokens - tokenSet := make(map[string]bool) - uniqueTokens := []string{} - for _, token := range tokens { - token = strings.ToLower(token) - if !tokenSet[token] && token != "" { - tokenSet[token] = true - uniqueTokens = append(uniqueTokens, token) - } - } - - return uniqueTokens -} - -// calculateMatchScore calculates how well a tool matches the context tokens -func calculateMatchScore(contextTokens []string, keywords []string) (int, []string) { - score := 0 - matchedKeywords := []string{} - - // Normalize keywords to lowercase - normalizedKeywords := make([]string, len(keywords)) - for i, keyword := range keywords { - normalizedKeywords[i] = strings.ToLower(keyword) - } - - // Check each context token against keywords - for _, token := range contextTokens { - for _, keyword := range normalizedKeywords { - // Exact match - if token == keyword { - score += 10 - matchedKeywords = append(matchedKeywords, keyword) - break - } - - // Partial match (substring) - if strings.Contains(keyword, token) || strings.Contains(token, keyword) { - score += 5 - matchedKeywords = append(matchedKeywords, keyword) - break - } - } - } - - return score, matchedKeywords -} diff --git a/backend/internal/handlers/trigger.go b/backend/internal/handlers/trigger.go deleted file mode 100644 index 1e651be7..00000000 --- a/backend/internal/handlers/trigger.go +++ /dev/null @@ -1,333 +0,0 @@ -package handlers - -import ( - "claraverse/internal/execution" - "claraverse/internal/models" - "claraverse/internal/services" - "context" - "log" - - "github.com/gofiber/fiber/v2" - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// TriggerHandler handles agent trigger endpoints (API key authenticated) -type TriggerHandler struct { - agentService *services.AgentService - executionService *services.ExecutionService - workflowEngine *execution.WorkflowEngine -} - -// NewTriggerHandler creates a new trigger handler -func NewTriggerHandler( - agentService *services.AgentService, - executionService *services.ExecutionService, - workflowEngine *execution.WorkflowEngine, -) *TriggerHandler { - return &TriggerHandler{ - agentService: agentService, - executionService: executionService, - workflowEngine: workflowEngine, - } -} - -// TriggerAgent executes an agent via API key -// POST /api/trigger/:agentId -func (h *TriggerHandler) TriggerAgent(c *fiber.Ctx) error { - agentID := c.Params("agentId") - userID := c.Locals("user_id").(string) - - // Parse request body - var req models.TriggerAgentRequest - if err := c.BodyParser(&req); err != nil && err.Error() != "Unprocessable Entity" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // Get the agent - agent, err := h.agentService.GetAgentByID(agentID) - if err != nil { - log.Printf("❌ [TRIGGER] Agent not found: %s", agentID) - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Agent not found", - }) - } - - // Verify the agent belongs to the user (API key owner) - if agent.UserID != userID { - log.Printf("🚫 [TRIGGER] User %s attempted to trigger agent %s (owned by %s)", userID, agentID, agent.UserID) - return c.Status(fiber.StatusForbidden).JSON(fiber.Map{ - "error": "You do not have permission to trigger this agent", - }) - } - - // Check if agent has a workflow - if agent.Workflow == nil || len(agent.Workflow.Blocks) == 0 { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Agent has no workflow configured", - }) - } - - // Get API key ID from context (for tracking) - var apiKeyID primitive.ObjectID - if apiKey, ok := c.Locals("api_key").(*models.APIKey); ok { - apiKeyID = apiKey.ID - } - - // Create execution record - execReq := &services.CreateExecutionRequest{ - AgentID: agentID, - UserID: userID, - WorkflowVersion: agent.Workflow.Version, - TriggerType: "api", - APIKeyID: apiKeyID, - Input: req.Input, - } - - execRecord, err := h.executionService.Create(c.Context(), execReq) - if err != nil { - log.Printf("❌ [TRIGGER] Failed to create execution record: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to create execution", - }) - } - - // Update execution status to running - if err := h.executionService.UpdateStatus(c.Context(), execRecord.ID, "running"); err != nil { - log.Printf("⚠️ [TRIGGER] Failed to update status: %v", err) - } - - // Build execution options - block checker is DISABLED for API triggers - // Block checker should only run during platform testing (WebSocket), not production API calls - execOpts := &ExecuteWorkflowOptions{ - AgentDescription: agent.Description, - EnableBlockChecker: false, // Disabled for API triggers - CheckerModelID: req.CheckerModelID, - } - - // Execute workflow asynchronously (pass userID for credential resolution) - go h.executeWorkflow(execRecord.ID, agent.Workflow, req.Input, userID, execOpts) - - log.Printf("🚀 [TRIGGER] Triggered agent %s via API (execution: %s)", agentID, execRecord.ID.Hex()) - - return c.Status(fiber.StatusAccepted).JSON(models.TriggerAgentResponse{ - ExecutionID: execRecord.ID.Hex(), - Status: "running", - Message: "Agent execution started", - }) -} - -// ExecuteWorkflowOptions contains options for executing a workflow -type ExecuteWorkflowOptions struct { - AgentDescription string - EnableBlockChecker bool - CheckerModelID string -} - -// executeWorkflow runs the workflow and updates the execution record -func (h *TriggerHandler) executeWorkflow(executionID primitive.ObjectID, workflow *models.Workflow, input map[string]interface{}, userID string, opts *ExecuteWorkflowOptions) { - ctx := context.Background() - - // Create a channel for status updates (we'll drain it since API triggers don't need real-time) - statusChan := make(chan models.ExecutionUpdate, 100) - go func() { - for range statusChan { - // Drain channel - future: could publish to Redis for status polling - } - }() - - // Transform input to properly wrap file references for Start blocks - transformedInput := h.transformInputForWorkflow(workflow, input) - - // Inject user context for credential resolution and tool execution - if transformedInput == nil { - transformedInput = make(map[string]interface{}) - } - transformedInput["__user_id__"] = userID - - // Build execution options - block checker DISABLED for API triggers - // Block checker should only run during platform testing (WebSocket), not production API calls - execOptions := &execution.ExecutionOptions{ - EnableBlockChecker: false, // Disabled for API triggers - } - if opts != nil { - execOptions.WorkflowGoal = opts.AgentDescription - execOptions.CheckerModelID = opts.CheckerModelID - } - log.Printf("🔍 [TRIGGER] Block checker disabled (API trigger - validation only runs during platform testing)") - - // Execute the workflow - result, err := h.workflowEngine.ExecuteWithOptions(ctx, workflow, transformedInput, statusChan, execOptions) - close(statusChan) - - // Update execution record - completeReq := &services.ExecutionCompleteRequest{ - Status: "completed", - } - - if err != nil { - completeReq.Status = "failed" - completeReq.Error = err.Error() - log.Printf("❌ [TRIGGER] Execution %s failed: %v", executionID.Hex(), err) - } else { - completeReq.Status = result.Status - completeReq.Output = result.Output - completeReq.BlockStates = result.BlockStates - if result.Error != "" { - completeReq.Error = result.Error - } - log.Printf("✅ [TRIGGER] Execution %s completed with status: %s", executionID.Hex(), result.Status) - } - - if err := h.executionService.Complete(ctx, executionID, completeReq); err != nil { - log.Printf("⚠️ [TRIGGER] Failed to complete execution record: %v", err) - } -} - -// GetExecutionStatus gets the status of an execution -// GET /api/trigger/status/:executionId -func (h *TriggerHandler) GetExecutionStatus(c *fiber.Ctx) error { - executionIDStr := c.Params("executionId") - userID := c.Locals("user_id").(string) - - executionID, err := primitive.ObjectIDFromHex(executionIDStr) - if err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid execution ID", - }) - } - - execution, err := h.executionService.GetByIDAndUser(c.Context(), executionID, userID) - if err != nil { - if err.Error() == "execution not found" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "Execution not found", - }) - } - log.Printf("❌ [TRIGGER] Failed to get execution: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to get execution status", - }) - } - - return c.JSON(execution) -} - -// transformInputForWorkflow transforms API input to match workflow expectations -// This handles the case where file reference fields (file_id, filename, mime_type) -// are passed directly in input but the Start block expects them nested under a variable name -func (h *TriggerHandler) transformInputForWorkflow(workflow *models.Workflow, input map[string]interface{}) map[string]interface{} { - if workflow == nil || input == nil { - return input - } - - // Find the Start block (variable block with operation "read") - var startBlockVariableName string - var startBlockInputType string - - for _, block := range workflow.Blocks { - if block.Type == "variable" { - config := block.Config - - operation, _ := config["operation"].(string) - if operation != "read" { - continue - } - - // Found a Start block - get its variable name and input type - if varName, ok := config["variableName"].(string); ok { - startBlockVariableName = varName - } - if inputType, ok := config["inputType"].(string); ok { - startBlockInputType = inputType - } - break - } - } - - // If no Start block found, return input as-is - if startBlockVariableName == "" { - return input - } - - log.Printf("🔧 [TRIGGER] Found Start block: variableName=%s, inputType=%s", startBlockVariableName, startBlockInputType) - - // Check if input contains file reference fields at top level - _, hasFileID := input["file_id"] - _, hasFilename := input["filename"] - _, hasMimeType := input["mime_type"] - - isFileReferenceInput := hasFileID && (hasFilename || hasMimeType) - - // If this is a file input type and we have file reference fields at top level, - // wrap them under the Start block's variable name - if startBlockInputType == "file" && isFileReferenceInput { - log.Printf("🔧 [TRIGGER] Wrapping file reference fields under variable '%s'", startBlockVariableName) - - fileRef := map[string]interface{}{ - "file_id": input["file_id"], - } - if hasFilename { - fileRef["filename"] = input["filename"] - } - if hasMimeType { - fileRef["mime_type"] = input["mime_type"] - } - - // Create new input with file reference wrapped - newInput := make(map[string]interface{}) - - // Copy non-file-reference fields - for k, v := range input { - if k != "file_id" && k != "filename" && k != "mime_type" { - newInput[k] = v - } - } - - // Add wrapped file reference - newInput[startBlockVariableName] = fileRef - - log.Printf("✅ [TRIGGER] Transformed input: %+v", newInput) - return newInput - } - - // For text inputs, check if the variable name doesn't exist but we have a single text value - if startBlockInputType == "text" || startBlockInputType == "" { - // If the variable already exists in input, no transformation needed - if _, exists := input[startBlockVariableName]; exists { - return input - } - - // If input has a single "text", "value", or "message" field, map it to the variable name - if text, ok := input["text"].(string); ok { - newInput := make(map[string]interface{}) - for k, v := range input { - newInput[k] = v - } - newInput[startBlockVariableName] = text - log.Printf("🔧 [TRIGGER] Mapped 'text' field to variable '%s'", startBlockVariableName) - return newInput - } - if value, ok := input["value"].(string); ok { - newInput := make(map[string]interface{}) - for k, v := range input { - newInput[k] = v - } - newInput[startBlockVariableName] = value - log.Printf("🔧 [TRIGGER] Mapped 'value' field to variable '%s'", startBlockVariableName) - return newInput - } - if message, ok := input["message"].(string); ok { - newInput := make(map[string]interface{}) - for k, v := range input { - newInput[k] = v - } - newInput[startBlockVariableName] = message - log.Printf("🔧 [TRIGGER] Mapped 'message' field to variable '%s'", startBlockVariableName) - return newInput - } - } - - return input -} diff --git a/backend/internal/handlers/upload.go b/backend/internal/handlers/upload.go deleted file mode 100644 index 8e6b997c..00000000 --- a/backend/internal/handlers/upload.go +++ /dev/null @@ -1,918 +0,0 @@ -package handlers - -import ( - "bytes" - "claraverse/internal/filecache" - "claraverse/internal/security" - "claraverse/internal/services" - "claraverse/internal/utils" - "encoding/csv" - "fmt" - "io" - "log" - "mime/multipart" - "net/http" - "os" - "path/filepath" - "strings" - "time" - - "github.com/gofiber/fiber/v2" - "github.com/google/uuid" -) - -// UploadHandler handles file upload requests -type UploadHandler struct { - uploadDir string - maxImageSize int64 - maxPDFSize int64 - maxDocSize int64 // For DOCX and PPTX - allowedTypes map[string]bool - fileCache *filecache.Service - usageLimiter *services.UsageLimiterService -} - -// NewUploadHandler creates a new upload handler -func NewUploadHandler(uploadDir string, usageLimiter *services.UsageLimiterService) *UploadHandler { - // Ensure upload directory exists with restricted permissions - if err := os.MkdirAll(uploadDir, 0700); err != nil { - log.Printf("⚠️ Warning: Could not create upload directory: %v", err) - } - - return &UploadHandler{ - uploadDir: uploadDir, - maxImageSize: 20 * 1024 * 1024, // 20MB for images - maxPDFSize: 10 * 1024 * 1024, // 10MB for PDFs - maxDocSize: 10 * 1024 * 1024, // 10MB for DOCX/PPTX - usageLimiter: usageLimiter, - allowedTypes: map[string]bool{ - "image/jpeg": true, - "image/jpg": true, - "image/png": true, - "image/webp": true, - "image/gif": true, - "application/pdf": true, - "text/csv": true, - "application/vnd.ms-excel": true, // .xls - "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet": true, // .xlsx - "application/json": true, - "text/plain": true, - // Office documents - "application/vnd.openxmlformats-officedocument.wordprocessingml.document": true, // .docx - "application/vnd.openxmlformats-officedocument.presentationml.presentation": true, // .pptx - // Audio files (for Whisper transcription) - "audio/mpeg": true, // .mp3 - "audio/mp3": true, // .mp3 alternate - "audio/wav": true, // .wav - "audio/x-wav": true, // .wav alternate - "audio/wave": true, // .wav alternate - "audio/mp4": true, // .m4a - "audio/x-m4a": true, // .m4a alternate - "audio/webm": true, // .webm - "audio/ogg": true, // .ogg - "audio/flac": true, // .flac - }, - fileCache: filecache.GetService(), - } -} - -// UploadResponse represents the upload API response -type UploadResponse struct { - FileID string `json:"file_id"` - Filename string `json:"filename"` - MimeType string `json:"mime_type"` - Size int64 `json:"size"` - Hash string `json:"hash,omitempty"` - PageCount int `json:"page_count,omitempty"` - WordCount int `json:"word_count,omitempty"` - Preview string `json:"preview,omitempty"` - ConversationID string `json:"conversation_id,omitempty"` - URL string `json:"url,omitempty"` // Deprecated for PDFs - use file_id - DataPreview *CSVPreview `json:"data_preview,omitempty"` -} - -// CSVPreview represents a preview of CSV/tabular data -type CSVPreview struct { - Headers []string `json:"headers"` - Rows [][]string `json:"rows"` - RowCount int `json:"row_count"` // Total rows in file - ColCount int `json:"col_count"` // Total columns -} - -// Upload handles file upload requests -func (h *UploadHandler) Upload(c *fiber.Ctx) error { - // Check authentication - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" || userID == "anonymous" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required for file uploads", - }) - } - - // Check file upload limit - if h.usageLimiter != nil { - ctx := c.Context() - if err := h.usageLimiter.CheckFileUploadLimit(ctx, userID); err != nil { - if limitErr, ok := err.(*services.LimitExceededError); ok { - log.Printf("⚠️ [LIMIT] File upload limit exceeded for user %s: %s", userID, limitErr.Message) - return c.Status(fiber.StatusTooManyRequests).JSON(fiber.Map{ - "error": limitErr.Message, - "error_code": limitErr.ErrorCode, - "limit": limitErr.Limit, - "used": limitErr.Used, - "reset_at": limitErr.ResetAt, - "upgrade_to": limitErr.UpgradeTo, - }) - } - } - - // Increment file upload count (check passed) - defer to ensure it runs even if upload fails - defer func() { - if err := h.usageLimiter.IncrementFileUploadCount(c.Context(), userID); err != nil { - log.Printf("⚠️ [LIMIT] Failed to increment file upload count for user %s: %v", userID, err) - } - }() - } - - // Get conversation_id from form or create new one - conversationID := c.FormValue("conversation_id") - if conversationID == "" { - conversationID = uuid.New().String() - } - - // Get uploaded file - fileHeader, err := c.FormFile("file") - if err != nil { - log.Printf("❌ [UPLOAD] Failed to parse file: %v", err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "No file provided or invalid file", - }) - } - - // Open uploaded file - file, err := fileHeader.Open() - if err != nil { - log.Printf("❌ [UPLOAD] Failed to open file: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to process file", - }) - } - defer file.Close() - - // Read file data into memory - fileData, err := io.ReadAll(file) - if err != nil { - log.Printf("❌ [UPLOAD] Failed to read file: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to read file", - }) - } - - // Detect content type - detectedMimeType := h.detectContentTypeFromData(fileData, fileHeader) - - // Strip charset and other parameters from MIME type (e.g., "text/plain; charset=utf-8" -> "text/plain") - mimeType := strings.Split(detectedMimeType, ";")[0] - mimeType = strings.TrimSpace(mimeType) - - // Validate content type - if !h.allowedTypes[mimeType] { - log.Printf("⚠️ [UPLOAD] Disallowed file type: %s (detected as: %s)", mimeType, detectedMimeType) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": fmt.Sprintf("File type not allowed: %s. Allowed types: PNG, JPG, WebP, GIF, PDF, DOCX, PPTX, CSV, Excel, JSON, MP3, WAV, M4A, OGG, FLAC", mimeType), - }) - } - - // Validate file size based on type - maxSize := h.maxImageSize - if mimeType == "application/pdf" { - maxSize = h.maxPDFSize - } else if mimeType == "application/vnd.openxmlformats-officedocument.wordprocessingml.document" || - mimeType == "application/vnd.openxmlformats-officedocument.presentationml.presentation" { - // DOCX, PPTX files: 10MB limit - maxSize = h.maxDocSize - } else if strings.HasPrefix(mimeType, "text/") || strings.Contains(mimeType, "json") || strings.Contains(mimeType, "spreadsheet") || strings.Contains(mimeType, "excel") { - // CSV, JSON, Excel files: 100MB limit - maxSize = 100 * 1024 * 1024 - } else if strings.HasPrefix(mimeType, "audio/") { - // Audio files: 25MB limit (OpenAI Whisper limit) - maxSize = 25 * 1024 * 1024 - } - - if fileHeader.Size > maxSize { - log.Printf("⚠️ [UPLOAD] File too large: %d bytes (max %d)", fileHeader.Size, maxSize) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": fmt.Sprintf("File too large. Maximum size is %d MB", maxSize/(1024*1024)), - }) - } - - // Calculate file hash (before encryption) - fileHash := security.CalculateDataHash(fileData) - - // Generate unique file ID - fileID := uuid.New().String() - - // Handle PDF files with secure processing - if mimeType == "application/pdf" { - return h.handlePDFUpload(c, fileID, userID, conversationID, fileHeader, fileData, fileHash) - } - - // Handle DOCX files with secure processing - if mimeType == "application/vnd.openxmlformats-officedocument.wordprocessingml.document" { - return h.handleDOCXUpload(c, fileID, userID, conversationID, fileHeader, fileData, fileHash) - } - - // Handle PPTX files with secure processing - if mimeType == "application/vnd.openxmlformats-officedocument.presentationml.presentation" { - return h.handlePPTXUpload(c, fileID, userID, conversationID, fileHeader, fileData, fileHash) - } - - // Handle CSV/Excel/JSON files for E2B tools - if strings.HasPrefix(mimeType, "text/csv") || strings.Contains(mimeType, "spreadsheet") || strings.Contains(mimeType, "excel") || mimeType == "application/json" || mimeType == "text/plain" { - return h.handleDataFileUpload(c, fileID, userID, fileHeader, fileData, mimeType, fileHash) - } - - // Handle audio files (for Whisper transcription) - if strings.HasPrefix(mimeType, "audio/") { - return h.handleAudioUpload(c, fileID, userID, fileHeader, fileData, mimeType, fileHash) - } - - // Handle image files (existing flow) - return h.handleImageUpload(c, fileID, userID, fileHeader, fileData, mimeType, fileHash) -} - -// handlePDFUpload processes PDF files with maximum security -func (h *UploadHandler) handlePDFUpload(c *fiber.Ctx, fileID, userID, conversationID string, fileHeader *multipart.FileHeader, fileData []byte, fileHash *security.Hash) error { - log.Printf("📄 [UPLOAD] Processing PDF: %s (user: %s, size: %d bytes)", fileHeader.Filename, userID, len(fileData)) - - // Validate PDF structure - if err := utils.ValidatePDF(fileData); err != nil { - log.Printf("❌ [UPLOAD] Invalid PDF: %v", err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid or corrupted PDF file", - }) - } - - // Create temporary encrypted file - tempDir := os.TempDir() - tempEncryptedPath := filepath.Join(tempDir, fileID+".encrypted") - - // Write encrypted file temporarily - encKey, err := security.GenerateKey() - if err != nil { - log.Printf("❌ [UPLOAD] Failed to generate encryption key: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to process file", - }) - } - - encryptedData, err := security.EncryptData(fileData, encKey) - if err != nil { - log.Printf("❌ [UPLOAD] Failed to encrypt file: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to process file", - }) - } - - if err := os.WriteFile(tempEncryptedPath, encryptedData, 0600); err != nil { - log.Printf("❌ [UPLOAD] Failed to write temp file: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to process file", - }) - } - - // Extract text from PDF (in memory) - metadata, err := utils.ExtractPDFText(fileData) - if err != nil { - // Clean up temp file before returning - security.SecureDeleteFile(tempEncryptedPath) - log.Printf("❌ [UPLOAD] Failed to extract PDF text: %v", err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Failed to extract text from PDF. File may be corrupted or scanned.", - }) - } - - // Delete encrypted file immediately (max 3 seconds on disk) - if err := security.SecureDeleteFile(tempEncryptedPath); err != nil { - log.Printf("⚠️ [UPLOAD] Failed to securely delete temp file: %v", err) - // Continue anyway - file is encrypted - } - - log.Printf("🗑️ [UPLOAD] Encrypted temp file deleted (file was on disk < 3 seconds)") - - // Store in memory cache only - cachedFile := &filecache.CachedFile{ - FileID: fileID, - UserID: userID, - ConversationID: conversationID, - ExtractedText: security.NewSecureString(metadata.Text), - FileHash: *fileHash, - Filename: fileHeader.Filename, - MimeType: "application/pdf", - Size: fileHeader.Size, - PageCount: metadata.PageCount, - WordCount: metadata.WordCount, - UploadedAt: time.Now(), - } - - h.fileCache.Store(cachedFile) - - // Generate preview - preview := utils.GetPDFPreview(metadata.Text, 200) - - log.Printf("✅ [UPLOAD] PDF uploaded successfully: %s (pages: %d, words: %d)", fileHeader.Filename, metadata.PageCount, metadata.WordCount) - - return c.Status(fiber.StatusCreated).JSON(UploadResponse{ - FileID: fileID, - Filename: fileHeader.Filename, - MimeType: "application/pdf", - Size: fileHeader.Size, - Hash: fileHash.String(), - PageCount: metadata.PageCount, - WordCount: metadata.WordCount, - Preview: preview, - ConversationID: conversationID, - }) -} - -// handleDOCXUpload processes DOCX files with secure text extraction -func (h *UploadHandler) handleDOCXUpload(c *fiber.Ctx, fileID, userID, conversationID string, fileHeader *multipart.FileHeader, fileData []byte, fileHash *security.Hash) error { - log.Printf("📄 [UPLOAD] Processing DOCX: %s (user: %s, size: %d bytes)", fileHeader.Filename, userID, len(fileData)) - - // Validate DOCX structure - if err := utils.ValidateDOCX(fileData); err != nil { - log.Printf("❌ [UPLOAD] Invalid DOCX: %v", err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid or corrupted DOCX file", - }) - } - - // Extract text from DOCX (in memory) - metadata, err := utils.ExtractDOCXText(fileData) - if err != nil { - log.Printf("❌ [UPLOAD] Failed to extract DOCX text: %v", err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Failed to extract text from DOCX. File may be corrupted.", - }) - } - - // Store in memory cache only - cachedFile := &filecache.CachedFile{ - FileID: fileID, - UserID: userID, - ConversationID: conversationID, - ExtractedText: security.NewSecureString(metadata.Text), - FileHash: *fileHash, - Filename: fileHeader.Filename, - MimeType: "application/vnd.openxmlformats-officedocument.wordprocessingml.document", - Size: fileHeader.Size, - PageCount: metadata.PageCount, - WordCount: metadata.WordCount, - UploadedAt: time.Now(), - } - - h.fileCache.Store(cachedFile) - - // Generate preview - preview := utils.GetDOCXPreview(metadata.Text, 200) - - log.Printf("✅ [UPLOAD] DOCX uploaded successfully: %s (pages: %d, words: %d)", fileHeader.Filename, metadata.PageCount, metadata.WordCount) - - return c.Status(fiber.StatusCreated).JSON(UploadResponse{ - FileID: fileID, - Filename: fileHeader.Filename, - MimeType: "application/vnd.openxmlformats-officedocument.wordprocessingml.document", - Size: fileHeader.Size, - Hash: fileHash.String(), - PageCount: metadata.PageCount, - WordCount: metadata.WordCount, - Preview: preview, - ConversationID: conversationID, - }) -} - -// handlePPTXUpload processes PPTX files with secure text extraction -func (h *UploadHandler) handlePPTXUpload(c *fiber.Ctx, fileID, userID, conversationID string, fileHeader *multipart.FileHeader, fileData []byte, fileHash *security.Hash) error { - log.Printf("📊 [UPLOAD] Processing PPTX: %s (user: %s, size: %d bytes)", fileHeader.Filename, userID, len(fileData)) - - // Validate PPTX structure - if err := utils.ValidatePPTX(fileData); err != nil { - log.Printf("❌ [UPLOAD] Invalid PPTX: %v", err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid or corrupted PPTX file", - }) - } - - // Extract text from PPTX (in memory) - metadata, err := utils.ExtractPPTXText(fileData) - if err != nil { - log.Printf("❌ [UPLOAD] Failed to extract PPTX text: %v", err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Failed to extract text from PPTX. File may be corrupted.", - }) - } - - // Store in memory cache only - cachedFile := &filecache.CachedFile{ - FileID: fileID, - UserID: userID, - ConversationID: conversationID, - ExtractedText: security.NewSecureString(metadata.Text), - FileHash: *fileHash, - Filename: fileHeader.Filename, - MimeType: "application/vnd.openxmlformats-officedocument.presentationml.presentation", - Size: fileHeader.Size, - PageCount: metadata.SlideCount, // Use SlideCount as PageCount - WordCount: metadata.WordCount, - UploadedAt: time.Now(), - } - - h.fileCache.Store(cachedFile) - - // Generate preview - preview := utils.GetPPTXPreview(metadata.Text, 200) - - log.Printf("✅ [UPLOAD] PPTX uploaded successfully: %s (slides: %d, words: %d)", fileHeader.Filename, metadata.SlideCount, metadata.WordCount) - - return c.Status(fiber.StatusCreated).JSON(UploadResponse{ - FileID: fileID, - Filename: fileHeader.Filename, - MimeType: "application/vnd.openxmlformats-officedocument.presentationml.presentation", - Size: fileHeader.Size, - Hash: fileHash.String(), - PageCount: metadata.SlideCount, - WordCount: metadata.WordCount, - Preview: preview, - ConversationID: conversationID, - }) -} - -// handleImageUpload processes image files (existing flow, now with hash) -func (h *UploadHandler) handleImageUpload(c *fiber.Ctx, fileID, userID string, fileHeader *multipart.FileHeader, fileData []byte, mimeType string, fileHash *security.Hash) error { - // Get conversation_id from form (may be empty) - conversationID := c.FormValue("conversation_id") - - // Generate filename with extension - ext := filepath.Ext(fileHeader.Filename) - if ext == "" { - ext = h.getExtensionFromMimeType(mimeType) - } - savedFilename := fileID + ext - filePath := filepath.Join(h.uploadDir, savedFilename) - - // Save image to disk with restricted permissions (owner read/write only for security) - if err := os.WriteFile(filePath, fileData, 0600); err != nil { - log.Printf("❌ [UPLOAD] Failed to save file: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to save file", - }) - } - - // Register image in cache for auto-deletion - cachedFile := &filecache.CachedFile{ - FileID: fileID, - UserID: userID, - ConversationID: conversationID, - FileHash: *fileHash, - Filename: fileHeader.Filename, - MimeType: mimeType, - Size: fileHeader.Size, - FilePath: filePath, // Track disk location - UploadedAt: time.Now(), - } - h.fileCache.Store(cachedFile) - - log.Printf("✅ [UPLOAD] Image uploaded successfully: %s (user: %s, size: %d bytes)", savedFilename, userID, fileHeader.Size) - - // Build file URL - fileURL := fmt.Sprintf("/uploads/%s", savedFilename) - - return c.Status(fiber.StatusCreated).JSON(UploadResponse{ - FileID: fileID, - URL: fileURL, - MimeType: mimeType, - Size: fileHeader.Size, - Filename: fileHeader.Filename, - Hash: fileHash.String(), - }) -} - -// handleDataFileUpload processes CSV/Excel/JSON files for E2B tools -func (h *UploadHandler) handleDataFileUpload(c *fiber.Ctx, fileID, userID string, fileHeader *multipart.FileHeader, fileData []byte, mimeType string, fileHash *security.Hash) error { - // Get conversation_id from form (may be empty) - conversationID := c.FormValue("conversation_id") - - // Generate filename with extension - ext := filepath.Ext(fileHeader.Filename) - if ext == "" { - ext = h.getExtensionFromMimeType(mimeType) - } - savedFilename := fileID + ext - filePath := filepath.Join(h.uploadDir, savedFilename) - - // Save file to disk with restricted permissions - if err := os.WriteFile(filePath, fileData, 0600); err != nil { - log.Printf("❌ [UPLOAD] Failed to save data file: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to save file", - }) - } - - // Register file in cache for auto-deletion - cachedFile := &filecache.CachedFile{ - FileID: fileID, - UserID: userID, - ConversationID: conversationID, - FileHash: *fileHash, - Filename: fileHeader.Filename, - MimeType: mimeType, - Size: fileHeader.Size, - FilePath: filePath, - UploadedAt: time.Now(), - } - h.fileCache.Store(cachedFile) - - // Parse CSV preview if it's a CSV file - var dataPreview *CSVPreview - if mimeType == "text/csv" || strings.HasSuffix(strings.ToLower(fileHeader.Filename), ".csv") { - dataPreview = h.parseCSVPreview(fileData) - } - - log.Printf("✅ [UPLOAD] Data file uploaded successfully: %s (user: %s, size: %d bytes, type: %s)", savedFilename, userID, fileHeader.Size, mimeType) - - // Build file URL - fileURL := fmt.Sprintf("/uploads/%s", savedFilename) - - return c.Status(fiber.StatusCreated).JSON(UploadResponse{ - FileID: fileID, - URL: fileURL, - MimeType: mimeType, - Size: fileHeader.Size, - Filename: fileHeader.Filename, - Hash: fileHash.String(), - DataPreview: dataPreview, - }) -} - -// handleAudioUpload processes audio files for Whisper transcription -func (h *UploadHandler) handleAudioUpload(c *fiber.Ctx, fileID, userID string, fileHeader *multipart.FileHeader, fileData []byte, mimeType string, fileHash *security.Hash) error { - // Get conversation_id from form (may be empty) - conversationID := c.FormValue("conversation_id") - - log.Printf("🎵 [UPLOAD] Processing audio: %s (user: %s, size: %d bytes, type: %s)", fileHeader.Filename, userID, len(fileData), mimeType) - - // Generate filename with extension - ext := filepath.Ext(fileHeader.Filename) - if ext == "" { - ext = h.getExtensionFromMimeType(mimeType) - } - savedFilename := fileID + ext - filePath := filepath.Join(h.uploadDir, savedFilename) - - // Save audio to disk with restricted permissions - if err := os.WriteFile(filePath, fileData, 0600); err != nil { - log.Printf("❌ [UPLOAD] Failed to save audio file: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to save file", - }) - } - - // Register file in cache for auto-deletion - cachedFile := &filecache.CachedFile{ - FileID: fileID, - UserID: userID, - ConversationID: conversationID, - FileHash: *fileHash, - Filename: fileHeader.Filename, - MimeType: mimeType, - Size: fileHeader.Size, - FilePath: filePath, - UploadedAt: time.Now(), - } - h.fileCache.Store(cachedFile) - - log.Printf("✅ [UPLOAD] Audio file uploaded successfully: %s (user: %s, size: %d bytes)", savedFilename, userID, fileHeader.Size) - - // Build file URL - fileURL := fmt.Sprintf("/uploads/%s", savedFilename) - - return c.Status(fiber.StatusCreated).JSON(UploadResponse{ - FileID: fileID, - URL: fileURL, - MimeType: mimeType, - Size: fileHeader.Size, - Filename: fileHeader.Filename, - Hash: fileHash.String(), - }) -} - -// parseCSVPreview extracts headers and first rows from CSV data -func (h *UploadHandler) parseCSVPreview(data []byte) *CSVPreview { - reader := csv.NewReader(bytes.NewReader(data)) - reader.LazyQuotes = true - reader.TrimLeadingSpace = true - - // Read all records to get total count - allRecords, err := reader.ReadAll() - if err != nil || len(allRecords) == 0 { - log.Printf("⚠️ [UPLOAD] Failed to parse CSV preview: %v", err) - return nil - } - - // First row is headers - headers := allRecords[0] - totalRows := len(allRecords) - 1 // Exclude header row - - // Get first 5 rows for preview (excluding header) - maxPreviewRows := 5 - if totalRows < maxPreviewRows { - maxPreviewRows = totalRows - } - - previewRows := make([][]string, maxPreviewRows) - for i := 0; i < maxPreviewRows; i++ { - previewRows[i] = allRecords[i+1] // +1 to skip header - } - - return &CSVPreview{ - Headers: headers, - Rows: previewRows, - RowCount: totalRows, - ColCount: len(headers), - } -} - -// detectContentTypeFromData detects MIME type from byte data -func (h *UploadHandler) detectContentTypeFromData(data []byte, header *multipart.FileHeader) string { - // Check PDF magic bytes first - if bytes.HasPrefix(data, []byte("%PDF-")) { - return "application/pdf" - } - - // Check for ZIP-based Office formats (DOCX, PPTX, XLSX start with PK) - // These are ZIP files, so we need to check extension - if bytes.HasPrefix(data, []byte("PK")) { - ext := strings.ToLower(filepath.Ext(header.Filename)) - switch ext { - case ".docx": - return "application/vnd.openxmlformats-officedocument.wordprocessingml.document" - case ".pptx": - return "application/vnd.openxmlformats-officedocument.presentationml.presentation" - case ".xlsx": - return "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet" - } - } - - // Use http.DetectContentType for other types - mimeType := http.DetectContentType(data) - - // Handle fallback for octet-stream or application/zip - if mimeType == "application/octet-stream" || mimeType == "application/zip" { - ext := strings.ToLower(filepath.Ext(header.Filename)) - switch ext { - case ".pdf": - return "application/pdf" - case ".jpg", ".jpeg": - return "image/jpeg" - case ".png": - return "image/png" - case ".gif": - return "image/gif" - case ".webp": - return "image/webp" - case ".docx": - return "application/vnd.openxmlformats-officedocument.wordprocessingml.document" - case ".pptx": - return "application/vnd.openxmlformats-officedocument.presentationml.presentation" - case ".xlsx": - return "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet" - // Audio formats - case ".mp3": - return "audio/mpeg" - case ".wav": - return "audio/wav" - case ".m4a": - return "audio/mp4" - case ".webm": - return "audio/webm" - case ".ogg": - return "audio/ogg" - case ".flac": - return "audio/flac" - } - } - - return mimeType -} - -// detectContentType detects the MIME type of the uploaded file -func (h *UploadHandler) detectContentType(file multipart.File, header *multipart.FileHeader) (string, error) { - // Read first 512 bytes for content type detection - buffer := make([]byte, 512) - n, err := file.Read(buffer) - if err != nil && err != io.EOF { - return "", err - } - - // Detect content type - mimeType := http.DetectContentType(buffer[:n]) - - // Handle some edge cases where DetectContentType returns generic types - if mimeType == "application/octet-stream" { - // Fall back to extension-based detection - ext := strings.ToLower(filepath.Ext(header.Filename)) - switch ext { - case ".jpg", ".jpeg": - mimeType = "image/jpeg" - case ".png": - mimeType = "image/png" - case ".gif": - mimeType = "image/gif" - case ".webp": - mimeType = "image/webp" - } - } - - return mimeType, nil -} - -// getExtensionFromMimeType returns file extension for a given MIME type -func (h *UploadHandler) getExtensionFromMimeType(mimeType string) string { - switch mimeType { - case "application/pdf": - return ".pdf" - case "image/jpeg", "image/jpg": - return ".jpg" - case "image/png": - return ".png" - case "image/gif": - return ".gif" - case "image/webp": - return ".webp" - case "text/csv": - return ".csv" - case "application/json": - return ".json" - case "text/plain": - return ".txt" - case "application/vnd.ms-excel": - return ".xls" - case "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet": - return ".xlsx" - case "application/vnd.openxmlformats-officedocument.wordprocessingml.document": - return ".docx" - case "application/vnd.openxmlformats-officedocument.presentationml.presentation": - return ".pptx" - // Audio formats - case "audio/mpeg", "audio/mp3": - return ".mp3" - case "audio/wav", "audio/x-wav", "audio/wave": - return ".wav" - case "audio/mp4", "audio/x-m4a": - return ".m4a" - case "audio/webm": - return ".webm" - case "audio/ogg": - return ".ogg" - case "audio/flac": - return ".flac" - default: - return ".bin" - } -} - -// saveFile saves the uploaded file to disk -func (h *UploadHandler) saveFile(src multipart.File, dst string) error { - // Create destination file - out, err := os.Create(dst) - if err != nil { - return err - } - defer out.Close() - - // Copy file contents - _, err = io.Copy(out, src) - return err -} - -// CheckFileStatus checks if a file is still available (not expired) -// This is used by the frontend to validate file references before workflow execution -func (h *UploadHandler) CheckFileStatus(c *fiber.Ctx) error { - // Get file ID from URL params - fileID := c.Params("id") - if fileID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "File ID required", - "available": false, - "expired": true, - }) - } - - // Check if user is authenticated (optional - for ownership validation) - userID, _ := c.Locals("user_id").(string) - - // Check file cache - if h.fileCache == nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "File cache service unavailable", - "available": false, - "expired": false, - }) - } - - // Try to get file from cache - var file *filecache.CachedFile - var err error - - if userID != "" && userID != "anonymous" { - // If authenticated, verify ownership - file, err = h.fileCache.GetByUser(fileID, userID) - } else { - // If not authenticated, just check existence - var found bool - file, found = h.fileCache.Get(fileID) - if !found { - err = fmt.Errorf("file not found or expired") - } - } - - if err != nil || file == nil { - // File not found or expired - log.Printf("⚠️ [UPLOAD] File status check - file not available: %s (user: %s)", fileID, userID) - return c.JSON(fiber.Map{ - "file_id": fileID, - "available": false, - "expired": true, - "error": "File not found or has expired. Files are only available for 30 minutes after upload.", - }) - } - - // File is available - log.Printf("✅ [UPLOAD] File status check - file available: %s (user: %s)", fileID, userID) - return c.JSON(fiber.Map{ - "file_id": fileID, - "available": true, - "expired": false, - "filename": file.Filename, - "mime_type": file.MimeType, - "size": file.Size, - }) -} - -// Delete removes an uploaded file -func (h *UploadHandler) Delete(c *fiber.Ctx) error { - // Check authentication - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" || userID == "anonymous" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - // Get file ID from URL params - fileID := c.Params("id") - if fileID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "File ID required", - }) - } - - // SECURITY: Validate fileID to prevent path traversal attacks - if err := security.ValidateFileID(fileID); err != nil { - log.Printf("⚠️ [UPLOAD] Invalid file ID in delete request: %v", err) - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid file ID format", - }) - } - - // Find file by ID (try all extensions) - var filePath string - extensions := []string{".jpg", ".jpeg", ".png", ".gif", ".webp"} - for _, ext := range extensions { - testPath := filepath.Join(h.uploadDir, fileID+ext) - if _, err := os.Stat(testPath); err == nil { - filePath = testPath - break - } - } - - if filePath == "" { - return c.Status(fiber.StatusNotFound).JSON(fiber.Map{ - "error": "File not found", - }) - } - - // Delete file - if err := os.Remove(filePath); err != nil { - log.Printf("❌ [UPLOAD] Failed to delete file %s: %v", filePath, err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to delete file", - }) - } - - log.Printf("🗑️ [UPLOAD] File deleted: %s (user: %s)", filePath, userID) - - return c.JSON(fiber.Map{ - "message": "File deleted successfully", - }) -} diff --git a/backend/internal/handlers/user.go b/backend/internal/handlers/user.go deleted file mode 100644 index f6b52024..00000000 --- a/backend/internal/handlers/user.go +++ /dev/null @@ -1,422 +0,0 @@ -package handlers - -import ( - "claraverse/internal/filecache" - "claraverse/internal/models" - "claraverse/internal/services" - "context" - "log" - "strings" - "time" - - "github.com/gofiber/fiber/v2" -) - -// UserHandler handles user data and GDPR compliance endpoints -type UserHandler struct { - chatService *services.ChatService - userService *services.UserService - agentService *services.AgentService - executionService *services.ExecutionService - apiKeyService *services.APIKeyService - credentialService *services.CredentialService - chatSyncService *services.ChatSyncService - schedulerService *services.SchedulerService - builderConvService *services.BuilderConversationService -} - -// NewUserHandler creates a new user handler -func NewUserHandler(chatService *services.ChatService, userService *services.UserService) *UserHandler { - return &UserHandler{ - chatService: chatService, - userService: userService, - } -} - -// SetGDPRServices sets optional services needed for complete GDPR deletion -func (h *UserHandler) SetGDPRServices( - agentService *services.AgentService, - executionService *services.ExecutionService, - apiKeyService *services.APIKeyService, - credentialService *services.CredentialService, - chatSyncService *services.ChatSyncService, - schedulerService *services.SchedulerService, - builderConvService *services.BuilderConversationService, -) { - h.agentService = agentService - h.executionService = executionService - h.apiKeyService = apiKeyService - h.credentialService = credentialService - h.chatSyncService = chatSyncService - h.schedulerService = schedulerService - h.builderConvService = builderConvService -} - -// GetPreferences returns user preferences -// GET /api/user/preferences -func (h *UserHandler) GetPreferences(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - if h.userService == nil { - // Fallback if MongoDB not configured - return c.JSON(models.UserPreferences{ - StoreBuilderChatHistory: true, - DefaultModelID: "", - }) - } - - prefs, err := h.userService.GetPreferences(c.Context(), userID) - if err != nil { - log.Printf("⚠️ Failed to get preferences for user %s: %v", userID, err) - // Return defaults on error - return c.JSON(models.UserPreferences{ - StoreBuilderChatHistory: true, - DefaultModelID: "", - }) - } - - return c.JSON(prefs) -} - -// UpdatePreferences updates user preferences -// PUT /api/user/preferences -func (h *UserHandler) UpdatePreferences(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - if h.userService == nil { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "User service not available", - }) - } - - var req models.UpdateUserPreferencesRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - prefs, err := h.userService.UpdatePreferences(c.Context(), userID, &req) - if err != nil { - log.Printf("❌ Failed to update preferences for user %s: %v", userID, err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to update preferences", - }) - } - - log.Printf("✅ Updated preferences for user %s", userID) - return c.JSON(prefs) -} - -// MarkWelcomePopupSeen marks the welcome popup as seen -// POST /api/user/welcome-popup-seen -func (h *UserHandler) MarkWelcomePopupSeen(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - if h.userService == nil { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "User service not available", - }) - } - - err := h.userService.MarkWelcomePopupSeen(c.Context(), userID) - if err != nil { - log.Printf("❌ Failed to mark welcome popup seen for user %s: %v", userID, err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to update popup status", - }) - } - - log.Printf("✅ Welcome popup marked as seen for user %s", userID) - return c.JSON(fiber.Map{ - "success": true, - "message": "Welcome popup marked as seen", - }) -} - -// ExportData exports all user data (GDPR Article 15 & 20 - Right to Access and Portability) -// GET /api/user/data -func (h *UserHandler) ExportData(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - userEmail, _ := c.Locals("user_email").(string) - - log.Printf("📦 [GDPR] Data export requested by user: %s", userID) - - // Get all conversations for this user - conversations, err := h.chatService.GetAllConversationsByUser(userID) - if err != nil { - log.Printf("❌ Failed to export conversations: %v", err) - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to export data", - }) - } - - // Get file metadata (uploaded files) - fileCache := filecache.GetService() - fileMetadata := fileCache.GetAllFilesByUser(userID) - - // Compile all user data - exportData := fiber.Map{ - "user_id": userID, - "user_email": userEmail, - "export_date": time.Now().Format(time.RFC3339), - "conversations": conversations, - "uploaded_files": fileMetadata, - "data_categories": []string{ - "conversations", - "messages", - "uploaded_files", - "user_profile", - }, - "privacy_notice": "This export contains all personal data we have stored for your account.", - } - - log.Printf("✅ [GDPR] Data exported for user %s: %d conversations, %d files", - userID, len(conversations), len(fileMetadata)) - - return c.JSON(exportData) -} - -// DeleteAccountRequest is the request body for account deletion -type DeleteAccountRequest struct { - Confirmation string `json:"confirmation"` -} - -// DeleteAccount deletes all user data (GDPR Article 17 - Right to Erasure) -// DELETE /api/user/account -// Requires confirmation phrase: "delete my account" -func (h *UserHandler) DeleteAccount(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - // Parse and validate confirmation phrase - var req DeleteAccountRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // Validate confirmation phrase (case-insensitive) - if strings.TrimSpace(strings.ToLower(req.Confirmation)) != "delete my account" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid confirmation phrase", - "required": "delete my account", - }) - } - - userEmail, _ := c.Locals("user_email").(string) - ctx := context.Background() - - log.Printf("🗑️ [GDPR] Account deletion CONFIRMED by user: %s (%s)", userID, userEmail) - - // Track deletion results - deletionResults := fiber.Map{} - - // 1. Delete schedules first (they reference agents) - if h.schedulerService != nil { - count, err := h.schedulerService.DeleteAllByUser(ctx, userID) - if err != nil { - log.Printf("⚠️ [GDPR] Failed to delete schedules: %v", err) - } else { - deletionResults["schedules"] = count - } - } - - // 2. Delete executions (they reference agents) - if h.executionService != nil { - count, err := h.executionService.DeleteAllByUser(ctx, userID) - if err != nil { - log.Printf("⚠️ [GDPR] Failed to delete executions: %v", err) - } else { - deletionResults["executions"] = count - } - } - - // 3. Delete agents, workflows, and workflow versions - if h.agentService != nil { - count, err := h.agentService.DeleteAllByUser(ctx, userID) - if err != nil { - log.Printf("⚠️ [GDPR] Failed to delete agents: %v", err) - } else { - deletionResults["agents"] = count - } - } - - // 4. Delete API keys - if h.apiKeyService != nil { - count, err := h.apiKeyService.DeleteAllByUser(ctx, userID) - if err != nil { - log.Printf("⚠️ [GDPR] Failed to delete API keys: %v", err) - } else { - deletionResults["api_keys"] = count - } - } - - // 5. Delete credentials - if h.credentialService != nil { - count, err := h.credentialService.DeleteAllByUser(ctx, userID) - if err != nil { - log.Printf("⚠️ [GDPR] Failed to delete credentials: %v", err) - } else { - deletionResults["credentials"] = count - } - } - - // 6. Delete cloud-synced chats - if h.chatSyncService != nil { - count, err := h.chatSyncService.DeleteAllUserChats(ctx, userID) - if err != nil { - log.Printf("⚠️ [GDPR] Failed to delete synced chats: %v", err) - } else { - deletionResults["synced_chats"] = count - } - } - - // 7. Delete builder conversations - if h.builderConvService != nil { - err := h.builderConvService.DeleteConversationsByUser(ctx, userID) - if err != nil { - log.Printf("⚠️ [GDPR] Failed to delete builder conversations: %v", err) - } else { - deletionResults["builder_conversations"] = "deleted" - } - } - - // 8. Delete SQL conversations - if err := h.chatService.DeleteAllConversationsByUser(userID); err != nil { - log.Printf("⚠️ [GDPR] Failed to delete conversations: %v", err) - } else { - deletionResults["conversations"] = "deleted" - } - - // 9. Delete all uploaded files - fileCache := filecache.GetService() - deletedFiles, err := fileCache.DeleteAllFilesByUser(userID) - if err != nil { - log.Printf("⚠️ [GDPR] Failed to delete some files: %v", err) - } - deletionResults["files"] = deletedFiles - - // 10. Delete user record (last) - if h.userService != nil { - err := h.userService.DeleteUser(ctx, userID) - if err != nil { - log.Printf("⚠️ [GDPR] Failed to delete user record: %v", err) - } else { - deletionResults["user_record"] = "deleted" - } - } - - log.Printf("✅ [GDPR] Account deletion completed for user %s", userID) - - return c.JSON(fiber.Map{ - "message": "Account and all associated data deleted successfully", - "user_id": userID, - "deletion_timestamp": time.Now().Format(time.RFC3339), - "deleted": deletionResults, - "retention_note": "Audit logs may be retained for up to 90 days for security purposes.", - }) -} - -// GetPrivacyPolicy returns privacy policy information (GDPR Article 13 - Transparency) -// GET /api/privacy-policy -func (h *UserHandler) GetPrivacyPolicy(c *fiber.Ctx) error { - policy := fiber.Map{ - "service_name": "ClaraVerse", - "last_updated": "2025-11-18", - - "data_collected": []string{ - "User ID (from authentication provider)", - "Email address (from authentication provider)", - "Chat messages and conversation history", - "Uploaded files (images, PDFs, CSV, Excel, JSON, text files)", - "Usage metadata (timestamps, model selections)", - }, - - "legal_basis": "Legitimate interest in providing the service (GDPR Article 6(1)(f))", - - "data_retention": fiber.Map{ - "conversations": "30 minutes (automatic deletion)", - "uploaded_files": "Linked to conversation lifetime (30 minutes) - includes images, PDFs, CSV, Excel, JSON, and all data files", - "audit_logs": "90 days (security and compliance purposes)", - }, - - "third_parties": []fiber.Map{ - { - "name": "Supabase", - "purpose": "Authentication and user management", - "data": []string{"user_id", "email", "authentication tokens"}, - }, - { - "name": "AI Model Providers", - "purpose": "Processing chat messages and generating responses", - "data": []string{"chat messages", "uploaded file content"}, - "note": "Varies based on selected model provider (OpenAI, Anthropic, etc.)", - }, - }, - - "user_rights": []fiber.Map{ - { - "right": "Right to Access (Art. 15)", - "description": "Download all your personal data", - "endpoint": "GET /api/user/data", - }, - { - "right": "Right to Erasure (Art. 17)", - "description": "Delete all your personal data", - "endpoint": "DELETE /api/user/account", - }, - { - "right": "Right to Data Portability (Art. 20)", - "description": "Export your data in machine-readable format (JSON)", - "endpoint": "GET /api/user/data", - }, - }, - - "security_measures": []string{ - "AES-256-GCM encryption for sensitive file content", - "JWT-based authentication", - "HTTPS encryption in transit (production)", - "Automatic data expiration (30 minutes)", - "Rate limiting and DDoS protection", - }, - - "contact": fiber.Map{ - "data_controller": "ClaraVerse Team", - "email": "privacy@claraverse.com", - "note": "For privacy inquiries, data requests, or to exercise your rights", - }, - - "cookie_policy": "This service uses minimal cookies for authentication purposes only.", - - "changes_to_policy": "We will notify users of significant changes via email or in-app notification.", - } - - return c.JSON(policy) -} diff --git a/backend/internal/handlers/user_preferences.go b/backend/internal/handlers/user_preferences.go deleted file mode 100644 index f54469c5..00000000 --- a/backend/internal/handlers/user_preferences.go +++ /dev/null @@ -1,89 +0,0 @@ -package handlers - -import ( - "claraverse/internal/models" - "claraverse/internal/services" - - "github.com/gofiber/fiber/v2" -) - -// UserPreferencesHandler handles user preferences HTTP requests -type UserPreferencesHandler struct { - userService *services.UserService -} - -// NewUserPreferencesHandler creates a new UserPreferencesHandler -func NewUserPreferencesHandler(userService *services.UserService) *UserPreferencesHandler { - return &UserPreferencesHandler{userService: userService} -} - -// Get retrieves user preferences -// GET /api/preferences -func (h *UserPreferencesHandler) Get(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - // Get email from context (set by auth middleware) - email, _ := c.Locals("user_email").(string) - - // Try to sync user first (creates if not exists) - _, err := h.userService.SyncUserFromSupabase(c.Context(), userID, email) - if err != nil { - // Log but don't fail - try to get preferences anyway - println("Warning: Failed to sync user:", err.Error()) - } - - prefs, err := h.userService.GetPreferences(c.Context(), userID) - if err != nil { - // Return default preferences if user not found - return c.JSON(models.UserPreferences{ - StoreBuilderChatHistory: true, - ChatPrivacyMode: "", - Theme: "dark", - FontSize: "medium", - }) - } - - return c.JSON(prefs) -} - -// Update updates user preferences -// PUT /api/preferences -func (h *UserPreferencesHandler) Update(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - var req models.UpdateUserPreferencesRequest - if err := c.BodyParser(&req); err != nil { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid request body", - }) - } - - // Validate chat privacy mode if provided - if req.ChatPrivacyMode != nil { - mode := *req.ChatPrivacyMode - if mode != models.ChatPrivacyModeLocal && mode != models.ChatPrivacyModeCloud && mode != "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid chat_privacy_mode. Must be 'local' or 'cloud'", - }) - } - } - - prefs, err := h.userService.UpdatePreferences(c.Context(), userID, &req) - if err != nil { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to update preferences", - }) - } - - return c.JSON(prefs) -} diff --git a/backend/internal/handlers/webhook.go b/backend/internal/handlers/webhook.go deleted file mode 100644 index caa80451..00000000 --- a/backend/internal/handlers/webhook.go +++ /dev/null @@ -1,93 +0,0 @@ -package handlers - -import ( - "claraverse/internal/services" - "context" - "log" - "net/http" - "strings" - - "github.com/gofiber/fiber/v2" -) - -// WebhookHandler handles DodoPayments webhooks -type WebhookHandler struct { - paymentService *services.PaymentService -} - -// NewWebhookHandler creates a new webhook handler -func NewWebhookHandler(paymentService *services.PaymentService) *WebhookHandler { - return &WebhookHandler{ - paymentService: paymentService, - } -} - -// HandleDodoWebhook handles incoming webhooks from DodoPayments -// POST /api/webhooks/dodo -// DodoPayments uses Standard Webhooks format with headers: -// - webhook-id: unique message ID -// - webhook-signature: v1, -// - webhook-timestamp: unix timestamp -func (h *WebhookHandler) HandleDodoWebhook(c *fiber.Ctx) error { - // Get payload - payload := c.Body() - if len(payload) == 0 { - log.Printf("❌ Webhook missing payload") - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Missing payload", - }) - } - - // Convert Fiber headers to http.Header for SDK - headers := make(http.Header) - c.Request().Header.VisitAll(func(key, value []byte) { - headers.Add(string(key), string(value)) - }) - - // Verify and parse webhook using SDK - event, err := h.paymentService.VerifyAndParseWebhook(payload, headers) - if err != nil { - log.Printf("❌ Webhook verification failed: %v", err) - - // Distinguish between parse errors (400) and auth errors (401) - if strings.Contains(err.Error(), "parse") || strings.Contains(err.Error(), "invalid character") { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Invalid payload format: " + err.Error(), - }) - } - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Invalid webhook: " + err.Error(), - }) - } - - // Handle event - ctx := context.Background() - if err := h.paymentService.HandleWebhookEvent(ctx, event); err != nil { - log.Printf("❌ Webhook processing error: %v", err) - - // Return 200 for idempotency errors (duplicate events) - no retry needed - if strings.Contains(err.Error(), "already processed") || - strings.Contains(err.Error(), "duplicate") { - log.Printf("⚠️ Duplicate webhook event %s (ID: %s) - already processed", event.Type, event.ID) - return c.Status(fiber.StatusOK).JSON(fiber.Map{ - "received": true, - "message": "Event already processed (idempotent)", - }) - } - - // Return 500 for actual processing failures to allow DodoPayments to retry - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Failed to process webhook", - "event_id": event.ID, - "type": event.Type, - }) - } - - log.Printf("✅ Webhook event processed: %s (ID: %s)", event.Type, event.ID) - return c.Status(fiber.StatusOK).JSON(fiber.Map{ - "received": true, - "event_id": event.ID, - "type": event.Type, - }) -} - diff --git a/backend/internal/handlers/webhook_test.go b/backend/internal/handlers/webhook_test.go deleted file mode 100644 index 16066b0b..00000000 --- a/backend/internal/handlers/webhook_test.go +++ /dev/null @@ -1,166 +0,0 @@ -package handlers - -import ( - "bytes" - "claraverse/internal/services" - "crypto/hmac" - "crypto/sha256" - "encoding/hex" - "net/http/httptest" - "testing" - - "github.com/gofiber/fiber/v2" -) - -func TestWebhookHandler_InvalidSignature(t *testing.T) { - app := fiber.New() - paymentService := services.NewPaymentService("", "secret123", "", nil, nil, nil) - handler := NewWebhookHandler(paymentService) - - app.Post("/api/webhooks/dodo", handler.HandleDodoWebhook) - - payload := []byte(`{"type":"subscription.active"}`) - req := httptest.NewRequest("POST", "/api/webhooks/dodo", bytes.NewBuffer(payload)) - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Webhook-Signature", "invalid_signature") - - resp, _ := app.Test(req) - - if resp.StatusCode != fiber.StatusUnauthorized { - t.Errorf("Expected 401, got %d", resp.StatusCode) - } -} - -func TestWebhookHandler_ValidSignature(t *testing.T) { - secret := "webhook_secret_123" - app := fiber.New() - paymentService := services.NewPaymentService("", secret, "", nil, nil, nil) - handler := NewWebhookHandler(paymentService) - - app.Post("/api/webhooks/dodo", handler.HandleDodoWebhook) - - payload := []byte(`{"type":"subscription.active","data":{"subscription_id":"sub_123"}}`) - - mac := hmac.New(sha256.New, []byte(secret)) - mac.Write(payload) - signature := hex.EncodeToString(mac.Sum(nil)) - - req := httptest.NewRequest("POST", "/api/webhooks/dodo", bytes.NewBuffer(payload)) - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Webhook-Signature", signature) - - resp, _ := app.Test(req) - - if resp.StatusCode != fiber.StatusOK { - t.Errorf("Expected 200, got %d", resp.StatusCode) - } -} - -func TestWebhookHandler_MissingSignature(t *testing.T) { - app := fiber.New() - paymentService := services.NewPaymentService("", "secret", "", nil, nil, nil) - handler := NewWebhookHandler(paymentService) - - app.Post("/api/webhooks/dodo", handler.HandleDodoWebhook) - - payload := []byte(`{"type":"subscription.active"}`) - req := httptest.NewRequest("POST", "/api/webhooks/dodo", bytes.NewBuffer(payload)) - req.Header.Set("Content-Type", "application/json") - // No signature header - - resp, _ := app.Test(req) - - if resp.StatusCode != fiber.StatusUnauthorized { - t.Errorf("Expected 401, got %d", resp.StatusCode) - } -} - -func TestWebhookHandler_InvalidJSON(t *testing.T) { - secret := "webhook_secret" - app := fiber.New() - paymentService := services.NewPaymentService("", secret, "", nil, nil, nil) - handler := NewWebhookHandler(paymentService) - - app.Post("/api/webhooks/dodo", handler.HandleDodoWebhook) - - payload := []byte(`{invalid json}`) - - mac := hmac.New(sha256.New, []byte(secret)) - mac.Write(payload) - signature := hex.EncodeToString(mac.Sum(nil)) - - req := httptest.NewRequest("POST", "/api/webhooks/dodo", bytes.NewBuffer(payload)) - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Webhook-Signature", signature) - - resp, _ := app.Test(req) - - if resp.StatusCode != fiber.StatusBadRequest { - t.Errorf("Expected 400, got %d", resp.StatusCode) - } -} - -func TestWebhookHandler_AllEventTypes(t *testing.T) { - secret := "webhook_secret" - app := fiber.New() - paymentService := services.NewPaymentService("", secret, "", nil, nil, nil) - handler := NewWebhookHandler(paymentService) - - app.Post("/api/webhooks/dodo", handler.HandleDodoWebhook) - - eventTypes := []string{ - "subscription.active", - "subscription.updated", - "subscription.on_hold", - "subscription.renewed", - "subscription.cancelled", - "payment.succeeded", - "payment.failed", - } - - for _, eventType := range eventTypes { - t.Run(eventType, func(t *testing.T) { - payload := []byte(`{"type":"` + eventType + `","data":{},"id":"evt_123"}`) - - mac := hmac.New(sha256.New, []byte(secret)) - mac.Write(payload) - signature := hex.EncodeToString(mac.Sum(nil)) - - req := httptest.NewRequest("POST", "/api/webhooks/dodo", bytes.NewBuffer(payload)) - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Webhook-Signature", signature) - - resp, _ := app.Test(req) - - if resp.StatusCode != fiber.StatusOK { - t.Errorf("Event %s: expected 200, got %d", eventType, resp.StatusCode) - } - }) - } -} - -func TestWebhookHandler_AlternativeSignatureHeader(t *testing.T) { - secret := "webhook_secret" - app := fiber.New() - paymentService := services.NewPaymentService("", secret, "", nil, nil, nil) - handler := NewWebhookHandler(paymentService) - - app.Post("/api/webhooks/dodo", handler.HandleDodoWebhook) - - payload := []byte(`{"type":"subscription.active","data":{},"id":"evt_123"}`) - - mac := hmac.New(sha256.New, []byte(secret)) - mac.Write(payload) - signature := hex.EncodeToString(mac.Sum(nil)) - - req := httptest.NewRequest("POST", "/api/webhooks/dodo", bytes.NewBuffer(payload)) - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Dodo-Signature", signature) // Alternative header name - - resp, _ := app.Test(req) - - if resp.StatusCode != fiber.StatusOK { - t.Errorf("Expected 200 with Dodo-Signature header, got %d", resp.StatusCode) - } -} - diff --git a/backend/internal/handlers/websocket.go b/backend/internal/handlers/websocket.go deleted file mode 100644 index 03dff6ae..00000000 --- a/backend/internal/handlers/websocket.go +++ /dev/null @@ -1,881 +0,0 @@ -package handlers - -import ( - "claraverse/internal/filecache" - "claraverse/internal/models" - "claraverse/internal/services" - "claraverse/internal/utils" - "context" - "encoding/json" - "fmt" - "log" - "os" - "path/filepath" - "strings" - "sync" - "time" - - "github.com/gofiber/contrib/websocket" - "github.com/google/uuid" -) - -// PromptResponse stores a user's response to an interactive prompt -type PromptResponse struct { - PromptID string - UserID string - Answers map[string]models.InteractiveAnswer - Skipped bool - ReceivedAt time.Time -} - -// PromptResponseCache stores prompt responses waiting to be processed -type PromptResponseCache struct { - responses map[string]*PromptResponse // promptID -> response - mutex sync.RWMutex -} - -// WebSocketHandler handles WebSocket connections -type WebSocketHandler struct { - connManager *services.ConnectionManager - chatService *services.ChatService - analyticsService *services.AnalyticsService // Optional: minimal usage tracking - usageLimiter *services.UsageLimiterService - promptCache *PromptResponseCache // Cache for interactive prompt responses -} - -// NewWebSocketHandler creates a new WebSocket handler -func NewWebSocketHandler(connManager *services.ConnectionManager, chatService *services.ChatService, analyticsService *services.AnalyticsService, usageLimiter *services.UsageLimiterService) *WebSocketHandler { - return &WebSocketHandler{ - connManager: connManager, - chatService: chatService, - analyticsService: analyticsService, - usageLimiter: usageLimiter, - promptCache: &PromptResponseCache{ - responses: make(map[string]*PromptResponse), - }, - } -} - -// Handle handles a new WebSocket connection -func (h *WebSocketHandler) Handle(c *websocket.Conn) { - connID := uuid.New().String() - userID := c.Locals("user_id").(string) - - // Create a done channel to signal goroutines to stop - done := make(chan struct{}) - - userConn := &models.UserConnection{ - ConnID: connID, - UserID: userID, - Conn: c, - ConversationID: "", - Messages: make([]map[string]interface{}, 0), - MessageCount: 0, - CreatedAt: time.Now(), - WriteChan: make(chan models.ServerMessage, 100), - StopChan: make(chan bool, 1), - // Create a waiter function that tools can use to wait for prompt responses - PromptWaiter: func(promptID string, timeout time.Duration) (map[string]models.InteractiveAnswer, bool, error) { - response, err := h.WaitForPromptResponse(promptID, timeout) - if err != nil { - return nil, false, err - } - return response.Answers, response.Skipped, nil - }, - } - - h.connManager.Add(userConn) - defer func() { - close(done) // Signal all goroutines to stop - h.connManager.Remove(connID) - - // Track session end (minimal analytics) - if h.analyticsService != nil && userConn.ConversationID != "" { - ctx := context.Background() - h.analyticsService.TrackChatSessionEnd(ctx, connID, userConn.MessageCount) - } - }() - - // Configure WebSocket timeouts for long-running operations - // Set read deadline to 6 minutes (allows for 5 min tool execution + buffer) - c.SetReadDeadline(time.Now().Add(360 * time.Second)) - - // Set up ping/pong handlers to keep connection alive during long tool executions - c.SetPongHandler(func(appData string) error { - // Reset read deadline on pong received - c.SetReadDeadline(time.Now().Add(360 * time.Second)) - return nil - }) - - // Start ping goroutine to keep connection alive - go h.pingLoop(userConn, done) - - // Start write goroutine - go h.writeLoop(userConn) - - // Send connected message (no conversation_id - that comes from client) - userConn.WriteChan <- models.ServerMessage{ - Type: "connected", - Content: "WebSocket connected. Ready to receive messages.", - } - - // Read loop - h.readLoop(userConn) -} - -// pingLoop sends periodic pings to keep the WebSocket connection alive -// This is crucial for long-running tool executions (e.g., Python runner up to 5 min) -func (h *WebSocketHandler) pingLoop(userConn *models.UserConnection, done <-chan struct{}) { - ticker := time.NewTicker(30 * time.Second) // Send ping every 30 seconds - defer ticker.Stop() - - for { - select { - case <-done: - return - case <-ticker.C: - userConn.Mutex.Lock() - if err := userConn.Conn.WriteControl(websocket.PingMessage, []byte{}, time.Now().Add(10*time.Second)); err != nil { - log.Printf("⚠️ Ping failed for %s: %v", userConn.ConnID, err) - userConn.Mutex.Unlock() - return - } - userConn.Mutex.Unlock() - } - } -} - -// readLoop handles incoming messages from the client -func (h *WebSocketHandler) readLoop(userConn *models.UserConnection) { - defer func() { - if r := recover(); r != nil { - log.Printf("❌ Panic in readLoop: %v", r) - } - }() - - for { - _, msg, err := userConn.Conn.ReadMessage() - if err != nil { - log.Printf("❌ WebSocket read error for %s: %v", userConn.ConnID, err) - break - } - - // Reset read deadline after successful read - userConn.Conn.SetReadDeadline(time.Now().Add(360 * time.Second)) - - var clientMsg models.ClientMessage - if err := json.Unmarshal(msg, &clientMsg); err != nil { - log.Printf("⚠️ Invalid message format from %s: %v", userConn.ConnID, err) - userConn.WriteChan <- models.ServerMessage{ - Type: "error", - ErrorCode: "invalid_format", - ErrorMessage: "Invalid message format", - } - continue - } - - switch clientMsg.Type { - case "ping": - // Respond to client heartbeat immediately - userConn.WriteChan <- models.ServerMessage{ - Type: "pong", - } - case "chat_message": - h.handleChatMessage(userConn, clientMsg) - case "new_conversation": - h.handleNewConversation(userConn, clientMsg) - case "stop_generation": - h.handleStopGeneration(userConn) - case "resume_stream": - h.handleResumeStream(userConn, clientMsg) - case "interactive_prompt_response": - h.handleInteractivePromptResponse(userConn, clientMsg) - default: - log.Printf("⚠️ Unknown message type: %s", clientMsg.Type) - } - } -} - -// handleChatMessage handles a chat message from the client -func (h *WebSocketHandler) handleChatMessage(userConn *models.UserConnection, clientMsg models.ClientMessage) { - // Update conversation ID if provided - if clientMsg.ConversationID != "" { - userConn.ConversationID = clientMsg.ConversationID - } - - // Update model ID if provided (platform model selection) - if clientMsg.ModelID != "" { - userConn.ModelID = clientMsg.ModelID - log.Printf("🎯 Model selected for %s: %s", userConn.ConnID, clientMsg.ModelID) - } - - // Update custom config if provided (BYOK) - if clientMsg.CustomConfig != nil { - userConn.CustomConfig = clientMsg.CustomConfig - log.Printf("🔑 BYOK config updated for %s: Model=%s", - userConn.ConnID, clientMsg.CustomConfig.Model) - } - - // Update system instructions if provided (per-request override) - if clientMsg.SystemInstructions != "" { - userConn.SystemInstructions = clientMsg.SystemInstructions - log.Printf("📝 System instructions updated for %s (length: %d chars)", - userConn.ConnID, len(clientMsg.SystemInstructions)) - } - - // Update disable tools flag (e.g., for agent builder) - userConn.DisableTools = clientMsg.DisableTools - if userConn.DisableTools { - log.Printf("🔒 Tools disabled for %s (agent builder mode)", userConn.ConnID) - } - - // Priority-based history handling: prefer backend cache, fall back to client history - userConn.Mutex.Lock() - - // Step 1: Try to get messages from backend cache first - var cachedMessages []map[string]interface{} - if userConn.ConversationID != "" { - cachedMessages = h.chatService.GetConversationMessages(userConn.ConversationID) - } - - if len(cachedMessages) > 0 { - // ✅ Cache HIT - backend has valid cache, use it (ignore client history) - userConn.Messages = cachedMessages - - // Count assistant messages from cache - assistantCount := 0 - for _, msg := range cachedMessages { - if role, ok := msg["role"].(string); ok && role == "assistant" { - assistantCount++ - } - } - userConn.MessageCount = assistantCount - - log.Printf("✅ [CACHE-HIT] Using backend cache for %s: %d messages (%d assistant)", - userConn.ConversationID, len(cachedMessages), assistantCount) - - } else if len(clientMsg.History) > 0 { - // ❌ Cache MISS - no backend cache, use client history and repopulate - userConn.Messages = clientMsg.History - - // Count assistant messages from client history - assistantCount := 0 - for _, msg := range clientMsg.History { - if role, ok := msg["role"].(string); ok && role == "assistant" { - assistantCount++ - } - } - userConn.MessageCount = assistantCount - - log.Printf("♻️ [CACHE-MISS] Recreating from client history for %s: %d messages (%d assistant)", - userConn.ConversationID, len(clientMsg.History), assistantCount) - - // Repopulate backend cache from client history - if userConn.ConversationID != "" { - h.chatService.SetConversationMessages(userConn.ConversationID, clientMsg.History) - } - - } else { - // 🆕 New conversation - no cache, no history - userConn.Messages = make([]map[string]interface{}, 0) - userConn.MessageCount = 0 - - log.Printf("🆕 [NEW-CONVERSATION] Starting fresh for %s", userConn.ConversationID) - - // Create conversation in database with ownership tracking - if userConn.ConversationID != "" { - if err := h.chatService.CreateConversation(userConn.ConversationID, userConn.UserID, "New Conversation"); err != nil { - log.Printf("⚠️ Failed to create conversation in database: %v", err) - // Continue anyway - conversation will work from cache - } - } - } - - // 🔍 DIAGNOSTIC: Log final state after history handling - log.Printf("🔍 [DIAGNOSTIC] After history handling - userConn.Messages count: %d, conversationID: %s", - len(userConn.Messages), userConn.ConversationID) - if len(userConn.Messages) > 0 { - firstMsg := userConn.Messages[0] - lastMsg := userConn.Messages[len(userConn.Messages)-1] - log.Printf("🔍 [DIAGNOSTIC] First message role: %v, Last message role: %v", - firstMsg["role"], lastMsg["role"]) - } - - userConn.Mutex.Unlock() - - // Add user message to conversation - userConn.Mutex.Lock() - - // Build message content based on whether there are attachments - var messageContent interface{} - var documentContext strings.Builder - var dataFileContext strings.Builder - var expiredFiles []string // Track expired files - - if len(clientMsg.Attachments) > 0 { - // Get file cache service - fileCache := filecache.GetService() - - // Process document attachments first (PDF, DOCX, PPTX) - for _, att := range clientMsg.Attachments { - // Check for document files (PDF, DOCX, PPTX) - isDocument := att.MimeType == "application/pdf" || - att.MimeType == "application/vnd.openxmlformats-officedocument.wordprocessingml.document" || - att.MimeType == "application/vnd.openxmlformats-officedocument.presentationml.presentation" - - if isDocument && att.FileID != "" { - // Fetch document text from cache - cachedFile, err := fileCache.GetByUserAndConversation( - att.FileID, - userConn.UserID, - userConn.ConversationID, - ) - - if err != nil { - log.Printf("⚠️ Failed to fetch document file %s: %v", att.FileID, err) - // Track expired file instead of returning error - expiredFiles = append(expiredFiles, att.Filename) - continue - } - - // Build document context - documentContext.WriteString(fmt.Sprintf("\n\n[Document: %s]\n", att.Filename)) - documentContext.WriteString(fmt.Sprintf("Pages: %d | Words: %d\n\n", cachedFile.PageCount, cachedFile.WordCount)) - documentContext.WriteString(cachedFile.ExtractedText.String()) - documentContext.WriteString("\n---\n") - - log.Printf("📄 Injected document context: %s (%d words) for %s", att.Filename, cachedFile.WordCount, userConn.ConnID) - } - } - - // Process CSV/Excel/JSON/Text data files (for context) - for _, att := range clientMsg.Attachments { - // Check if it's a data file (CSV, Excel, JSON, Text) - isDataFile := att.MimeType == "text/csv" || - att.MimeType == "text/plain" || - att.MimeType == "application/json" || - att.MimeType == "application/vnd.ms-excel" || - att.MimeType == "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet" - - if isDataFile && att.FileID != "" { - // Fetch data file from cache - cachedFile, err := fileCache.GetByUserAndConversation( - att.FileID, - userConn.UserID, - userConn.ConversationID, - ) - - if err != nil { - log.Printf("⚠️ Failed to fetch data file %s: %v", att.FileID, err) - // Track expired file instead of returning error - expiredFiles = append(expiredFiles, att.Filename) - continue - } - - // Read file content to get preview (first 10 lines for CSV/text) - uploadDir := os.Getenv("UPLOAD_DIR") - if uploadDir == "" { - uploadDir = "./uploads" - } - - fileContent, err := os.ReadFile(cachedFile.FilePath) - if err != nil { - log.Printf("⚠️ Failed to read data file %s: %v", att.FileID, err) - continue - } - - // Get first 10 lines as preview - lines := strings.Split(string(fileContent), "\n") - previewLines := 10 - if len(lines) < previewLines { - previewLines = len(lines) - } - preview := strings.Join(lines[:previewLines], "\n") - - // Build data file context - dataFileContext.WriteString(fmt.Sprintf("\n\n[Data File: %s]\n", att.Filename)) - dataFileContext.WriteString(fmt.Sprintf("File ID: %s\n", att.FileID)) - dataFileContext.WriteString(fmt.Sprintf("Type: %s | Size: %d bytes\n", att.MimeType, cachedFile.Size)) - dataFileContext.WriteString(fmt.Sprintf("\nPreview (first %d lines):\n", previewLines)) - dataFileContext.WriteString("```\n") - dataFileContext.WriteString(preview) - dataFileContext.WriteString("\n```\n") - dataFileContext.WriteString("---\n") - - log.Printf("📊 Injected data file context: %s (file_id: %s) for %s", att.Filename, att.FileID, userConn.ConnID) - } - } - - // Check if we have ACTUAL images (for vision models) - // CSV/Excel/JSON/Text files should NOT be treated as images - hasImages := false - imageRegistry := services.GetImageRegistryService() - for _, att := range clientMsg.Attachments { - // Only count as image if Type is "image" AND MimeType starts with "image/" - isActualImage := att.Type == "image" && strings.HasPrefix(att.MimeType, "image/") - log.Printf("📎 [ATTACHMENT] Type=%s, MimeType=%s, IsActualImage=%v, Filename=%s", - att.Type, att.MimeType, isActualImage, att.Filename) - if isActualImage { - hasImages = true - - // Register image in the image registry for LLM referencing - if att.FileID != "" && clientMsg.ConversationID != "" { - handle := imageRegistry.RegisterUploadedImage( - clientMsg.ConversationID, - att.FileID, - att.Filename, - 0, 0, // Width/height not available here, could be extracted from image if needed - ) - log.Printf("📸 [IMAGE-REGISTRY] Registered uploaded image as %s (file_id: %s)", handle, att.FileID) - } - } - } - - if hasImages { - // Vision model format - array of content parts - contentParts := []map[string]interface{}{} - - // Build combined text content with PDF and data file contexts - textContent := clientMsg.Content - if documentContext.Len() > 0 { - textContent = documentContext.String() + "\n" + textContent - } - if dataFileContext.Len() > 0 { - textContent = dataFileContext.String() + "\n" + textContent - } - // Add user query label - if documentContext.Len() > 0 || dataFileContext.Len() > 0 { - textContent = textContent + "\n\nUser query: " + clientMsg.Content - } - - contentParts = append(contentParts, map[string]interface{}{ - "type": "text", - "text": textContent, - }) - - // Add image attachments (only actual images, not CSV/data files) - imageUtils := utils.NewImageUtils() - for _, att := range clientMsg.Attachments { - // Only process actual images (not CSV/Excel/JSON disguised as images) - isActualImage := att.Type == "image" && strings.HasPrefix(att.MimeType, "image/") - if isActualImage { - var imageURL string - - // If URL is relative (starts with /uploads/), convert to base64 - if strings.HasPrefix(att.URL, "/uploads/") { - // Extract filename and build local path - filename := filepath.Base(att.URL) - localPath := filepath.Join("./uploads", filename) - - // Convert to base64 data URL - base64URL, err := imageUtils.EncodeToBase64(localPath) - if err != nil { - log.Printf("⚠️ Failed to encode image to base64: %v", err) - // Fall back to original URL - imageURL = att.URL - } else { - imageURL = base64URL - log.Printf("🔄 Converted local image to base64 (size: %d bytes)", att.Size) - } - } else { - // Already a full URL (http:// or https://) - imageURL = att.URL - } - - contentParts = append(contentParts, map[string]interface{}{ - "type": "image_url", - "image_url": map[string]interface{}{ - "url": imageURL, - }, - }) - } - } - - messageContent = contentParts - log.Printf("🖼️ Chat message from %s with %d attachment(s)", userConn.ConnID, len(clientMsg.Attachments)) - } else if documentContext.Len() > 0 || dataFileContext.Len() > 0 { - // Document/Data file message (no images) - var combinedContext strings.Builder - if documentContext.Len() > 0 { - combinedContext.WriteString(documentContext.String()) - } - if dataFileContext.Len() > 0 { - combinedContext.WriteString(dataFileContext.String()) - } - combinedContext.WriteString("\n\nUser query: ") - combinedContext.WriteString(clientMsg.Content) - messageContent = combinedContext.String() - } else { - // No usable attachments - messageContent = clientMsg.Content - } - } else { - // Text-only message - messageContent = clientMsg.Content - } - - userConn.Mutex.Unlock() - - // Check message limit before processing - if h.usageLimiter != nil { - ctx := context.Background() - if err := h.usageLimiter.CheckMessageLimit(ctx, userConn.UserID); err != nil { - if limitErr, ok := err.(*services.LimitExceededError); ok { - userConn.WriteChan <- models.ServerMessage{ - Type: "limit_exceeded", - ErrorCode: limitErr.ErrorCode, - ErrorMessage: limitErr.Message, - Arguments: map[string]interface{}{ - "limit": limitErr.Limit, - "used": limitErr.Used, - "reset_at": limitErr.ResetAt, - "upgrade_to": limitErr.UpgradeTo, - }, - } - log.Printf("⚠️ [LIMIT] Message limit exceeded for user %s: %s", userConn.UserID, limitErr.Message) - return - } - } - - // Increment message count (check passed) - go func() { - if err := h.usageLimiter.IncrementMessageCount(context.Background(), userConn.UserID); err != nil { - log.Printf("⚠️ [LIMIT] Failed to increment message count for user %s: %v", userConn.UserID, err) - } - }() - } - - // Add user message to conversation cache via ChatService - h.chatService.AddUserMessage(userConn.ConversationID, messageContent) - - log.Printf("💬 Chat message from %s (user: %s, length: %d chars)", - userConn.ConnID, userConn.UserID, len(clientMsg.Content)) - - // Send warning if any files have expired - if len(expiredFiles) > 0 { - warningMsg := fmt.Sprintf("⚠ Warning: %d file(s) expired and unavailable: %s", - len(expiredFiles), strings.Join(expiredFiles, ", ")) - log.Printf("⚠️ [FILE-EXPIRED] %s", warningMsg) - - userConn.WriteChan <- models.ServerMessage{ - Type: "files_expired", - ErrorCode: "files_expired", - ErrorMessage: warningMsg, - Content: strings.Join(expiredFiles, ", "), // File names as comma-separated string - } - } - - // Stream response - go func() { - if err := h.chatService.StreamChatCompletion(userConn); err != nil { - log.Printf("❌ Chat completion error: %v", err) - userConn.WriteChan <- models.ServerMessage{ - Type: "error", - ErrorCode: "chat_error", - ErrorMessage: err.Error(), - } - } - }() -} - -// handleNewConversation handles starting a new conversation (clears history) -func (h *WebSocketHandler) handleNewConversation(userConn *models.UserConnection, clientMsg models.ClientMessage) { - userConn.Mutex.Lock() - - // Clear all conversation history - userConn.Messages = make([]map[string]interface{}, 0) - userConn.MessageCount = 0 // Reset message counter for new conversation - - // Update conversation ID - if clientMsg.ConversationID != "" { - userConn.ConversationID = clientMsg.ConversationID - } else { - userConn.ConversationID = uuid.New().String() - } - - // Update model if provided - if clientMsg.ModelID != "" { - userConn.ModelID = clientMsg.ModelID - } - - // Update system instructions if provided - if clientMsg.SystemInstructions != "" { - userConn.SystemInstructions = clientMsg.SystemInstructions - } - - // Update custom config if provided - if clientMsg.CustomConfig != nil { - userConn.CustomConfig = clientMsg.CustomConfig - } - - userConn.Mutex.Unlock() - - // Clear conversation cache - h.chatService.ClearConversation(userConn.ConversationID) - - // Create conversation in database with ownership tracking - if err := h.chatService.CreateConversation(userConn.ConversationID, userConn.UserID, "New Conversation"); err != nil { - log.Printf("⚠️ Failed to create conversation in database: %v", err) - // Continue anyway - conversation will work from cache - } - - // Track chat session start (minimal analytics) - if h.analyticsService != nil { - ctx := context.Background() - h.analyticsService.TrackChatSessionStart(ctx, userConn.ConnID, userConn.UserID, userConn.ConversationID) - - // Update model info if available - if userConn.ModelID != "" { - h.analyticsService.UpdateChatSessionModel(ctx, userConn.ConnID, userConn.ModelID, userConn.DisableTools) - } - } - - log.Printf("🆕 New conversation started for %s: conversation_id=%s, model=%s", - userConn.ConnID, userConn.ConversationID, userConn.ModelID) - - // Send acknowledgment - userConn.WriteChan <- models.ServerMessage{ - Type: "conversation_reset", - ConversationID: userConn.ConversationID, - Content: "New conversation started", - } -} - -// handleStopGeneration handles a stop generation request -func (h *WebSocketHandler) handleStopGeneration(userConn *models.UserConnection) { - select { - case userConn.StopChan <- true: - log.Printf("⏹️ Stop signal sent for %s", userConn.ConnID) - default: - log.Printf("⚠️ Stop channel full or closed for %s", userConn.ConnID) - } -} - -// handleResumeStream handles a request to resume a disconnected stream -func (h *WebSocketHandler) handleResumeStream(userConn *models.UserConnection, clientMsg models.ClientMessage) { - conversationID := clientMsg.ConversationID - if conversationID == "" { - log.Printf("⚠️ Resume stream request with empty conversation ID from %s", userConn.ConnID) - userConn.WriteChan <- models.ServerMessage{ - Type: "error", - ErrorCode: "missing_conversation_id", - ErrorMessage: "Conversation ID is required for resume", - } - return - } - - log.Printf("🔄 [RESUME] Resume stream request for conversation %s from %s", conversationID, userConn.ConnID) - - // Get the stream buffer - streamBuffer := h.chatService.GetStreamBuffer() - bufferData, err := streamBuffer.GetBufferData(conversationID) - - if err != nil { - // Buffer not found or rate limited - log.Printf("⚠️ [RESUME] Buffer not available for %s: %v", conversationID, err) - userConn.WriteChan <- models.ServerMessage{ - Type: "stream_missed", - ConversationID: conversationID, - Reason: "expired", - } - return - } - - // Validate user owns this buffer - if bufferData.UserID != userConn.UserID { - log.Printf("⚠️ [RESUME] User %s attempted to resume buffer owned by %s", userConn.UserID, bufferData.UserID) - userConn.WriteChan <- models.ServerMessage{ - Type: "stream_missed", - ConversationID: conversationID, - Reason: "not_found", - } - return - } - - log.Printf("📦 [RESUME] Sending %d buffered chunks (%d bytes), %d pending messages for conversation %s (complete: %v)", - bufferData.ChunkCount, len(bufferData.CombinedChunks), len(bufferData.PendingMessages), conversationID, bufferData.IsComplete) - - // First, replay any pending messages (tool results with artifacts, etc.) - // These are critical messages that might have been missed during disconnect - for _, pendingMsg := range bufferData.PendingMessages { - // Skip already delivered messages (prevents duplicates on rapid reconnects) - if pendingMsg.Delivered { - continue - } - - log.Printf("📦 [RESUME] Replaying pending message type=%s tool=%s for conversation %s", - pendingMsg.Type, pendingMsg.ToolName, conversationID) - - // Convert BufferedMessage to ServerMessage with all fields - serverMsg := models.ServerMessage{ - Type: pendingMsg.Type, - ToolName: pendingMsg.ToolName, - ToolDisplayName: pendingMsg.ToolDisplayName, - ToolIcon: pendingMsg.ToolIcon, - ToolDescription: pendingMsg.ToolDescription, - Status: pendingMsg.Status, - Result: pendingMsg.Result, - } - - // Handle plots (for image artifacts) - if pendingMsg.Plots != nil { - if plots, ok := pendingMsg.Plots.([]models.PlotData); ok { - serverMsg.Plots = plots - } else { - log.Printf("⚠️ [RESUME] Failed to cast plots for %s - type: %T", pendingMsg.ToolName, pendingMsg.Plots) - } - } - - userConn.WriteChan <- serverMsg - } - - // Mark pending messages as delivered (prevents duplicates on next resume) - streamBuffer.MarkMessagesDelivered(conversationID) - - // Send the resume message with all buffered text content - if len(bufferData.CombinedChunks) > 0 { - userConn.WriteChan <- models.ServerMessage{ - Type: "stream_resume", - ConversationID: conversationID, - Content: bufferData.CombinedChunks, - IsComplete: bufferData.IsComplete, - } - } - - // If the stream is complete, also send stream_end - if bufferData.IsComplete { - userConn.WriteChan <- models.ServerMessage{ - Type: "stream_end", - ConversationID: conversationID, - } - // Clear the buffer since it's complete and delivered - streamBuffer.ClearBuffer(conversationID) - log.Printf("📦 [RESUME] Stream complete, buffer cleared for conversation %s", conversationID) - } else { - // Stream still in progress - update connection ID so new chunks go to this connection - // Note: The stream buffer continues to collect chunks from the ongoing generation - log.Printf("📦 [RESUME] Stream still in progress for conversation %s", conversationID) - } -} - -// handleInteractivePromptResponse handles a user's response to an interactive prompt -func (h *WebSocketHandler) handleInteractivePromptResponse(userConn *models.UserConnection, clientMsg models.ClientMessage) { - promptID := clientMsg.PromptID - if promptID == "" { - log.Printf("⚠️ Interactive prompt response with empty prompt ID from %s", userConn.ConnID) - userConn.WriteChan <- models.ServerMessage{ - Type: "error", - ErrorCode: "missing_prompt_id", - ErrorMessage: "Prompt ID is required", - } - return - } - - if clientMsg.Skipped { - log.Printf("📋 [PROMPT] User %s skipped prompt %s", userConn.UserID, promptID) - } else { - log.Printf("📋 [PROMPT] User %s answered prompt %s with %d answers", userConn.UserID, promptID, len(clientMsg.Answers)) - - // Log each answer for debugging - for questionID, answer := range clientMsg.Answers { - log.Printf(" Question %s: %v (is_other: %v)", questionID, answer.Value, answer.IsOther) - } - } - - // Store the response in cache for the waiting tool execution - h.promptCache.mutex.Lock() - h.promptCache.responses[promptID] = &PromptResponse{ - PromptID: promptID, - UserID: userConn.UserID, - Answers: clientMsg.Answers, - Skipped: clientMsg.Skipped, - ReceivedAt: time.Now(), - } - h.promptCache.mutex.Unlock() - - log.Printf("✅ [PROMPT] Prompt %s response stored in cache (waiting tool will receive it)", promptID) -} - -// SendInteractivePrompt sends an interactive prompt to the client -// This can be called from anywhere (e.g., during tool execution) to ask the user questions -func (h *WebSocketHandler) SendInteractivePrompt(userConn *models.UserConnection, prompt models.ServerMessage) bool { - if prompt.Type != "interactive_prompt" { - log.Printf("⚠️ SendInteractivePrompt called with invalid type: %s", prompt.Type) - return false - } - - if prompt.PromptID == "" { - log.Printf("⚠️ SendInteractivePrompt called with empty PromptID") - return false - } - - if len(prompt.Questions) == 0 { - log.Printf("⚠️ SendInteractivePrompt called with no questions") - return false - } - - // Set conversation ID - prompt.ConversationID = userConn.ConversationID - - // Send the prompt - success := userConn.SafeSend(prompt) - if success { - log.Printf("📋 [PROMPT] Sent interactive prompt %s with %d questions to user %s", - prompt.PromptID, len(prompt.Questions), userConn.UserID) - } else { - log.Printf("❌ [PROMPT] Failed to send interactive prompt %s to user %s", - prompt.PromptID, userConn.UserID) - } - - return success -} - -// WaitForPromptResponse waits for a user to respond to an interactive prompt -// Blocks until response is received or timeout occurs (default 5 minutes) -func (h *WebSocketHandler) WaitForPromptResponse(promptID string, timeout time.Duration) (*PromptResponse, error) { - deadline := time.Now().Add(timeout) - pollInterval := 100 * time.Millisecond - - log.Printf("⏳ [PROMPT] Waiting for response to prompt %s (timeout: %v)", promptID, timeout) - - for time.Now().Before(deadline) { - // Check if response exists - h.promptCache.mutex.RLock() - response, exists := h.promptCache.responses[promptID] - h.promptCache.mutex.RUnlock() - - if exists { - // Remove from cache - h.promptCache.mutex.Lock() - delete(h.promptCache.responses, promptID) - h.promptCache.mutex.Unlock() - - log.Printf("✅ [PROMPT] Received response for prompt %s after %.2f seconds", - promptID, time.Since(response.ReceivedAt.Add(-time.Since(response.ReceivedAt))).Seconds()) - return response, nil - } - - // Sleep before next poll - time.Sleep(pollInterval) - } - - // Timeout - log.Printf("⏱️ [PROMPT] Timeout waiting for response to prompt %s", promptID) - return nil, fmt.Errorf("timeout waiting for user response") -} - -// writeLoop handles outgoing messages to the client -func (h *WebSocketHandler) writeLoop(userConn *models.UserConnection) { - defer func() { - if r := recover(); r != nil { - log.Printf("❌ Panic in writeLoop: %v", r) - } - }() - - for msg := range userConn.WriteChan { - if err := userConn.Conn.WriteJSON(msg); err != nil { - log.Printf("❌ WebSocket write error for %s: %v", userConn.ConnID, err) - return - } - } -} diff --git a/backend/internal/handlers/workflow_websocket.go b/backend/internal/handlers/workflow_websocket.go deleted file mode 100644 index 5546181a..00000000 --- a/backend/internal/handlers/workflow_websocket.go +++ /dev/null @@ -1,315 +0,0 @@ -package handlers - -import ( - "claraverse/internal/execution" - "claraverse/internal/middleware" - "claraverse/internal/models" - "claraverse/internal/services" - "context" - "encoding/json" - "log" - "time" - - "github.com/gofiber/contrib/websocket" - "github.com/google/uuid" - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// WorkflowWebSocketHandler handles WebSocket connections for workflow execution -type WorkflowWebSocketHandler struct { - agentService *services.AgentService - executionService *services.ExecutionService - workflowEngine *execution.WorkflowEngine - executionLimiter *middleware.ExecutionLimiter -} - -// NewWorkflowWebSocketHandler creates a new workflow WebSocket handler -func NewWorkflowWebSocketHandler( - agentService *services.AgentService, - workflowEngine *execution.WorkflowEngine, - executionLimiter *middleware.ExecutionLimiter, -) *WorkflowWebSocketHandler { - return &WorkflowWebSocketHandler{ - agentService: agentService, - workflowEngine: workflowEngine, - executionLimiter: executionLimiter, - } -} - -// SetExecutionService sets the execution service (optional, for MongoDB execution tracking) -func (h *WorkflowWebSocketHandler) SetExecutionService(svc *services.ExecutionService) { - h.executionService = svc -} - -// WorkflowClientMessage represents a message from the client -type WorkflowClientMessage struct { - Type string `json:"type"` // execute_workflow, cancel_execution - AgentID string `json:"agent_id,omitempty"` - Input map[string]any `json:"input,omitempty"` - - // EnableBlockChecker enables block completion validation (optional) - // When true, each block is checked to ensure it accomplished its job - EnableBlockChecker bool `json:"enable_block_checker,omitempty"` - - // CheckerModelID is the model to use for block checking (optional) - // Defaults to gpt-4o-mini for fast, cheap validation - CheckerModelID string `json:"checker_model_id,omitempty"` -} - -// WorkflowServerMessage represents a message to send to the client -type WorkflowServerMessage struct { - Type string `json:"type"` // connected, execution_started, execution_update, execution_complete, error - ExecutionID string `json:"execution_id,omitempty"` - BlockID string `json:"block_id,omitempty"` - Status string `json:"status,omitempty"` - Inputs map[string]any `json:"inputs,omitempty"` - Output map[string]any `json:"output,omitempty"` - FinalOutput map[string]any `json:"final_output,omitempty"` - Duration int64 `json:"duration_ms,omitempty"` - Error string `json:"error,omitempty"` - - // APIResponse is the standardized, clean response for API consumers - // This provides a well-structured output with result, artifacts, files, etc. - APIResponse *models.ExecutionAPIResponse `json:"api_response,omitempty"` -} - -// Handle handles a new WebSocket connection for workflow execution -func (h *WorkflowWebSocketHandler) Handle(c *websocket.Conn) { - userID := c.Locals("user_id").(string) - connID := uuid.New().String() - - log.Printf("🔌 [WORKFLOW-WS] New connection: connID=%s, userID=%s", connID, userID) - - // Send connected message - if err := c.WriteJSON(WorkflowServerMessage{ - Type: "connected", - }); err != nil { - log.Printf("❌ [WORKFLOW-WS] Failed to send connected message: %v", err) - return - } - - // Context for cancellation - ctx, cancel := context.WithCancel(context.Background()) - defer cancel() - - // Read loop - for { - _, msg, err := c.ReadMessage() - if err != nil { - log.Printf("❌ [WORKFLOW-WS] Read error for %s: %v", connID, err) - break - } - - var clientMsg WorkflowClientMessage - if err := json.Unmarshal(msg, &clientMsg); err != nil { - log.Printf("⚠️ [WORKFLOW-WS] Invalid message format from %s: %v", connID, err) - c.WriteJSON(WorkflowServerMessage{ - Type: "error", - Error: "Invalid message format", - }) - continue - } - - switch clientMsg.Type { - case "execute_workflow": - h.handleExecuteWorkflow(ctx, c, userID, clientMsg) - case "cancel_execution": - cancel() - ctx, cancel = context.WithCancel(context.Background()) - default: - log.Printf("⚠️ [WORKFLOW-WS] Unknown message type: %s", clientMsg.Type) - } - } -} - -// handleExecuteWorkflow handles a workflow execution request -func (h *WorkflowWebSocketHandler) handleExecuteWorkflow( - ctx context.Context, - c *websocket.Conn, - userID string, - msg WorkflowClientMessage, -) { - startTime := time.Now() - - log.Printf("🔍 [WORKFLOW-WS] Received execute request: AgentID=%s, Input=%+v", msg.AgentID, msg.Input) - - // Check daily execution limit - if h.executionLimiter != nil { - remaining, err := h.executionLimiter.GetRemainingExecutions(userID) - if err != nil { - log.Printf("⚠️ [WORKFLOW-WS] Failed to check execution limit: %v", err) - // Continue on error, don't block execution - } else if remaining == 0 { - log.Printf("⚠️ [WORKFLOW-WS] User %s exceeded daily execution limit", userID) - c.WriteJSON(WorkflowServerMessage{ - Type: "error", - Error: "Daily execution limit exceeded. Please upgrade your plan or wait until tomorrow.", - }) - return - } else if remaining > 0 { - log.Printf("✅ [WORKFLOW-WS] User %s has %d executions remaining today", userID, remaining) - } - } - - // Get agent and workflow - agent, err := h.agentService.GetAgent(msg.AgentID, userID) - if err != nil { - log.Printf("❌ [WORKFLOW-WS] Agent not found: %s", msg.AgentID) - c.WriteJSON(WorkflowServerMessage{ - Type: "error", - Error: "Agent not found: " + err.Error(), - }) - return - } - - if agent.Workflow == nil { - log.Printf("❌ [WORKFLOW-WS] No workflow for agent: %s", msg.AgentID) - c.WriteJSON(WorkflowServerMessage{ - Type: "error", - Error: "Agent has no workflow defined", - }) - return - } - - // Create execution record using ExecutionService (MongoDB) if available - var execID string - var execObjectID primitive.ObjectID - - if h.executionService != nil { - execRecord, err := h.executionService.Create(ctx, &services.CreateExecutionRequest{ - AgentID: msg.AgentID, - UserID: userID, - WorkflowVersion: agent.Workflow.Version, - TriggerType: "manual", - Input: msg.Input, - }) - if err != nil { - log.Printf("❌ [WORKFLOW-WS] Failed to create execution: %v", err) - c.WriteJSON(WorkflowServerMessage{ - Type: "error", - Error: "Failed to create execution: " + err.Error(), - }) - return - } - execID = execRecord.ID.Hex() - execObjectID = execRecord.ID - } else { - // Fallback: generate a local ID if ExecutionService is not available - execID = uuid.New().String() - log.Printf("⚠️ [WORKFLOW-WS] ExecutionService not available, using local ID: %s", execID) - } - - log.Printf("🚀 [WORKFLOW-WS] Starting execution %s for agent %s", execID, msg.AgentID) - - // Send execution started message - c.WriteJSON(WorkflowServerMessage{ - Type: "execution_started", - ExecutionID: execID, - }) - - // Increment execution counter for today - if h.executionLimiter != nil { - if err := h.executionLimiter.IncrementCount(userID); err != nil { - log.Printf("⚠️ [WORKFLOW-WS] Failed to increment execution count: %v", err) - // Don't fail the execution if counter increment fails - } - } - - // Create status channel - statusChan := make(chan models.ExecutionUpdate, 100) - - // Start goroutine to forward status updates to WebSocket - go func() { - for update := range statusChan { - update.ExecutionID = execID - c.WriteJSON(WorkflowServerMessage{ - Type: "execution_update", - ExecutionID: execID, - BlockID: update.BlockID, - Status: update.Status, - Inputs: update.Inputs, - Output: update.Output, - Error: update.Error, - }) - } - }() - - // Inject user context into input for credential resolution and tool execution - if msg.Input == nil { - msg.Input = make(map[string]interface{}) - } - msg.Input["__user_id__"] = userID - - // Build execution options - block checker is controlled by client request - // When enabled, it validates that each block actually accomplished its job - execOptions := &execution.ExecutionOptions{ - WorkflowGoal: agent.Description, // Use agent description as workflow goal - EnableBlockChecker: msg.EnableBlockChecker, // Controlled by frontend toggle - CheckerModelID: msg.CheckerModelID, - } - if msg.EnableBlockChecker { - log.Printf("🔍 [WORKFLOW-WS] Block checker ENABLED (model: %s)", execOptions.CheckerModelID) - } else { - log.Printf("🔍 [WORKFLOW-WS] Block checker DISABLED") - } - - // Execute workflow - log.Printf("🔍 [WORKFLOW-WS] Executing with input: %+v", msg.Input) - result, err := h.workflowEngine.ExecuteWithOptions(ctx, agent.Workflow, msg.Input, statusChan, execOptions) - close(statusChan) - - duration := time.Since(startTime).Milliseconds() - - if err != nil { - log.Printf("❌ [WORKFLOW-WS] Execution failed: %v", err) - - // Update execution status using ExecutionService if available - if h.executionService != nil { - h.executionService.Complete(ctx, execObjectID, &services.ExecutionCompleteRequest{ - Status: "failed", - Error: err.Error(), - }) - } - - c.WriteJSON(WorkflowServerMessage{ - Type: "execution_complete", - ExecutionID: execID, - Status: "failed", - Duration: duration, - Error: err.Error(), - }) - return - } - - // Build the standardized API response - apiResponse := h.workflowEngine.BuildAPIResponse(result, agent.Workflow, execID, duration) - apiResponse.Metadata.AgentID = msg.AgentID - - // Update execution status in database using ExecutionService if available - if h.executionService != nil { - h.executionService.Complete(ctx, execObjectID, &services.ExecutionCompleteRequest{ - Status: result.Status, - Output: result.Output, - BlockStates: result.BlockStates, - Error: result.Error, - // Store clean API response fields - Result: apiResponse.Result, - Artifacts: apiResponse.Artifacts, - Files: apiResponse.Files, - }) - } - - log.Printf("✅ [WORKFLOW-WS] Execution %s completed: status=%s, duration=%dms, result=%d chars", - execID, result.Status, duration, len(apiResponse.Result)) - - // Send completion message with both legacy and new API response format - c.WriteJSON(WorkflowServerMessage{ - Type: "execution_complete", - ExecutionID: execID, - Status: result.Status, - FinalOutput: result.Output, // Legacy format (backward compat) - Duration: duration, - Error: result.Error, - APIResponse: apiResponse, // New standardized format - }) -} diff --git a/backend/internal/jobs/grace_period_checker.go b/backend/internal/jobs/grace_period_checker.go deleted file mode 100644 index 9993a442..00000000 --- a/backend/internal/jobs/grace_period_checker.go +++ /dev/null @@ -1,174 +0,0 @@ -package jobs - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "claraverse/internal/services" - "context" - "log" - "time" - - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/mongo" -) - -// GracePeriodChecker handles expiration of grace periods for ON_HOLD subscriptions -type GracePeriodChecker struct { - mongoDB *database.MongoDB - userService *services.UserService - tierService *services.TierService - gracePeriodDays int - subscriptions *mongo.Collection -} - -// NewGracePeriodChecker creates a new grace period checker -func NewGracePeriodChecker( - mongoDB *database.MongoDB, - userService *services.UserService, - tierService *services.TierService, - gracePeriodDays int, -) *GracePeriodChecker { - if gracePeriodDays == 0 { - gracePeriodDays = 7 // Default: 7 day grace period - } - - var subscriptions *mongo.Collection - if mongoDB != nil { - subscriptions = mongoDB.Database().Collection("subscriptions") - } - - return &GracePeriodChecker{ - mongoDB: mongoDB, - userService: userService, - tierService: tierService, - gracePeriodDays: gracePeriodDays, - subscriptions: subscriptions, - } -} - -// Run checks for expired grace periods and downgrades subscriptions -func (g *GracePeriodChecker) Run(ctx context.Context) error { - if g.mongoDB == nil || g.userService == nil || g.tierService == nil { - log.Println("⚠️ [GRACE-PERIOD] Grace period checker disabled (requires MongoDB, UserService, TierService)") - return nil - } - - log.Println("⏰ [GRACE-PERIOD] Checking for expired grace periods...") - startTime := time.Now() - - // Calculate cutoff date - cutoffDate := time.Now().UTC().AddDate(0, 0, -g.gracePeriodDays) - - // Find subscriptions that are ON_HOLD and past grace period - filter := bson.M{ - "status": models.SubStatusOnHold, - "updatedAt": bson.M{ - "$lt": cutoffDate, - }, - } - - cursor, err := g.subscriptions.Find(ctx, filter) - if err != nil { - log.Printf("❌ [GRACE-PERIOD] Failed to query subscriptions: %v", err) - return err - } - defer cursor.Close(ctx) - - expiredCount := 0 - for cursor.Next(ctx) { - var sub models.Subscription - if err := cursor.Decode(&sub); err != nil { - log.Printf("⚠️ [GRACE-PERIOD] Failed to decode subscription: %v", err) - continue - } - - if err := g.expireSubscription(ctx, &sub); err != nil { - log.Printf("⚠️ [GRACE-PERIOD] Failed to expire subscription %s: %v", sub.ID.Hex(), err) - continue - } - - expiredCount++ - log.Printf("✅ [GRACE-PERIOD] Expired subscription %s for user %s (on hold for %d days)", - sub.ID.Hex(), sub.UserID, g.gracePeriodDays) - } - - duration := time.Since(startTime) - log.Printf("✅ [GRACE-PERIOD] Check complete: expired %d subscriptions in %v", expiredCount, duration) - - return nil -} - -// expireSubscription downgrades a subscription after grace period expires -func (g *GracePeriodChecker) expireSubscription(ctx context.Context, sub *models.Subscription) error { - // Update subscription to cancelled - now := time.Now() - update := bson.M{ - "$set": bson.M{ - "tier": models.TierFree, - "status": models.SubStatusCancelled, - "cancelledAt": now, - "updatedAt": now, - }, - } - - _, err := g.subscriptions.UpdateOne(ctx, bson.M{"_id": sub.ID}, update) - if err != nil { - return err - } - - // Update user tier to free - if g.userService != nil { - err = g.userService.UpdateSubscriptionWithStatus( - ctx, - sub.UserID, - models.TierFree, - models.SubStatusCancelled, - nil, - ) - if err != nil { - log.Printf("⚠️ [GRACE-PERIOD] Failed to update user tier: %v", err) - // Don't fail the job if user update fails - } - } - - // Invalidate tier cache - if g.tierService != nil { - g.tierService.InvalidateCache(sub.UserID) - } - - return nil -} - -// GetNextRunTime returns when the job should run next (hourly) -func (g *GracePeriodChecker) GetNextRunTime() time.Time { - return time.Now().UTC().Add(1 * time.Hour) -} - -// GetExpiredSubscriptions returns subscriptions past grace period (for monitoring) -func (g *GracePeriodChecker) GetExpiredSubscriptions(ctx context.Context) ([]models.Subscription, error) { - if g.mongoDB == nil { - return nil, nil - } - - cutoffDate := time.Now().UTC().AddDate(0, 0, -g.gracePeriodDays) - - filter := bson.M{ - "status": models.SubStatusOnHold, - "updatedAt": bson.M{ - "$lt": cutoffDate, - }, - } - - cursor, err := g.subscriptions.Find(ctx, filter) - if err != nil { - return nil, err - } - defer cursor.Close(ctx) - - var subscriptions []models.Subscription - if err := cursor.All(ctx, &subscriptions); err != nil { - return nil, err - } - - return subscriptions, nil -} diff --git a/backend/internal/jobs/promo_expiration_checker.go b/backend/internal/jobs/promo_expiration_checker.go deleted file mode 100644 index 314649e1..00000000 --- a/backend/internal/jobs/promo_expiration_checker.go +++ /dev/null @@ -1,122 +0,0 @@ -package jobs - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "claraverse/internal/services" - "context" - "log" - "time" - - "go.mongodb.org/mongo-driver/bson" -) - -// PromoExpirationChecker handles expiration of promotional pro subscriptions -type PromoExpirationChecker struct { - mongoDB *database.MongoDB - userService *services.UserService - tierService *services.TierService -} - -// NewPromoExpirationChecker creates a new promo expiration checker -func NewPromoExpirationChecker( - mongoDB *database.MongoDB, - userService *services.UserService, - tierService *services.TierService, -) *PromoExpirationChecker { - return &PromoExpirationChecker{ - mongoDB: mongoDB, - userService: userService, - tierService: tierService, - } -} - -// Run checks for expired promotional subscriptions and downgrades users -func (p *PromoExpirationChecker) Run(ctx context.Context) error { - if p.mongoDB == nil || p.userService == nil || p.tierService == nil { - log.Println("⚠️ [PROMO-EXPIRATION] Promo expiration checker disabled (requires MongoDB, UserService, TierService)") - return nil - } - - log.Println("⏰ [PROMO-EXPIRATION] Checking for expired promotional subscriptions...") - startTime := time.Now() - - // Find users collection - collection := p.mongoDB.Database().Collection("users") - - // Find promo users with expired subscriptions - // Criteria: - // - subscriptionTier = "pro" - // - subscriptionExpiresAt < now - // - dodoSubscriptionId is empty (not a paid subscriber) - filter := bson.M{ - "subscriptionTier": models.TierPro, - "subscriptionExpiresAt": bson.M{ - "$lt": time.Now().UTC(), - }, - "$or": []bson.M{ - {"dodoSubscriptionId": ""}, - {"dodoSubscriptionId": bson.M{"$exists": false}}, - }, - } - - cursor, err := collection.Find(ctx, filter) - if err != nil { - log.Printf("❌ [PROMO-EXPIRATION] Failed to query users: %v", err) - return err - } - defer cursor.Close(ctx) - - expiredCount := 0 - for cursor.Next(ctx) { - var user models.User - if err := cursor.Decode(&user); err != nil { - log.Printf("⚠️ [PROMO-EXPIRATION] Failed to decode user: %v", err) - continue - } - - if err := p.expirePromoSubscription(ctx, &user); err != nil { - log.Printf("⚠️ [PROMO-EXPIRATION] Failed to expire promo for user %s: %v", user.SupabaseUserID, err) - continue - } - - expiredCount++ - log.Printf("✅ [PROMO-EXPIRATION] Expired promo subscription for user %s (promo ended %v ago)", - user.SupabaseUserID, time.Since(*user.SubscriptionExpiresAt).Round(time.Hour)) - } - - duration := time.Since(startTime) - log.Printf("✅ [PROMO-EXPIRATION] Check complete: expired %d promotional subscriptions in %v", expiredCount, duration) - - return nil -} - -// expirePromoSubscription downgrades a user from promo pro to free tier -func (p *PromoExpirationChecker) expirePromoSubscription(ctx context.Context, user *models.User) error { - // Update user to free tier with cancelled status - collection := p.mongoDB.Database().Collection("users") - - update := bson.M{ - "$set": bson.M{ - "subscriptionTier": models.TierFree, - "subscriptionStatus": models.SubStatusCancelled, - }, - } - - _, err := collection.UpdateOne(ctx, bson.M{"_id": user.ID}, update) - if err != nil { - return err - } - - // Invalidate tier cache so user immediately sees free tier on next request - if p.tierService != nil { - p.tierService.InvalidateCache(user.SupabaseUserID) - } - - return nil -} - -// GetNextRunTime returns when the job should run next (hourly) -func (p *PromoExpirationChecker) GetNextRunTime() time.Time { - return time.Now().UTC().Add(1 * time.Hour) -} diff --git a/backend/internal/jobs/retention_cleanup.go b/backend/internal/jobs/retention_cleanup.go deleted file mode 100644 index 144909b4..00000000 --- a/backend/internal/jobs/retention_cleanup.go +++ /dev/null @@ -1,193 +0,0 @@ -package jobs - -import ( - "claraverse/internal/database" - "claraverse/internal/services" - "context" - "log" - "time" - - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// RetentionCleanupJob handles deletion of old execution data based on tier retention limits -type RetentionCleanupJob struct { - mongoDB *database.MongoDB - tierService *services.TierService - executions interface{} // Will be *mongo.Collection -} - -// NewRetentionCleanupJob creates a new retention cleanup job -func NewRetentionCleanupJob(mongoDB *database.MongoDB, tierService *services.TierService) *RetentionCleanupJob { - var executions interface{} - if mongoDB != nil { - executions = mongoDB.Database().Collection("executions") - } - - return &RetentionCleanupJob{ - mongoDB: mongoDB, - tierService: tierService, - executions: executions, - } -} - -// Run executes the retention cleanup for all users -func (j *RetentionCleanupJob) Run(ctx context.Context) error { - if j.mongoDB == nil || j.tierService == nil { - log.Println("⚠️ [RETENTION] Retention cleanup disabled (requires MongoDB and TierService)") - return nil - } - - log.Println("🗑️ [RETENTION] Starting execution retention cleanup...") - startTime := time.Now() - - // Get all unique user IDs from executions collection - userIDs, err := j.getUniqueUserIDs(ctx) - if err != nil { - log.Printf("❌ [RETENTION] Failed to get user IDs: %v", err) - return err - } - - log.Printf("🔍 [RETENTION] Found %d users with executions", len(userIDs)) - - totalDeleted := 0 - for _, userID := range userIDs { - deleted, err := j.cleanupUserExecutions(ctx, userID) - if err != nil { - log.Printf("⚠️ [RETENTION] Failed to cleanup executions for user %s: %v", userID, err) - continue - } - if deleted > 0 { - totalDeleted += deleted - log.Printf("✅ [RETENTION] Deleted %d old executions for user %s", deleted, userID) - } - } - - duration := time.Since(startTime) - log.Printf("✅ [RETENTION] Cleanup complete: deleted %d executions in %v", totalDeleted, duration) - - return nil -} - -// getUniqueUserIDs returns all unique user IDs that have executions -func (j *RetentionCleanupJob) getUniqueUserIDs(ctx context.Context) ([]string, error) { - collection := j.executions.(interface { - Distinct(context.Context, string, interface{}) ([]interface{}, error) - }) - - results, err := collection.Distinct(ctx, "userId", bson.M{}) - if err != nil { - return nil, err - } - - userIDs := make([]string, 0, len(results)) - for _, result := range results { - if userID, ok := result.(string); ok { - userIDs = append(userIDs, userID) - } - } - - return userIDs, nil -} - -// cleanupUserExecutions deletes old executions for a specific user based on their tier retention -func (j *RetentionCleanupJob) cleanupUserExecutions(ctx context.Context, userID string) (int, error) { - // Get user's tier limits - limits := j.tierService.GetLimits(ctx, userID) - - // Calculate cutoff date - cutoffDate := time.Now().UTC().AddDate(0, 0, -limits.RetentionDays) - - // Delete executions older than retention period - collection := j.executions.(interface { - DeleteMany(context.Context, interface{}) (interface{ DeletedCount() int64 }, error) - }) - - filter := bson.M{ - "userId": userID, - "createdAt": bson.M{ - "$lt": cutoffDate, - }, - } - - result, err := collection.DeleteMany(ctx, filter) - if err != nil { - return 0, err - } - - return int(result.DeletedCount()), nil -} - -// GetNextRunTime returns when the job should run next (daily at 2 AM UTC) -func (j *RetentionCleanupJob) GetNextRunTime() time.Time { - now := time.Now().UTC() - - // Schedule for 2 AM UTC - nextRun := time.Date(now.Year(), now.Month(), now.Day(), 2, 0, 0, 0, time.UTC) - - // If we've passed 2 AM today, schedule for tomorrow - if now.After(nextRun) { - nextRun = nextRun.Add(24 * time.Hour) - } - - return nextRun -} - -// GetStats returns statistics about execution retention -func (j *RetentionCleanupJob) GetStats(ctx context.Context, userID string) (*RetentionStats, error) { - if j.mongoDB == nil || j.tierService == nil { - return nil, nil - } - - limits := j.tierService.GetLimits(ctx, userID) - cutoffDate := time.Now().UTC().AddDate(0, 0, -limits.RetentionDays) - - collection := j.executions.(interface { - CountDocuments(context.Context, interface{}) (int64, error) - }) - - // Count total executions - total, err := collection.CountDocuments(ctx, bson.M{"userId": userID}) - if err != nil { - return nil, err - } - - // Count old executions (will be deleted) - old, err := collection.CountDocuments(ctx, bson.M{ - "userId": userID, - "createdAt": bson.M{ - "$lt": cutoffDate, - }, - }) - if err != nil { - return nil, err - } - - return &RetentionStats{ - TotalExecutions: int(total), - RetainedExecutions: int(total - old), - DeletableExecutions: int(old), - RetentionDays: limits.RetentionDays, - CutoffDate: cutoffDate, - }, nil -} - -// RetentionStats provides statistics about execution retention -type RetentionStats struct { - TotalExecutions int `json:"total_executions"` - RetainedExecutions int `json:"retained_executions"` - DeletableExecutions int `json:"deletable_executions"` - RetentionDays int `json:"retention_days"` - CutoffDate time.Time `json:"cutoff_date"` -} - -// ExecutionRetention model for tracking deletion events (audit log) -type ExecutionRetention struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - UserID string `bson:"userId" json:"user_id"` - DeletedAt time.Time `bson:"deletedAt" json:"deleted_at"` - Count int `bson:"count" json:"count"` - RetentionDays int `bson:"retentionDays" json:"retention_days"` - CutoffDate time.Time `bson:"cutoffDate" json:"cutoff_date"` -} diff --git a/backend/internal/jobs/scheduler.go b/backend/internal/jobs/scheduler.go deleted file mode 100644 index c23f6bdc..00000000 --- a/backend/internal/jobs/scheduler.go +++ /dev/null @@ -1,171 +0,0 @@ -package jobs - -import ( - "context" - "log" - "sync" - "time" -) - -// Job interface that all scheduled jobs must implement -type Job interface { - Run(ctx context.Context) error - GetNextRunTime() time.Time -} - -// JobScheduler manages and runs scheduled jobs -type JobScheduler struct { - jobs map[string]Job - timers map[string]*time.Timer - ctx context.Context - cancel context.CancelFunc - wg sync.WaitGroup - mu sync.Mutex - running bool -} - -// NewJobScheduler creates a new job scheduler -func NewJobScheduler() *JobScheduler { - ctx, cancel := context.WithCancel(context.Background()) - return &JobScheduler{ - jobs: make(map[string]Job), - timers: make(map[string]*time.Timer), - ctx: ctx, - cancel: cancel, - } -} - -// Register adds a job to the scheduler -func (s *JobScheduler) Register(name string, job Job) { - s.mu.Lock() - defer s.mu.Unlock() - - s.jobs[name] = job - log.Printf("✅ [SCHEDULER] Registered job: %s", name) -} - -// Start begins running all registered jobs -func (s *JobScheduler) Start() error { - s.mu.Lock() - defer s.mu.Unlock() - - if s.running { - return nil - } - - s.running = true - log.Printf("🚀 [SCHEDULER] Starting job scheduler with %d jobs", len(s.jobs)) - - // Schedule all jobs - for name, job := range s.jobs { - s.scheduleJob(name, job) - } - - return nil -} - -// scheduleJob schedules a single job -func (s *JobScheduler) scheduleJob(name string, job Job) { - nextRun := job.GetNextRunTime() - duration := time.Until(nextRun) - - log.Printf("⏰ [SCHEDULER] Job '%s' scheduled to run at %s (in %v)", - name, nextRun.Format(time.RFC3339), duration) - - timer := time.AfterFunc(duration, func() { - s.runJob(name, job) - }) - - s.timers[name] = timer -} - -// runJob executes a job and reschedules it -func (s *JobScheduler) runJob(name string, job Job) { - s.wg.Add(1) - defer s.wg.Done() - - log.Printf("▶️ [SCHEDULER] Running job: %s", name) - startTime := time.Now() - - // Run the job - if err := job.Run(s.ctx); err != nil { - log.Printf("❌ [SCHEDULER] Job '%s' failed: %v", name, err) - } - - duration := time.Since(startTime) - log.Printf("✅ [SCHEDULER] Job '%s' completed in %v", name, duration) - - // Reschedule the job - s.mu.Lock() - defer s.mu.Unlock() - - if s.running { - s.scheduleJob(name, job) - } -} - -// Stop gracefully stops all jobs -func (s *JobScheduler) Stop() { - s.mu.Lock() - if !s.running { - s.mu.Unlock() - return - } - - log.Println("🛑 [SCHEDULER] Stopping job scheduler...") - s.running = false - - // Stop all timers - for name, timer := range s.timers { - timer.Stop() - log.Printf("⏹️ [SCHEDULER] Stopped job: %s", name) - } - s.timers = make(map[string]*time.Timer) - - s.mu.Unlock() - - // Cancel context and wait for running jobs - s.cancel() - s.wg.Wait() - - log.Println("✅ [SCHEDULER] Job scheduler stopped") -} - -// RunNow immediately runs a specific job (useful for testing) -func (s *JobScheduler) RunNow(name string) error { - s.mu.Lock() - job, exists := s.jobs[name] - s.mu.Unlock() - - if !exists { - log.Printf("⚠️ [SCHEDULER] Job '%s' not found", name) - return nil - } - - log.Printf("🚀 [SCHEDULER] Running job '%s' immediately", name) - return job.Run(s.ctx) -} - -// GetStatus returns the status of all jobs -func (s *JobScheduler) GetStatus() map[string]JobStatus { - s.mu.Lock() - defer s.mu.Unlock() - - status := make(map[string]JobStatus) - for name, job := range s.jobs { - status[name] = JobStatus{ - Name: name, - NextRunTime: job.GetNextRunTime(), - Registered: true, - } - } - - return status -} - -// JobStatus represents the status of a job -type JobStatus struct { - Name string `json:"name"` - NextRunTime time.Time `json:"next_run_time"` - Registered bool `json:"registered"` -} diff --git a/backend/internal/middleware/admin.go b/backend/internal/middleware/admin.go deleted file mode 100644 index 93cdd16b..00000000 --- a/backend/internal/middleware/admin.go +++ /dev/null @@ -1,60 +0,0 @@ -package middleware - -import ( - "claraverse/internal/config" - "log" - - "github.com/gofiber/fiber/v2" -) - -// AdminMiddleware checks if the authenticated user is a superadmin -func AdminMiddleware(cfg *config.Config) fiber.Handler { - return func(c *fiber.Ctx) error { - userID, ok := c.Locals("user_id").(string) - if !ok || userID == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Authentication required", - }) - } - - // Check role from context (set by JWT middleware) - role, hasRole := c.Locals("user_role").(string) - - // First check role field (preferred method) - if hasRole && role == "admin" { - c.Locals("is_superadmin", true) - log.Printf("✅ Admin access granted to user %s (role: %s)", userID, role) - return c.Next() - } - - // Fallback: Check if user is in superadmin list (legacy support) - isSuperadmin := false - for _, adminID := range cfg.SuperadminUserIDs { - if adminID == userID { - isSuperadmin = true - break - } - } - - if !isSuperadmin { - log.Printf("🚫 Non-admin user %s attempted to access admin endpoint (role: %s)", userID, role) - return c.Status(fiber.StatusForbidden).JSON(fiber.Map{ - "error": "Admin access required", - }) - } - - // Store admin flag for handlers to use - c.Locals("is_superadmin", true) - return c.Next() - } -} - -// IsSuperadmin is a helper function to check if a user ID is a superadmin -func IsSuperadmin(userID string, cfg *config.Config) bool { - for _, adminID := range cfg.SuperadminUserIDs { - if adminID == userID { - return true - } - } - return false -} diff --git a/backend/internal/middleware/apikey.go b/backend/internal/middleware/apikey.go deleted file mode 100644 index f6b6d413..00000000 --- a/backend/internal/middleware/apikey.go +++ /dev/null @@ -1,235 +0,0 @@ -package middleware - -import ( - "claraverse/internal/models" - "claraverse/internal/services" - "log" - "strconv" - - "github.com/gofiber/fiber/v2" -) - -// APIKeyMiddleware validates API keys for programmatic access -// This middleware checks the X-API-Key header and validates the key -func APIKeyMiddleware(apiKeyService *services.APIKeyService) fiber.Handler { - return func(c *fiber.Ctx) error { - // Get API key from header - apiKey := c.Get("X-API-Key") - if apiKey == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Missing API key. Include X-API-Key header.", - }) - } - - // Validate the key - key, err := apiKeyService.ValidateKey(c.Context(), apiKey) - if err != nil { - log.Printf("❌ [APIKEY-AUTH] Invalid key attempt: %v", err) - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Invalid or expired API key", - }) - } - - // Check if revoked - if key.IsRevoked() { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "API key has been revoked", - }) - } - - // Store key info in context for handlers - c.Locals("api_key", key) - c.Locals("user_id", key.UserID) - c.Locals("auth_type", "api_key") - - log.Printf("🔑 [APIKEY-AUTH] Authenticated via API key %s (user: %s)", key.KeyPrefix, key.UserID) - - return c.Next() - } -} - -// APIKeyOrJWTMiddleware allows authentication via either API key or JWT -// Checks API key first, falls back to JWT -func APIKeyOrJWTMiddleware(apiKeyService *services.APIKeyService, jwtMiddleware fiber.Handler) fiber.Handler { - return func(c *fiber.Ctx) error { - // Check for API key first - apiKey := c.Get("X-API-Key") - if apiKey != "" { - // Validate the key - key, err := apiKeyService.ValidateKey(c.Context(), apiKey) - if err != nil { - log.Printf("❌ [APIKEY-AUTH] Invalid key: %v", err) - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Invalid or expired API key", - }) - } - - if key.IsRevoked() { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "API key has been revoked", - }) - } - - // Authenticated via API key - c.Locals("api_key", key) - c.Locals("user_id", key.UserID) - c.Locals("auth_type", "api_key") - - log.Printf("🔑 [APIKEY-AUTH] Authenticated via API key %s", key.KeyPrefix) - return c.Next() - } - - // Fall back to JWT middleware - return jwtMiddleware(c) - } -} - -// RequireScope middleware checks if the authenticated API key has a specific scope -func RequireScope(scope string) fiber.Handler { - return func(c *fiber.Ctx) error { - // Check if authenticated via API key - authType, _ := c.Locals("auth_type").(string) - if authType != "api_key" { - // JWT auth - allow through (JWT has full access) - return c.Next() - } - - // Get API key from context - key, ok := c.Locals("api_key").(*models.APIKey) - if !ok { - // Fallback - allow through - return c.Next() - } - - // Check if key has required scope - if !key.HasScope(scope) { - log.Printf("🚫 [APIKEY-AUTH] Scope denied: %s (has: %v)", scope, key.Scopes) - return c.Status(fiber.StatusForbidden).JSON(fiber.Map{ - "error": "API key does not have required permission: " + scope, - }) - } - - return c.Next() - } -} - -// RequireExecuteScope middleware checks if the API key can execute a specific agent -func RequireExecuteScope(agentIDParam string) fiber.Handler { - return func(c *fiber.Ctx) error { - // Check if authenticated via API key - authType, _ := c.Locals("auth_type").(string) - if authType != "api_key" { - // JWT auth - allow through - return c.Next() - } - - // Get agent ID from params - agentID := c.Params(agentIDParam) - if agentID == "" { - return c.Status(fiber.StatusBadRequest).JSON(fiber.Map{ - "error": "Missing agent ID", - }) - } - - // Get API key from context - key, ok := c.Locals("api_key").(*models.APIKey) - if !ok { - return c.Next() - } - - // Check if key can execute this agent - if !key.HasExecuteScope(agentID) { - log.Printf("🚫 [APIKEY-AUTH] Execute denied for agent %s (has: %v)", agentID, key.Scopes) - return c.Status(fiber.StatusForbidden).JSON(fiber.Map{ - "error": "API key cannot execute this agent", - }) - } - - return c.Next() - } -} - -// RateLimitByAPIKey applies rate limiting based on API key limits -func RateLimitByAPIKey(redisService *services.RedisService) fiber.Handler { - return func(c *fiber.Ctx) error { - // Only apply to API key auth - authType, _ := c.Locals("auth_type").(string) - if authType != "api_key" { - return c.Next() - } - - // Get API key from context - key, ok := c.Locals("api_key").(*models.APIKey) - if !ok || redisService == nil { - return c.Next() - } - - // Get rate limits - var perMinute, perHour int64 = 60, 1000 // Defaults - if key.RateLimit != nil { - perMinute = key.RateLimit.RequestsPerMinute - perHour = key.RateLimit.RequestsPerHour - } - - // Check rate limits using Redis - keyPrefix := key.KeyPrefix - ctx := c.Context() - - // Check per-minute limit - minuteKey := "ratelimit:minute:" + keyPrefix - count, err := redisService.Incr(ctx, minuteKey) - if err != nil { - log.Printf("⚠️ [RATE-LIMIT] Redis error: %v", err) - return c.Next() // Allow on error - } - - if count == 1 { - // First request, set expiry - redisService.Expire(ctx, minuteKey, 60) - } - - if count > perMinute { - return c.Status(fiber.StatusTooManyRequests).JSON(fiber.Map{ - "error": "Rate limit exceeded (per minute)", - "retry_after": "60 seconds", - }) - } - - // Check per-hour limit - hourKey := "ratelimit:hour:" + keyPrefix - hourCount, err := redisService.Incr(ctx, hourKey) - if err != nil { - return c.Next() - } - - if hourCount == 1 { - redisService.Expire(ctx, hourKey, 3600) - } - - if hourCount > perHour { - return c.Status(fiber.StatusTooManyRequests).JSON(fiber.Map{ - "error": "Rate limit exceeded (per hour)", - "retry_after": "3600 seconds", - }) - } - - // Add rate limit headers - c.Set("X-RateLimit-Limit-Minute", formatInt64(perMinute)) - c.Set("X-RateLimit-Remaining-Minute", formatInt64(max(0, perMinute-count))) - c.Set("X-RateLimit-Limit-Hour", formatInt64(perHour)) - c.Set("X-RateLimit-Remaining-Hour", formatInt64(max(0, perHour-hourCount))) - - return c.Next() - } -} - -func formatInt64(n int64) string { - return strconv.FormatInt(n, 10) -} - -func max(a, b int64) int64 { - if a > b { - return a - } - return b -} diff --git a/backend/internal/middleware/auth.go b/backend/internal/middleware/auth.go deleted file mode 100644 index 10bfa528..00000000 --- a/backend/internal/middleware/auth.go +++ /dev/null @@ -1,151 +0,0 @@ -package middleware - -import ( - "claraverse/pkg/auth" - "log" - "os" - - "github.com/gofiber/fiber/v2" -) - -// AuthMiddleware verifies Supabase JWT tokens -// Supports both Authorization header and query parameter (for WebSocket connections) -func AuthMiddleware(supabaseAuth *auth.SupabaseAuth) fiber.Handler { - return func(c *fiber.Ctx) error { - // SECURITY: DEV_API_KEY bypass has been removed for security reasons. - // Use proper Supabase authentication or separate development/staging environments. - - // Skip auth if Supabase is not configured (development mode ONLY) - if supabaseAuth.URL == "" { - environment := os.Getenv("ENVIRONMENT") - - // CRITICAL: Never allow auth bypass in production - if environment == "production" { - log.Fatal("❌ CRITICAL SECURITY ERROR: Supabase not configured in production environment. Authentication is required.") - } - - // Only allow bypass in development/testing - if environment != "development" && environment != "testing" && environment != "" { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "Authentication service unavailable", - }) - } - - log.Println("⚠️ Auth skipped: Supabase not configured (development mode)") - c.Locals("user_id", "dev-user") - c.Locals("user_email", "dev@localhost") - c.Locals("user_role", "authenticated") - return c.Next() - } - - // Try to extract token from multiple sources - var token string - - // 1. Try Authorization header first - authHeader := c.Get("Authorization") - if authHeader != "" { - extractedToken, err := auth.ExtractToken(authHeader) - if err == nil { - token = extractedToken - } - } - - // 2. Try query parameter (for WebSocket connections) - if token == "" { - token = c.Query("token") - } - - // No token found - if token == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Missing or invalid authorization token", - }) - } - - // Verify token with Supabase - user, err := supabaseAuth.VerifyToken(token) - if err != nil { - log.Printf("❌ Auth failed: %v", err) - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Invalid or expired token", - }) - } - - // Store user info in context - c.Locals("user_id", user.ID) - c.Locals("user_email", user.Email) - c.Locals("user_role", user.Role) - - log.Printf("✅ Authenticated user: %s (%s)", user.Email, user.ID) - return c.Next() - } -} - -// OptionalAuthMiddleware makes authentication optional -// Supports both Authorization header and query parameter (for WebSocket) -func OptionalAuthMiddleware(supabaseAuth *auth.SupabaseAuth) fiber.Handler { - return func(c *fiber.Ctx) error { - // Try to extract token from multiple sources - var token string - - // 1. Try Authorization header first - authHeader := c.Get("Authorization") - if authHeader != "" { - extractedToken, err := auth.ExtractToken(authHeader) - if err == nil { - token = extractedToken - } - } - - // 2. Try query parameter (for WebSocket connections) - if token == "" { - token = c.Query("token") - } - - // If no token found, proceed as anonymous - if token == "" { - c.Locals("user_id", "anonymous") - log.Println("🔓 Anonymous connection") - return c.Next() - } - - // Skip validation if Supabase is not configured (development mode ONLY) - if supabaseAuth == nil || supabaseAuth.URL == "" { - environment := os.Getenv("ENVIRONMENT") - - // CRITICAL: Never allow auth bypass in production - if environment == "production" { - log.Fatal("❌ CRITICAL SECURITY ERROR: Supabase not configured in production environment. Authentication is required.") - } - - // Only allow in development/testing - if environment != "development" && environment != "testing" && environment != "" { - c.Locals("user_id", "anonymous") - log.Println("⚠️ Supabase unavailable, proceeding as anonymous") - return c.Next() - } - - c.Locals("user_id", "dev-user-" + token[:min(8, len(token))]) - c.Locals("user_email", "dev@localhost") - c.Locals("user_role", "authenticated") - log.Println("⚠️ Auth skipped: Supabase not configured (dev mode)") - return c.Next() - } - - // Verify token with Supabase - user, err := supabaseAuth.VerifyToken(token) - if err != nil { - log.Printf("⚠️ Token validation failed: %v (continuing as anonymous)", err) - c.Locals("user_id", "anonymous") - return c.Next() - } - - // Store authenticated user info - c.Locals("user_id", user.ID) - c.Locals("user_email", user.Email) - c.Locals("user_role", user.Role) - - log.Printf("✅ Authenticated user: %s (%s)", user.Email, user.ID) - return c.Next() - } -} diff --git a/backend/internal/middleware/auth_local.go b/backend/internal/middleware/auth_local.go deleted file mode 100644 index 737b5508..00000000 --- a/backend/internal/middleware/auth_local.go +++ /dev/null @@ -1,152 +0,0 @@ -package middleware - -import ( - "claraverse/pkg/auth" - "log" - "os" - - "github.com/gofiber/fiber/v2" -) - -// LocalAuthMiddleware verifies local JWT tokens -// Supports both Authorization header and query parameter (for WebSocket connections) -func LocalAuthMiddleware(jwtAuth *auth.LocalJWTAuth) fiber.Handler { - return func(c *fiber.Ctx) error { - // Skip auth if JWT secret is not configured (development mode ONLY) - environment := os.Getenv("ENVIRONMENT") - - if jwtAuth == nil { - // CRITICAL: Never allow auth bypass in production - if environment == "production" { - log.Fatal("❌ CRITICAL SECURITY ERROR: JWT auth not configured in production environment. Authentication is required.") - } - - // Only allow bypass in development/testing - if environment != "development" && environment != "testing" && environment != "" { - return c.Status(fiber.StatusServiceUnavailable).JSON(fiber.Map{ - "error": "Authentication service unavailable", - }) - } - - log.Println("⚠️ Auth skipped: JWT not configured (development mode)") - c.Locals("user_id", "dev-user") - c.Locals("user_email", "dev@localhost") - c.Locals("user_role", "user") - return c.Next() - } - - // Try to extract token from multiple sources - var token string - - // 1. Try Authorization header first - authHeader := c.Get("Authorization") - if authHeader != "" { - extractedToken, err := auth.ExtractToken(authHeader) - if err == nil { - token = extractedToken - } - } - - // 2. Try query parameter (for WebSocket connections) - if token == "" { - token = c.Query("token") - } - - // No token found - if token == "" { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Missing or invalid authorization token", - }) - } - - // Verify JWT token - user, err := jwtAuth.VerifyAccessToken(token) - if err != nil { - log.Printf("❌ Auth failed: %v", err) - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Invalid or expired token", - }) - } - - // Store user info in context - c.Locals("user_id", user.ID) - c.Locals("user_email", user.Email) - c.Locals("user_role", user.Role) - - log.Printf("✅ Authenticated user: %s (%s)", user.Email, user.ID) - return c.Next() - } -} - -// OptionalLocalAuthMiddleware makes authentication optional -// Supports both Authorization header and query parameter (for WebSocket) -func OptionalLocalAuthMiddleware(jwtAuth *auth.LocalJWTAuth) fiber.Handler { - return func(c *fiber.Ctx) error { - // Try to extract token from multiple sources - var token string - - // 1. Try Authorization header first - authHeader := c.Get("Authorization") - if authHeader != "" { - extractedToken, err := auth.ExtractToken(authHeader) - if err == nil { - token = extractedToken - } - } - - // 2. Try query parameter (for WebSocket connections) - if token == "" { - token = c.Query("token") - } - - // If no token found, proceed as anonymous - if token == "" { - c.Locals("user_id", "anonymous") - log.Println("🔓 Anonymous connection") - return c.Next() - } - - // Skip validation if JWT auth is not configured (development mode ONLY) - environment := os.Getenv("ENVIRONMENT") - - if jwtAuth == nil { - // CRITICAL: Never allow auth bypass in production - if environment == "production" { - log.Fatal("❌ CRITICAL SECURITY ERROR: JWT auth not configured in production environment") - } - - // Only allow in development/testing - if environment != "development" && environment != "testing" && environment != "" { - c.Locals("user_id", "anonymous") - log.Println("⚠️ JWT unavailable, proceeding as anonymous") - return c.Next() - } - - c.Locals("user_id", "dev-user") - c.Locals("user_email", "dev@localhost") - c.Locals("user_role", "user") - log.Println("⚠️ Auth skipped: JWT not configured (dev mode)") - return c.Next() - } - - // Verify JWT token - user, err := jwtAuth.VerifyAccessToken(token) - if err != nil { - log.Printf("⚠️ Token validation failed: %v (continuing as anonymous)", err) - c.Locals("user_id", "anonymous") - return c.Next() - } - - // Store authenticated user info - c.Locals("user_id", user.ID) - c.Locals("user_email", user.Email) - c.Locals("user_role", user.Role) - - log.Printf("✅ Authenticated user: %s (%s)", user.Email, user.ID) - return c.Next() - } -} - -// RateLimitedAuthMiddleware combines rate limiting with authentication -// Rate limit: 5 attempts per 15 minutes per IP -// Note: This function is currently unused. Apply rate limiting separately in routes if needed. diff --git a/backend/internal/middleware/execution_limiter.go b/backend/internal/middleware/execution_limiter.go deleted file mode 100644 index 14b1e004..00000000 --- a/backend/internal/middleware/execution_limiter.go +++ /dev/null @@ -1,149 +0,0 @@ -package middleware - -import ( - "claraverse/internal/services" - "context" - "fmt" - "log" - "time" - - "github.com/gofiber/fiber/v2" - "github.com/redis/go-redis/v9" -) - -// ExecutionLimiter middleware checks daily execution limits based on user tier -type ExecutionLimiter struct { - tierService *services.TierService - redis *redis.Client -} - -// NewExecutionLimiter creates a new execution limiter middleware -func NewExecutionLimiter(tierService *services.TierService, redisClient *redis.Client) *ExecutionLimiter { - return &ExecutionLimiter{ - tierService: tierService, - redis: redisClient, - } -} - -// CheckLimit verifies if user can execute another workflow today -func (el *ExecutionLimiter) CheckLimit(c *fiber.Ctx) error { - userID := c.Locals("user_id") - if userID == nil { - return c.Status(fiber.StatusUnauthorized).JSON(fiber.Map{ - "error": "Unauthorized", - }) - } - - userIDStr, ok := userID.(string) - if !ok { - return c.Status(fiber.StatusInternalServerError).JSON(fiber.Map{ - "error": "Invalid user ID", - }) - } - - ctx := context.Background() - - // Get user's tier limits - limits := el.tierService.GetLimits(ctx, userIDStr) - - // If unlimited executions, skip check - if limits.MaxExecutionsPerDay == -1 { - return c.Next() - } - - // Get today's execution count from Redis - today := time.Now().UTC().Format("2006-01-02") - key := fmt.Sprintf("executions:%s:%s", userIDStr, today) - - // Get current count - count, err := el.redis.Get(ctx, key).Int64() - if err != nil && err != redis.Nil { - log.Printf("⚠️ Failed to get execution count from Redis: %v", err) - // On Redis error, allow execution but log warning - return c.Next() - } - - // Check if limit exceeded - if count >= limits.MaxExecutionsPerDay { - return c.Status(fiber.StatusTooManyRequests).JSON(fiber.Map{ - "error": "Daily execution limit exceeded", - "limit": limits.MaxExecutionsPerDay, - "used": count, - "reset_at": getNextMidnightUTC(), - }) - } - - // Store current count in context for post-execution increment - c.Locals("execution_count_key", key) - - return c.Next() -} - -// IncrementCount increments the execution counter after successful execution start -func (el *ExecutionLimiter) IncrementCount(userID string) error { - if el.redis == nil { - return nil // Redis not available, skip increment - } - - ctx := context.Background() - today := time.Now().UTC().Format("2006-01-02") - key := fmt.Sprintf("executions:%s:%s", userID, today) - - // Increment counter - pipe := el.redis.Pipeline() - pipe.Incr(ctx, key) - - // Set expiry to end of day + 1 day (to allow historical querying) - midnight := getNextMidnightUTC() - expiryDuration := time.Until(midnight) + 24*time.Hour - pipe.Expire(ctx, key, expiryDuration) - - _, err := pipe.Exec(ctx) - if err != nil { - log.Printf("⚠️ Failed to increment execution count: %v", err) - return err - } - - log.Printf("✅ Incremented execution count for user %s (key: %s)", userID, key) - return nil -} - -// GetRemainingExecutions returns how many executions user has left today -func (el *ExecutionLimiter) GetRemainingExecutions(userID string) (int64, error) { - if el.redis == nil { - return -1, nil // Redis not available, return unlimited - } - - ctx := context.Background() - - // Get user's tier limits - limits := el.tierService.GetLimits(ctx, userID) - if limits.MaxExecutionsPerDay == -1 { - return -1, nil // Unlimited - } - - // Get today's count - today := time.Now().UTC().Format("2006-01-02") - key := fmt.Sprintf("executions:%s:%s", userID, today) - - count, err := el.redis.Get(ctx, key).Int64() - if err == redis.Nil { - return limits.MaxExecutionsPerDay, nil // No executions today - } - if err != nil { - return -1, err - } - - remaining := limits.MaxExecutionsPerDay - count - if remaining < 0 { - return 0, nil - } - - return remaining, nil -} - -// getNextMidnightUTC returns the next midnight UTC -func getNextMidnightUTC() time.Time { - now := time.Now().UTC() - return time.Date(now.Year(), now.Month(), now.Day()+1, 0, 0, 0, 0, time.UTC) -} diff --git a/backend/internal/middleware/ratelimit.go b/backend/internal/middleware/ratelimit.go deleted file mode 100644 index 7b881543..00000000 --- a/backend/internal/middleware/ratelimit.go +++ /dev/null @@ -1,245 +0,0 @@ -package middleware - -import ( - "log" - "os" - "strconv" - "time" - - "github.com/gofiber/fiber/v2" - "github.com/gofiber/fiber/v2/middleware/limiter" -) - -// RateLimitConfig holds rate limiting settings -type RateLimitConfig struct { - // Global limits (per IP) - GlobalAPIMax int // Max requests per minute for all API endpoints - GlobalAPIExpiration time.Duration // Expiration window - - // Public endpoint limits (per IP) - read-only, cacheable - PublicReadMax int - PublicReadExpiration time.Duration - - // Authenticated endpoint limits (per user ID) - AuthenticatedMax int - AuthenticatedExpiration time.Duration - - // Heavy operation limits - TranscribeMax int - TranscribeExpiration time.Duration - - // WebSocket/Connection limits (per IP) - WebSocketMax int - WebSocketExpiration time.Duration - - // Image proxy limits (per IP) - can be abused for bandwidth - ImageProxyMax int - ImageProxyExpiration time.Duration -} - -// DefaultRateLimitConfig returns production-safe defaults -// These are designed to prevent abuse while avoiding false positives -func DefaultRateLimitConfig() *RateLimitConfig { - return &RateLimitConfig{ - // Global: 200/min = ~3.3 req/sec - very generous for normal use - GlobalAPIMax: 200, - GlobalAPIExpiration: 1 * time.Minute, - - // Public read endpoints: 120/min = 2 req/sec - PublicReadMax: 120, - PublicReadExpiration: 1 * time.Minute, - - // Authenticated operations: 60/min = 1 req/sec average - AuthenticatedMax: 60, - AuthenticatedExpiration: 1 * time.Minute, - - // Transcription: 10/min (expensive GPU operation) - TranscribeMax: 10, - TranscribeExpiration: 1 * time.Minute, - - // WebSocket: 20 connections/min in production - WebSocketMax: 20, - WebSocketExpiration: 1 * time.Minute, - - // Image proxy: 60/min (bandwidth protection) - ImageProxyMax: 60, - ImageProxyExpiration: 1 * time.Minute, - } -} - -// LoadRateLimitConfig loads config from environment variables with defaults -func LoadRateLimitConfig() *RateLimitConfig { - config := DefaultRateLimitConfig() - - // Allow environment overrides for tuning - if v := os.Getenv("RATE_LIMIT_GLOBAL_API"); v != "" { - if n, err := strconv.Atoi(v); err == nil && n > 0 { - config.GlobalAPIMax = n - } - } - - if v := os.Getenv("RATE_LIMIT_PUBLIC_READ"); v != "" { - if n, err := strconv.Atoi(v); err == nil && n > 0 { - config.PublicReadMax = n - } - } - - if v := os.Getenv("RATE_LIMIT_AUTHENTICATED"); v != "" { - if n, err := strconv.Atoi(v); err == nil && n > 0 { - config.AuthenticatedMax = n - } - } - - if v := os.Getenv("RATE_LIMIT_WEBSOCKET"); v != "" { - if n, err := strconv.Atoi(v); err == nil && n > 0 { - config.WebSocketMax = n - } - } - - if v := os.Getenv("RATE_LIMIT_IMAGE_PROXY"); v != "" { - if n, err := strconv.Atoi(v); err == nil && n > 0 { - config.ImageProxyMax = n - } - } - - // Development mode: more lenient limits - if os.Getenv("ENVIRONMENT") == "development" { - config.GlobalAPIMax = 1000 // Very high for dev - config.WebSocketMax = 100 // Keep high for dev - config.ImageProxyMax = 200 // More lenient - log.Println("⚠️ [RATE-LIMIT] Development mode: using relaxed rate limits") - } - - return config -} - -// GlobalAPIRateLimiter creates a rate limiter for all API requests -// This is the first line of defense against DDoS -func GlobalAPIRateLimiter(config *RateLimitConfig) fiber.Handler { - return limiter.New(limiter.Config{ - Max: config.GlobalAPIMax, - Expiration: config.GlobalAPIExpiration, - KeyGenerator: func(c *fiber.Ctx) string { - return "global:" + c.IP() - }, - LimitReached: func(c *fiber.Ctx) error { - log.Printf("🚫 [RATE-LIMIT] Global limit reached for IP: %s", c.IP()) - return c.Status(fiber.StatusTooManyRequests).JSON(fiber.Map{ - "error": "Too many requests. Please slow down.", - "retry_after": int(config.GlobalAPIExpiration.Seconds()), - }) - }, - SkipFailedRequests: false, - SkipSuccessfulRequests: false, - }) -} - -// PublicReadRateLimiter for public read-only endpoints -func PublicReadRateLimiter(config *RateLimitConfig) fiber.Handler { - return limiter.New(limiter.Config{ - Max: config.PublicReadMax, - Expiration: config.PublicReadExpiration, - KeyGenerator: func(c *fiber.Ctx) string { - return "public:" + c.IP() - }, - LimitReached: func(c *fiber.Ctx) error { - log.Printf("⚠️ [RATE-LIMIT] Public endpoint limit reached for IP: %s on %s", c.IP(), c.Path()) - return c.Status(fiber.StatusTooManyRequests).JSON(fiber.Map{ - "error": "Too many requests to this endpoint.", - "retry_after": int(config.PublicReadExpiration.Seconds()), - }) - }, - }) -} - -// AuthenticatedRateLimiter for authenticated endpoints (uses user ID) -func AuthenticatedRateLimiter(config *RateLimitConfig) fiber.Handler { - return limiter.New(limiter.Config{ - Max: config.AuthenticatedMax, - Expiration: config.AuthenticatedExpiration, - KeyGenerator: func(c *fiber.Ctx) string { - // Use user ID if available, fall back to IP - if userID, ok := c.Locals("user_id").(string); ok && userID != "" && userID != "anonymous" { - return "auth:" + userID - } - return "auth-ip:" + c.IP() - }, - LimitReached: func(c *fiber.Ctx) error { - userID, _ := c.Locals("user_id").(string) - log.Printf("⚠️ [RATE-LIMIT] Auth endpoint limit reached for user: %s on %s", userID, c.Path()) - return c.Status(fiber.StatusTooManyRequests).JSON(fiber.Map{ - "error": "Too many requests. Please wait before trying again.", - "retry_after": int(config.AuthenticatedExpiration.Seconds()), - }) - }, - }) -} - -// TranscribeRateLimiter for expensive audio transcription -func TranscribeRateLimiter(config *RateLimitConfig) fiber.Handler { - return limiter.New(limiter.Config{ - Max: config.TranscribeMax, - Expiration: config.TranscribeExpiration, - KeyGenerator: func(c *fiber.Ctx) string { - if userID, ok := c.Locals("user_id").(string); ok && userID != "" && userID != "anonymous" { - return "transcribe:" + userID - } - return "transcribe-ip:" + c.IP() - }, - LimitReached: func(c *fiber.Ctx) error { - log.Printf("⚠️ [RATE-LIMIT] Transcription limit reached for: %v", c.Locals("user_id")) - return c.Status(fiber.StatusTooManyRequests).JSON(fiber.Map{ - "error": "Transcription rate limit reached. Please wait before transcribing more audio.", - "retry_after": int(config.TranscribeExpiration.Seconds()), - }) - }, - }) -} - -// WebSocketRateLimiter for WebSocket connection attempts -func WebSocketRateLimiter(config *RateLimitConfig) fiber.Handler { - return limiter.New(limiter.Config{ - Max: config.WebSocketMax, - Expiration: config.WebSocketExpiration, - KeyGenerator: func(c *fiber.Ctx) string { - return "ws:" + c.IP() - }, - LimitReached: func(c *fiber.Ctx) error { - log.Printf("🚫 [RATE-LIMIT] WebSocket connection limit reached for IP: %s", c.IP()) - return c.Status(fiber.StatusTooManyRequests).JSON(fiber.Map{ - "error": "Too many connection attempts. Please wait before reconnecting.", - "retry_after": int(config.WebSocketExpiration.Seconds()), - }) - }, - }) -} - -// ImageProxyRateLimiter for image proxy requests (bandwidth protection) -func ImageProxyRateLimiter(config *RateLimitConfig) fiber.Handler { - return limiter.New(limiter.Config{ - Max: config.ImageProxyMax, - Expiration: config.ImageProxyExpiration, - KeyGenerator: func(c *fiber.Ctx) string { - return "imgproxy:" + c.IP() - }, - LimitReached: func(c *fiber.Ctx) error { - log.Printf("⚠️ [RATE-LIMIT] Image proxy limit reached for IP: %s", c.IP()) - return c.Status(fiber.StatusTooManyRequests).JSON(fiber.Map{ - "error": "Too many image requests. Please wait.", - "retry_after": int(config.ImageProxyExpiration.Seconds()), - }) - }, - }) -} - -// SlowdownMiddleware adds progressive delays for rapid requests -// This discourages automated attacks without hard blocking -func SlowdownMiddleware(threshold int, delay time.Duration) fiber.Handler { - // Use a simple in-memory counter (for single-instance deployments) - // For multi-instance, use Redis-backed rate limiting - return func(c *fiber.Ctx) error { - // This is a placeholder - the limiter middleware handles the actual limiting - // This could be enhanced to add progressive delays before hard blocking - return c.Next() - } -} diff --git a/backend/internal/models/agent.go b/backend/internal/models/agent.go deleted file mode 100644 index fbd2d61f..00000000 --- a/backend/internal/models/agent.go +++ /dev/null @@ -1,380 +0,0 @@ -package models - -import "time" - -// Agent represents a workflow automation agent -type Agent struct { - ID string `json:"id"` - UserID string `json:"user_id"` - Name string `json:"name"` - Description string `json:"description,omitempty"` - Status string `json:"status"` // draft, active, deployed - Workflow *Workflow `json:"workflow,omitempty"` - CreatedAt time.Time `json:"created_at"` - UpdatedAt time.Time `json:"updated_at"` -} - -// Workflow represents a DAG of blocks for an agent -type Workflow struct { - ID string `json:"id"` - AgentID string `json:"agent_id"` - Blocks []Block `json:"blocks"` - Connections []Connection `json:"connections"` - Variables []Variable `json:"variables"` - Version int `json:"version"` - CreatedAt time.Time `json:"created_at,omitempty"` - UpdatedAt time.Time `json:"updated_at,omitempty"` -} - -// Block represents a single node in the workflow DAG -type Block struct { - ID string `json:"id"` - NormalizedID string `json:"normalizedId"` // Normalized name for variable interpolation (e.g., "search-latest-news") - Type string `json:"type"` // llm_inference, tool_execution, webhook, variable, python_tool - Name string `json:"name"` - Description string `json:"description,omitempty"` - Config map[string]any `json:"config"` - Position Position `json:"position"` - Timeout int `json:"timeout"` // seconds, default 30 -} - -// Position represents x,y coordinates for canvas layout -type Position struct { - X float64 `json:"x"` - Y float64 `json:"y"` -} - -// Connection represents an edge between two blocks -type Connection struct { - ID string `json:"id"` - SourceBlockID string `json:"sourceBlockId"` - SourceOutput string `json:"sourceOutput,omitempty"` - TargetBlockID string `json:"targetBlockId"` - TargetInput string `json:"targetInput,omitempty"` -} - -// Variable represents a workflow-level variable -type Variable struct { - Name string `json:"name"` - Type string `json:"type"` // string, number, boolean, array, object - DefaultValue any `json:"defaultValue,omitempty"` -} - -// Execution represents a single workflow execution run -type Execution struct { - ID string `json:"id"` - AgentID string `json:"agent_id"` - WorkflowVersion int `json:"workflow_version"` - Status string `json:"status"` // pending, running, completed, failed, partial_failure - Input map[string]any `json:"input,omitempty"` - Output map[string]any `json:"output,omitempty"` - BlockStates map[string]*BlockState `json:"block_states,omitempty"` - StartedAt *time.Time `json:"started_at,omitempty"` - CompletedAt *time.Time `json:"completed_at,omitempty"` -} - -// BlockState represents the execution state of a single block -type BlockState struct { - Status string `json:"status"` // pending, running, completed, failed, skipped - Inputs map[string]any `json:"inputs,omitempty"` - Outputs map[string]any `json:"outputs,omitempty"` - Error string `json:"error,omitempty"` - StartedAt *time.Time `json:"started_at,omitempty"` - CompletedAt *time.Time `json:"completed_at,omitempty"` - - // Retry tracking - RetryCount int `json:"retry_count,omitempty"` // Number of retries attempted - RetryHistory []RetryAttempt `json:"retry_history,omitempty"` // Detailed retry history -} - -// RetryAttempt records a single retry attempt for debugging and monitoring -type RetryAttempt struct { - Attempt int `json:"attempt"` // 0-indexed attempt number - Error string `json:"error"` // Error message from this attempt - ErrorType string `json:"error_type"` // "timeout", "rate_limit", "server_error", etc. - Timestamp time.Time `json:"timestamp"` // When the attempt occurred - Duration int64 `json:"duration_ms"` // How long the attempt took -} - -// ExecutionUpdate is sent via WebSocket to stream execution progress -type ExecutionUpdate struct { - Type string `json:"type"` // execution_update - ExecutionID string `json:"execution_id"` - BlockID string `json:"block_id"` - Status string `json:"status"` - Inputs map[string]any `json:"inputs,omitempty"` // Available inputs for debugging - Output map[string]any `json:"output,omitempty"` - Error string `json:"error,omitempty"` -} - -// ExecutionComplete is sent when workflow execution finishes -type ExecutionComplete struct { - Type string `json:"type"` // execution_complete - ExecutionID string `json:"execution_id"` - Status string `json:"status"` // completed, failed, partial_failure - FinalOutput map[string]any `json:"final_output,omitempty"` - Duration int64 `json:"duration_ms"` -} - -// ============================================================================ -// Standardized API Response Types -// Clean, well-structured output for API consumers -// ============================================================================ - -// ExecutionAPIResponse is the standardized response for workflow execution -// This provides a clean, predictable structure for API consumers -type ExecutionAPIResponse struct { - // Status of the execution: completed, failed, partial - Status string `json:"status"` - - // Result contains the primary output from the workflow - // This is the "answer" - extracted from the final block's response - Result string `json:"result"` - - // Data contains the structured JSON data from the final block (if it was a structured output block) - // This is populated when the terminal block has outputFormat="json" and valid parsed data - Data any `json:"data,omitempty"` - - // Artifacts contains all generated charts, images, visualizations - // Each artifact has type, format, and base64 data - Artifacts []APIArtifact `json:"artifacts,omitempty"` - - // Files contains all generated files with download URLs - Files []APIFile `json:"files,omitempty"` - - // Blocks contains detailed output from each block (for debugging/advanced use) - Blocks map[string]APIBlockOutput `json:"blocks,omitempty"` - - // Metadata contains execution statistics - Metadata ExecutionMetadata `json:"metadata"` - - // Error contains error message if status is failed - Error string `json:"error,omitempty"` -} - -// APIArtifact represents a generated artifact (chart, image, etc.) -type APIArtifact struct { - Type string `json:"type"` // "chart", "image", "plot" - Format string `json:"format"` // "png", "jpeg", "svg" - Data string `json:"data"` // Base64 encoded data - Title string `json:"title,omitempty"` // Description/title - SourceBlock string `json:"source_block,omitempty"` // Which block generated this -} - -// APIFile represents a generated file -type APIFile struct { - FileID string `json:"file_id"` - Filename string `json:"filename"` - DownloadURL string `json:"download_url"` - MimeType string `json:"mime_type,omitempty"` - Size int64 `json:"size,omitempty"` - SourceBlock string `json:"source_block,omitempty"` -} - -// APIBlockOutput is a clean representation of a block's output -type APIBlockOutput struct { - Name string `json:"name"` - Type string `json:"type"` - Status string `json:"status"` - Response string `json:"response,omitempty"` // Primary text output - Data map[string]any `json:"data,omitempty"` // Structured data (if JSON output) - Error string `json:"error,omitempty"` - DurationMs int64 `json:"duration_ms,omitempty"` -} - -// ExecutionMetadata contains execution statistics -type ExecutionMetadata struct { - ExecutionID string `json:"execution_id"` - AgentID string `json:"agent_id,omitempty"` - WorkflowVersion int `json:"workflow_version,omitempty"` - DurationMs int64 `json:"duration_ms"` - TotalTokens int `json:"total_tokens,omitempty"` - BlocksExecuted int `json:"blocks_executed"` - BlocksFailed int `json:"blocks_failed"` -} - -// ExecuteWorkflowRequest is received from the client to start execution -type ExecuteWorkflowRequest struct { - Type string `json:"type"` // execute_workflow - AgentID string `json:"agent_id"` - Input map[string]any `json:"input,omitempty"` -} - -// CreateAgentRequest is the request body for creating an agent -type CreateAgentRequest struct { - Name string `json:"name"` - Description string `json:"description,omitempty"` -} - -// UpdateAgentRequest is the request body for updating an agent -type UpdateAgentRequest struct { - Name string `json:"name,omitempty"` - Description string `json:"description,omitempty"` - Status string `json:"status,omitempty"` -} - -// SaveWorkflowRequest is the request body for saving a workflow -type SaveWorkflowRequest struct { - Blocks []Block `json:"blocks"` - Connections []Connection `json:"connections"` - Variables []Variable `json:"variables,omitempty"` - CreateVersion bool `json:"createVersion,omitempty"` // Only create version snapshot if true - VersionDescription string `json:"versionDescription,omitempty"` // Description for the version (if created) -} - -// ============================================================================ -// Agent-Per-Block Architecture Types (Sprint 4) -// Each LLM block can act as a mini-agent with tool access and structured output -// ============================================================================ - -// AgentBlockConfig defines the configuration for an LLM block with agent capabilities -type AgentBlockConfig struct { - // Model Configuration - Model string `json:"model,omitempty"` // Default: "sonnet-4.5" (resolves to glm-4.6) - Temperature float64 `json:"temperature,omitempty"` // Default: 0.7 - - // Prompts - SystemPrompt string `json:"systemPrompt,omitempty"` - UserPrompt string `json:"userPrompt,omitempty"` // Supports {{variable}} interpolation - - // Tool Configuration - EnabledTools []string `json:"enabledTools,omitempty"` // e.g., ["search_web", "calculate_math"] - MaxToolCalls int `json:"maxToolCalls,omitempty"` // Default: 15 - Credentials []string `json:"credentials,omitempty"` // Credential IDs for tool authentication - - // Execution Mode Configuration (for deterministic block execution) - RequireToolUsage bool `json:"requireToolUsage,omitempty"` // Default: true when tools exist - forces tool calls - MaxRetries int `json:"maxRetries,omitempty"` // Default: 2 - retry attempts if tool not called - RequiredTools []string `json:"requiredTools,omitempty"` // Specific tools that MUST be called - - // Retry Policy for LLM API calls (transient error handling) - RetryPolicy *RetryPolicy `json:"retryPolicy,omitempty"` // Optional retry configuration for API failures - - // Output Configuration - OutputSchema *JSONSchema `json:"outputSchema,omitempty"` // JSON schema for validation - StrictOutput bool `json:"strictOutput,omitempty"` // Require exact schema match -} - -// RetryPolicy defines retry behavior for block execution (LLM API calls) -type RetryPolicy struct { - // MaxRetries is the maximum number of retry attempts (default: 1) - MaxRetries int `json:"maxRetries,omitempty"` - - // InitialDelay is the initial delay before first retry in milliseconds (default: 1000) - InitialDelay int `json:"initialDelay,omitempty"` - - // MaxDelay is the maximum delay between retries in milliseconds (default: 30000) - MaxDelay int `json:"maxDelay,omitempty"` - - // BackoffMultiplier is the exponential backoff multiplier (default: 2.0) - BackoffMultiplier float64 `json:"backoffMultiplier,omitempty"` - - // RetryOn specifies which error types to retry (default: ["timeout", "rate_limit", "server_error"]) - RetryOn []string `json:"retryOn,omitempty"` - - // JitterPercent adds randomness to delay to prevent thundering herd (0-100, default: 20) - JitterPercent int `json:"jitterPercent,omitempty"` -} - -// DefaultRetryPolicy returns sensible production defaults for retry behavior -func DefaultRetryPolicy() *RetryPolicy { - return &RetryPolicy{ - MaxRetries: 1, - InitialDelay: 1000, // 1 second - MaxDelay: 30000, // 30 seconds - BackoffMultiplier: 2.0, - RetryOn: []string{"timeout", "rate_limit", "server_error"}, - JitterPercent: 20, - } -} - -// JSONSchema represents a JSON Schema for output validation -type JSONSchema struct { - Type string `json:"type"` - Properties map[string]*JSONSchema `json:"properties,omitempty"` - Items *JSONSchema `json:"items,omitempty"` - Required []string `json:"required,omitempty"` - AdditionalProperties *bool `json:"additionalProperties,omitempty"` - Description string `json:"description,omitempty"` - Enum []string `json:"enum,omitempty"` - Default any `json:"default,omitempty"` -} - -// AgentBlockResult represents the result of an agent block execution -type AgentBlockResult struct { - // The validated output (matches OutputSchema if provided) - Output map[string]any `json:"output"` - - // Raw LLM response (before parsing) - RawResponse string `json:"rawResponse,omitempty"` - - // Model used for execution - Model string `json:"model"` - - // Token usage - Tokens TokenUsage `json:"tokens"` - - // Tool calls made during execution - ToolCalls []ToolCallRecord `json:"toolCalls,omitempty"` - - // Number of iterations in the agent loop - Iterations int `json:"iterations"` -} - -// ToolCallRecord records a tool call made during execution -type ToolCallRecord struct { - Name string `json:"name"` - Arguments map[string]any `json:"arguments"` - Result string `json:"result"` - Error string `json:"error,omitempty"` - Duration int64 `json:"durationMs"` -} - -// ============================================================================ -// Pagination Types -// ============================================================================ - -// AgentListItem is a lightweight agent representation for list views -type AgentListItem struct { - ID string `json:"id"` - Name string `json:"name"` - Description string `json:"description,omitempty"` - Status string `json:"status"` - HasWorkflow bool `json:"has_workflow"` - BlockCount int `json:"block_count"` - CreatedAt time.Time `json:"created_at"` - UpdatedAt time.Time `json:"updated_at"` -} - -// PaginatedAgentsResponse is the response for paginated agent list -type PaginatedAgentsResponse struct { - Agents []AgentListItem `json:"agents"` - Total int `json:"total"` - Limit int `json:"limit"` - Offset int `json:"offset"` - HasMore bool `json:"has_more"` -} - -// RecentAgentsResponse is the response for recent agents (landing page) -type RecentAgentsResponse struct { - Agents []AgentListItem `json:"agents"` -} - -// ============================================================================ -// Sync Types (for first-message persistence) -// ============================================================================ - -// SyncAgentRequest is the request body for syncing a local agent to backend -type SyncAgentRequest struct { - Name string `json:"name"` - Description string `json:"description,omitempty"` - Workflow SaveWorkflowRequest `json:"workflow"` - ModelID string `json:"model_id,omitempty"` -} - -// SyncAgentResponse is the response after syncing an agent -type SyncAgentResponse struct { - Agent *Agent `json:"agent"` - Workflow *Workflow `json:"workflow"` - ConversationID string `json:"conversation_id"` -} diff --git a/backend/internal/models/apikey.go b/backend/internal/models/apikey.go deleted file mode 100644 index 9bcf25fd..00000000 --- a/backend/internal/models/apikey.go +++ /dev/null @@ -1,195 +0,0 @@ -package models - -import ( - "time" - - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// APIKey represents an API key for programmatic access -type APIKey struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - UserID string `bson:"userId" json:"userId"` - - // Key info (hash stored, never plain text) - KeyPrefix string `bson:"keyPrefix" json:"keyPrefix"` // First 8 chars for display (e.g., "clv_a1b2") - KeyHash string `bson:"keyHash" json:"-"` // bcrypt hash, never exposed in JSON - PlainKey string `bson:"plainKey,omitempty" json:"key"` // TEMPORARY: Plain key for early platform phase - - // Metadata - Name string `bson:"name" json:"name"` - Description string `bson:"description,omitempty" json:"description,omitempty"` - - // Permissions - Scopes []string `bson:"scopes" json:"scopes"` // e.g., ["execute:*"], ["execute:agent-123", "read:executions"] - - // Rate limits (tier-based defaults can be overridden) - RateLimit *APIKeyRateLimit `bson:"rateLimit,omitempty" json:"rateLimit,omitempty"` - - // Status - LastUsedAt *time.Time `bson:"lastUsedAt,omitempty" json:"lastUsedAt,omitempty"` - RevokedAt *time.Time `bson:"revokedAt,omitempty" json:"revokedAt,omitempty"` // Soft delete - ExpiresAt *time.Time `bson:"expiresAt,omitempty" json:"expiresAt,omitempty"` // Optional expiration - - CreatedAt time.Time `bson:"createdAt" json:"createdAt"` - UpdatedAt time.Time `bson:"updatedAt" json:"updatedAt"` -} - -// APIKeyRateLimit defines custom rate limits for an API key -type APIKeyRateLimit struct { - RequestsPerMinute int64 `bson:"requestsPerMinute" json:"requestsPerMinute"` - RequestsPerHour int64 `bson:"requestsPerHour" json:"requestsPerHour"` -} - -// IsRevoked returns true if the API key has been revoked -func (k *APIKey) IsRevoked() bool { - return k.RevokedAt != nil -} - -// IsExpired returns true if the API key has expired -func (k *APIKey) IsExpired() bool { - if k.ExpiresAt == nil { - return false - } - return time.Now().After(*k.ExpiresAt) -} - -// IsValid returns true if the API key is not revoked and not expired -func (k *APIKey) IsValid() bool { - return !k.IsRevoked() && !k.IsExpired() -} - -// HasScope checks if the API key has a specific scope -func (k *APIKey) HasScope(scope string) bool { - for _, s := range k.Scopes { - if s == scope || s == "*" { - return true - } - // Check wildcard patterns like "execute:*" - if matchWildcardScope(s, scope) { - return true - } - } - return false -} - -// HasExecuteScope checks if the API key can execute a specific agent -func (k *APIKey) HasExecuteScope(agentID string) bool { - // Check for universal execute permission - if k.HasScope("execute:*") { - return true - } - // Check for specific agent permission - return k.HasScope("execute:" + agentID) -} - -// HasReadExecutionsScope checks if the API key can read executions -func (k *APIKey) HasReadExecutionsScope() bool { - return k.HasScope("read:executions") || k.HasScope("read:*") || k.HasScope("*") -} - -// matchWildcardScope checks if a wildcard scope matches a target scope -// e.g., "execute:*" matches "execute:agent-123" -func matchWildcardScope(pattern, target string) bool { - if len(pattern) < 2 || pattern[len(pattern)-1] != '*' { - return false - } - prefix := pattern[:len(pattern)-1] // Remove the '*' - return len(target) >= len(prefix) && target[:len(prefix)] == prefix -} - -// CreateAPIKeyRequest is the request body for creating an API key -type CreateAPIKeyRequest struct { - Name string `json:"name"` - Description string `json:"description,omitempty"` - Scopes []string `json:"scopes"` // Required: what the key can do - RateLimit *APIKeyRateLimit `json:"rateLimit,omitempty"` // Optional: custom rate limits - ExpiresIn int `json:"expiresIn,omitempty"` // Optional: expiration in days -} - -// CreateAPIKeyResponse is returned after creating an API key -// This is the ONLY time the full key is returned -type CreateAPIKeyResponse struct { - ID string `json:"id"` - Key string `json:"key"` // Full API key (ONLY shown once) - KeyPrefix string `json:"keyPrefix"` // Display prefix - Name string `json:"name"` - Scopes []string `json:"scopes"` - ExpiresAt *time.Time `json:"expiresAt,omitempty"` - CreatedAt time.Time `json:"createdAt"` -} - -// APIKeyListItem is a safe representation of an API key for listing -// Never includes the key hash -type APIKeyListItem struct { - ID string `json:"id"` - KeyPrefix string `json:"keyPrefix"` - Key string `json:"key,omitempty"` // TEMPORARY: Plain key for early platform phase - Name string `json:"name"` - Description string `json:"description,omitempty"` - Scopes []string `json:"scopes"` - LastUsedAt *time.Time `json:"lastUsedAt,omitempty"` - ExpiresAt *time.Time `json:"expiresAt,omitempty"` - IsRevoked bool `json:"isRevoked"` - CreatedAt time.Time `json:"createdAt"` -} - -// ToListItem converts an APIKey to a safe list representation -func (k *APIKey) ToListItem() *APIKeyListItem { - return &APIKeyListItem{ - ID: k.ID.Hex(), - KeyPrefix: k.KeyPrefix, - Key: k.PlainKey, // TEMPORARY: Include plain key for early platform phase - Name: k.Name, - Description: k.Description, - Scopes: k.Scopes, - LastUsedAt: k.LastUsedAt, - ExpiresAt: k.ExpiresAt, - IsRevoked: k.IsRevoked(), - CreatedAt: k.CreatedAt, - } -} - -// TriggerAgentRequest is the request body for triggering an agent via API -type TriggerAgentRequest struct { - Input map[string]interface{} `json:"input,omitempty"` - - // EnableBlockChecker enables block completion validation (optional) - // When true, each block is checked to ensure it accomplished its job - EnableBlockChecker bool `json:"enable_block_checker,omitempty"` - - // CheckerModelID is the model to use for block checking (optional) - // Defaults to gpt-4o-mini for fast, cheap validation - CheckerModelID string `json:"checker_model_id,omitempty"` -} - -// TriggerAgentResponse is returned after triggering an agent -type TriggerAgentResponse struct { - ExecutionID string `json:"executionId"` - Status string `json:"status"` // "queued" or "running" - Message string `json:"message"` -} - -// ValidScopes lists all valid API key scopes -var ValidScopes = []string{ - "execute:*", // Execute any agent - "upload", // Upload files for workflow inputs - "read:executions", // Read execution history - "read:*", // Read all resources - "*", // Full access (admin) -} - -// IsValidScope checks if a scope is valid -func IsValidScope(scope string) bool { - // Check exact match - for _, valid := range ValidScopes { - if scope == valid { - return true - } - } - // Check agent-specific execute scope (execute:agent-xxx) - if len(scope) > 8 && scope[:8] == "execute:" { - return true - } - return false -} diff --git a/backend/internal/models/chat.go b/backend/internal/models/chat.go deleted file mode 100644 index 41f7a52e..00000000 --- a/backend/internal/models/chat.go +++ /dev/null @@ -1,178 +0,0 @@ -package models - -import ( - "time" - - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// EncryptedChat represents a chat stored in MongoDB with encrypted messages -type EncryptedChat struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - UserID string `bson:"userId" json:"user_id"` - ChatID string `bson:"chatId" json:"chat_id"` // Frontend-generated UUID - Title string `bson:"title" json:"title"` // Plaintext title for listing - EncryptedMessages string `bson:"encryptedMessages" json:"-"` // AES-256-GCM encrypted JSON array of messages - IsStarred bool `bson:"isStarred" json:"is_starred"` - Model string `bson:"model,omitempty" json:"model,omitempty"` // Selected model for this chat - Version int64 `bson:"version" json:"version"` // Optimistic locking - CreatedAt time.Time `bson:"createdAt" json:"created_at"` - UpdatedAt time.Time `bson:"updatedAt" json:"updated_at"` -} - -// ChatMessage represents a single message in a chat (unencrypted form) -type ChatMessage struct { - ID string `json:"id"` - Role string `json:"role"` // "user", "assistant", "system" - Content string `json:"content"` - Timestamp int64 `json:"timestamp"` // Unix milliseconds - IsStreaming bool `json:"isStreaming,omitempty"` - Status string `json:"status,omitempty"` // "sending", "sent", "error" - Error string `json:"error,omitempty"` - Attachments []ChatAttachment `json:"attachments,omitempty"` - ToolCalls []ToolCall `json:"toolCalls,omitempty"` - Reasoning string `json:"reasoning,omitempty"` // Thinking/reasoning process - Artifacts []Artifact `json:"artifacts,omitempty"` - AgentId string `json:"agentId,omitempty"` - AgentName string `json:"agentName,omitempty"` - AgentAvatar string `json:"agentAvatar,omitempty"` - - // Response versioning fields - VersionGroupId string `json:"versionGroupId,omitempty"` // Groups all versions of same response - VersionNumber int `json:"versionNumber,omitempty"` // 1, 2, 3... within the group - IsHidden bool `json:"isHidden,omitempty"` // Hidden versions (not current) - RetryType string `json:"retryType,omitempty"` // Type of retry: regenerate, add_details, etc. -} - -// ToolCall represents a tool invocation in a message -type ToolCall struct { - ID string `json:"id"` - Name string `json:"name"` - DisplayName string `json:"displayName,omitempty"` - Icon string `json:"icon,omitempty"` - Status string `json:"status"` // "executing", "completed" - Query string `json:"query,omitempty"` - Result string `json:"result,omitempty"` - Plots []PlotData `json:"plots,omitempty"` - Timestamp int64 `json:"timestamp"` - IsExpanded bool `json:"isExpanded,omitempty"` -} - -// Artifact represents renderable content (HTML, SVG, Mermaid) -type Artifact struct { - ID string `json:"id"` - Type string `json:"type"` // "html", "svg", "mermaid", "image" - Title string `json:"title"` - Content string `json:"content"` - Images []ArtifactImage `json:"images,omitempty"` - Metadata map[string]interface{} `json:"metadata,omitempty"` -} - -// ArtifactImage represents an image in an artifact -type ArtifactImage struct { - Data string `json:"data"` // Base64-encoded - Format string `json:"format"` // png, jpg, svg - Caption string `json:"caption,omitempty"` -} - -// ChatAttachment represents a file attached to a message -type ChatAttachment struct { - FileID string `json:"file_id"` - Type string `json:"type"` // Attachment type: "image", "document", "data" - URL string `json:"url"` - MimeType string `json:"mime_type"` - Size int64 `json:"size"` - Filename string `json:"filename,omitempty"` - Expired bool `json:"expired,omitempty"` - // Document-specific fields - PageCount int `json:"page_count,omitempty"` - WordCount int `json:"word_count,omitempty"` - Preview string `json:"preview,omitempty"` // Text preview or thumbnail - // Data file-specific fields - DataPreview *DataPreview `json:"data_preview,omitempty"` -} - -// DataPreview represents a preview of CSV/tabular data -type DataPreview struct { - Headers []string `json:"headers"` - Rows [][]string `json:"rows"` - RowCount int `json:"row_count"` // Total rows in file - ColCount int `json:"col_count"` // Total columns -} - -// ChatResponse is the decrypted chat returned to the frontend -type ChatResponse struct { - ID string `json:"id"` - Title string `json:"title"` - Messages []ChatMessage `json:"messages"` - IsStarred bool `json:"is_starred"` - Model string `json:"model,omitempty"` - Version int64 `json:"version"` - CreatedAt time.Time `json:"created_at"` - UpdatedAt time.Time `json:"updated_at"` -} - -// ChatListItem is a summary of a chat for listing (no messages) -type ChatListItem struct { - ID string `json:"id"` - Title string `json:"title"` - IsStarred bool `json:"is_starred"` - Model string `json:"model,omitempty"` - MessageCount int `json:"message_count"` - Version int64 `json:"version"` - CreatedAt time.Time `json:"created_at"` - UpdatedAt time.Time `json:"updated_at"` -} - -// CreateChatRequest is the request body for creating/updating a chat -type CreateChatRequest struct { - ID string `json:"id"` // Frontend-generated UUID - Title string `json:"title"` - Messages []ChatMessage `json:"messages"` - IsStarred bool `json:"is_starred"` - Model string `json:"model,omitempty"` - Version int64 `json:"version,omitempty"` // For optimistic locking on updates -} - -// UpdateChatRequest is the request body for partial chat updates -type UpdateChatRequest struct { - Title *string `json:"title,omitempty"` - IsStarred *bool `json:"is_starred,omitempty"` - Model *string `json:"model,omitempty"` - Version int64 `json:"version"` // Required for optimistic locking -} - -// ChatAddMessageRequest is the request body for adding a single message to a synced chat -type ChatAddMessageRequest struct { - Message ChatMessage `json:"message"` - Version int64 `json:"version"` // For optimistic locking -} - -// BulkSyncRequest is the request body for uploading multiple chats -type BulkSyncRequest struct { - Chats []CreateChatRequest `json:"chats"` -} - -// BulkSyncResponse is the response for bulk sync operation -type BulkSyncResponse struct { - Synced int `json:"synced"` - Failed int `json:"failed"` - Errors []string `json:"errors,omitempty"` - ChatIDs []string `json:"chat_ids"` // IDs of successfully synced chats -} - -// SyncAllResponse is the response for fetching all chats for initial sync -type SyncAllResponse struct { - Chats []ChatResponse `json:"chats"` - TotalCount int `json:"total_count"` - SyncedAt time.Time `json:"synced_at"` -} - -// ChatListResponse is the paginated response for listing chats -type ChatListResponse struct { - Chats []ChatListItem `json:"chats"` - TotalCount int64 `json:"total_count"` - Page int `json:"page"` - PageSize int `json:"page_size"` - HasMore bool `json:"has_more"` -} diff --git a/backend/internal/models/config.go b/backend/internal/models/config.go deleted file mode 100644 index 7925935e..00000000 --- a/backend/internal/models/config.go +++ /dev/null @@ -1,8 +0,0 @@ -package models - -// Config represents API configuration (legacy, kept for backward compatibility) -type Config struct { - BaseURL string `json:"base_url"` - APIKey string `json:"api_key"` - Model string `json:"model"` -} diff --git a/backend/internal/models/conversation.go b/backend/internal/models/conversation.go deleted file mode 100644 index 8570d5b0..00000000 --- a/backend/internal/models/conversation.go +++ /dev/null @@ -1,88 +0,0 @@ -package models - -import ( - "time" - - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// BuilderConversation represents a chat history for building an agent -type BuilderConversation struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - AgentID string `bson:"agentId" json:"agent_id"` // String ID (timestamp-based from SQLite) - UserID string `bson:"userId" json:"user_id"` // Supabase user ID - Messages []BuilderMessage `bson:"messages" json:"messages"` - ModelID string `bson:"modelId" json:"model_id"` - CreatedAt time.Time `bson:"createdAt" json:"created_at"` - UpdatedAt time.Time `bson:"updatedAt" json:"updated_at"` - ExpiresAt *time.Time `bson:"expiresAt,omitempty" json:"expires_at,omitempty"` // TTL for auto-deletion if user opts out -} - -// BuilderMessage represents a single message in the builder conversation -type BuilderMessage struct { - ID string `bson:"id" json:"id"` - Role string `bson:"role" json:"role"` // "user" or "assistant" - Content string `bson:"content" json:"content"` - Timestamp time.Time `bson:"timestamp" json:"timestamp"` - WorkflowSnapshot *WorkflowSnapshot `bson:"workflowSnapshot,omitempty" json:"workflow_snapshot,omitempty"` -} - -// WorkflowSnapshot captures the state of the workflow at a message point -type WorkflowSnapshot struct { - Version int `bson:"version" json:"version"` - Action string `bson:"action,omitempty" json:"action,omitempty"` // "create" or "modify" or null - Explanation string `bson:"explanation,omitempty" json:"explanation,omitempty"` -} - -// EncryptedBuilderConversation stores encrypted conversation data -// The Messages field is encrypted as a JSON blob -type EncryptedBuilderConversation struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - AgentID string `bson:"agentId" json:"agent_id"` // String ID (timestamp-based from SQLite) - UserID string `bson:"userId" json:"user_id"` // Supabase user ID - EncryptedMessages string `bson:"encryptedMessages" json:"-"` // Base64-encoded encrypted JSON - ModelID string `bson:"modelId" json:"model_id"` - MessageCount int `bson:"messageCount" json:"message_count"` // For display without decryption - CreatedAt time.Time `bson:"createdAt" json:"created_at"` - UpdatedAt time.Time `bson:"updatedAt" json:"updated_at"` - ExpiresAt *time.Time `bson:"expiresAt,omitempty" json:"expires_at,omitempty"` -} - -// ConversationListItem is a summary for listing conversations -type ConversationListItem struct { - ID string `json:"id"` - AgentID string `json:"agent_id"` - ModelID string `json:"model_id"` - MessageCount int `json:"message_count"` - CreatedAt time.Time `json:"created_at"` - UpdatedAt time.Time `json:"updated_at"` -} - -// AddMessageRequest is the request body for adding a message to a conversation -type AddMessageRequest struct { - Role string `json:"role"` - Content string `json:"content"` - WorkflowSnapshot *WorkflowSnapshot `json:"workflow_snapshot,omitempty"` -} - -// ConversationResponse is the full conversation response -type ConversationResponse struct { - ID string `json:"id"` - AgentID string `json:"agent_id"` - ModelID string `json:"model_id"` - Messages []BuilderMessage `json:"messages"` - CreatedAt time.Time `json:"created_at"` - UpdatedAt time.Time `json:"updated_at"` -} - -// ToListItem converts an EncryptedBuilderConversation to ConversationListItem -func (c *EncryptedBuilderConversation) ToListItem() ConversationListItem { - return ConversationListItem{ - ID: c.ID.Hex(), - AgentID: c.AgentID, // AgentID is already a string - ModelID: c.ModelID, - MessageCount: c.MessageCount, - CreatedAt: c.CreatedAt, - UpdatedAt: c.UpdatedAt, - } -} diff --git a/backend/internal/models/credential.go b/backend/internal/models/credential.go deleted file mode 100644 index 510ecf5c..00000000 --- a/backend/internal/models/credential.go +++ /dev/null @@ -1,153 +0,0 @@ -package models - -import ( - "time" - - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// Credential represents an encrypted credential for external integrations -type Credential struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - UserID string `bson:"userId" json:"userId"` - Name string `bson:"name" json:"name"` - IntegrationType string `bson:"integrationType" json:"integrationType"` - - // Encrypted data - NEVER exposed to frontend or LLM - EncryptedData string `bson:"encryptedData" json:"-"` // json:"-" ensures it's never serialized - - // Metadata (safe to expose) - Metadata CredentialMetadata `bson:"metadata" json:"metadata"` - - CreatedAt time.Time `bson:"createdAt" json:"createdAt"` - UpdatedAt time.Time `bson:"updatedAt" json:"updatedAt"` -} - -// CredentialMetadata contains non-sensitive information about a credential -type CredentialMetadata struct { - MaskedPreview string `bson:"maskedPreview" json:"maskedPreview"` // e.g., "https://discord...xxx" - Icon string `bson:"icon,omitempty" json:"icon,omitempty"` - LastUsedAt *time.Time `bson:"lastUsedAt,omitempty" json:"lastUsedAt,omitempty"` - UsageCount int64 `bson:"usageCount" json:"usageCount"` - LastTestAt *time.Time `bson:"lastTestAt,omitempty" json:"lastTestAt,omitempty"` - TestStatus string `bson:"testStatus,omitempty" json:"testStatus,omitempty"` // "success", "failed", "pending" -} - -// CredentialListItem is a safe representation for listing credentials -// Never includes the encrypted data -type CredentialListItem struct { - ID string `json:"id"` - UserID string `json:"userId"` - Name string `json:"name"` - IntegrationType string `json:"integrationType"` - Metadata CredentialMetadata `json:"metadata"` - CreatedAt time.Time `json:"createdAt"` - UpdatedAt time.Time `json:"updatedAt"` -} - -// ToListItem converts a Credential to a safe list representation -func (c *Credential) ToListItem() *CredentialListItem { - return &CredentialListItem{ - ID: c.ID.Hex(), - UserID: c.UserID, - Name: c.Name, - IntegrationType: c.IntegrationType, - Metadata: c.Metadata, - CreatedAt: c.CreatedAt, - UpdatedAt: c.UpdatedAt, - } -} - -// DecryptedCredential is used internally by tools to access credential data -// This should NEVER be returned to the frontend or exposed to the LLM -type DecryptedCredential struct { - ID string `json:"id"` - Name string `json:"name"` - IntegrationType string `json:"integrationType"` - Data map[string]interface{} `json:"data"` // The actual credential values -} - -// CreateCredentialRequest is the request body for creating a credential -type CreateCredentialRequest struct { - Name string `json:"name" validate:"required,min=1,max=100"` - IntegrationType string `json:"integrationType" validate:"required"` - Data map[string]interface{} `json:"data" validate:"required"` // Will be encrypted -} - -// UpdateCredentialRequest is the request body for updating a credential -type UpdateCredentialRequest struct { - Name string `json:"name,omitempty"` - Data map[string]interface{} `json:"data,omitempty"` // Will be re-encrypted if provided -} - -// TestCredentialResponse is returned after testing a credential -type TestCredentialResponse struct { - Success bool `json:"success"` - Message string `json:"message"` - Details string `json:"details,omitempty"` // Additional info (sanitized) -} - -// CredentialReference is used in block configs to reference credentials -// The LLM only sees the Name, tools use the ID to fetch the actual credential -type CredentialReference struct { - ID string `json:"id"` - Name string `json:"name"` - IntegrationType string `json:"integrationType"` -} - -// Integration represents a supported external integration type -type Integration struct { - ID string `json:"id"` // e.g., "discord", "notion" - Name string `json:"name"` // e.g., "Discord", "Notion" - Description string `json:"description"` // Short description - Icon string `json:"icon"` // Icon identifier (lucide or custom) - Category string `json:"category"` // e.g., "communication", "productivity" - Fields []IntegrationField `json:"fields"` // Required/optional fields - Tools []string `json:"tools"` // Which tools use this integration - DocsURL string `json:"docsUrl,omitempty"` - ComingSoon bool `json:"comingSoon,omitempty"` // If true, integration is not yet available -} - -// IntegrationField defines a field required for an integration -type IntegrationField struct { - Key string `json:"key"` // e.g., "webhook_url", "api_key" - Label string `json:"label"` // e.g., "Webhook URL", "API Key" - Type string `json:"type"` // "api_key", "webhook_url", "token", "text", "select", "json" - Required bool `json:"required"` // Is this field required? - Placeholder string `json:"placeholder,omitempty"` // Placeholder text - HelpText string `json:"helpText,omitempty"` // Help text for the user - Options []string `json:"options,omitempty"` // For select type - Default string `json:"default,omitempty"` // Default value - Sensitive bool `json:"sensitive"` // Should this be masked in UI? -} - -// IntegrationCategory represents a category of integrations -type IntegrationCategory struct { - ID string `json:"id"` - Name string `json:"name"` - Icon string `json:"icon"` - Integrations []Integration `json:"integrations"` -} - -// GetIntegrationsResponse is the response for listing available integrations -type GetIntegrationsResponse struct { - Categories []IntegrationCategory `json:"categories"` -} - -// GetCredentialsResponse is the response for listing user credentials -type GetCredentialsResponse struct { - Credentials []*CredentialListItem `json:"credentials"` - Total int `json:"total"` -} - -// CredentialsByIntegration groups credentials by integration type -type CredentialsByIntegration struct { - IntegrationType string `json:"integrationType"` - Integration Integration `json:"integration"` - Credentials []*CredentialListItem `json:"credentials"` -} - -// GetCredentialsByIntegrationResponse groups credentials by integration -type GetCredentialsByIntegrationResponse struct { - Integrations []CredentialsByIntegration `json:"integrations"` -} diff --git a/backend/internal/models/integration_registry.go b/backend/internal/models/integration_registry.go deleted file mode 100644 index 4b8af1b0..00000000 --- a/backend/internal/models/integration_registry.go +++ /dev/null @@ -1,1323 +0,0 @@ -package models - -// IntegrationRegistry contains all supported integrations -// This is the source of truth for what integrations are available -var IntegrationRegistry = map[string]Integration{ - // ============================================ - // COMMUNICATION - // ============================================ - "discord": { - ID: "discord", - Name: "Discord", - Description: "Send messages to Discord channels via webhooks", - Icon: "discord", - Category: "communication", - Fields: []IntegrationField{ - { - Key: "webhook_url", - Label: "Webhook URL", - Type: "webhook_url", - Required: true, - Placeholder: "https://discord.com/api/webhooks/...", - HelpText: "Create a webhook in Discord: Server Settings → Integrations → Webhooks", - Sensitive: true, - }, - }, - Tools: []string{"send_discord_message"}, - DocsURL: "https://support.discord.com/hc/en-us/articles/228383668-Intro-to-Webhooks", - }, - - "slack": { - ID: "slack", - Name: "Slack", - Description: "Send messages to Slack channels via webhooks", - Icon: "slack", - Category: "communication", - Fields: []IntegrationField{ - { - Key: "webhook_url", - Label: "Webhook URL", - Type: "webhook_url", - Required: true, - Placeholder: "https://hooks.slack.com/services/...", - HelpText: "Create an Incoming Webhook in your Slack App settings", - Sensitive: true, - }, - }, - Tools: []string{"send_slack_message"}, - DocsURL: "https://api.slack.com/messaging/webhooks", - }, - - "telegram": { - ID: "telegram", - Name: "Telegram", - Description: "Send messages via Telegram Bot API", - Icon: "telegram", - Category: "communication", - Fields: []IntegrationField{ - { - Key: "bot_token", - Label: "Bot Token", - Type: "api_key", - Required: true, - Placeholder: "123456789:ABCdefGHIjklMNOpqrsTUVwxyz", - HelpText: "Get your bot token from @BotFather on Telegram", - Sensitive: true, - }, - { - Key: "chat_id", - Label: "Chat ID", - Type: "text", - Required: true, - Placeholder: "-1001234567890", - HelpText: "The chat ID where messages will be sent (use @userinfobot to find it)", - Sensitive: false, - }, - }, - Tools: []string{"send_telegram_message"}, - DocsURL: "https://core.telegram.org/bots/api", - }, - - "teams": { - ID: "teams", - Name: "Microsoft Teams", - Description: "Send messages to Microsoft Teams channels", - Icon: "microsoft", - Category: "communication", - Fields: []IntegrationField{ - { - Key: "webhook_url", - Label: "Webhook URL", - Type: "webhook_url", - Required: true, - Placeholder: "https://outlook.office.com/webhook/...", - HelpText: "Create an Incoming Webhook connector in your Teams channel", - Sensitive: true, - }, - }, - Tools: []string{"send_teams_message"}, - DocsURL: "https://learn.microsoft.com/en-us/microsoftteams/platform/webhooks-and-connectors/how-to/add-incoming-webhook", - }, - - "google_chat": { - ID: "google_chat", - Name: "Google Chat", - Description: "Send messages to Google Chat spaces via webhooks", - Icon: "google", - Category: "communication", - Fields: []IntegrationField{ - { - Key: "webhook_url", - Label: "Webhook URL", - Type: "webhook_url", - Required: true, - Placeholder: "https://chat.googleapis.com/v1/spaces/.../messages?key=...", - HelpText: "Create a webhook in Google Chat: Space settings → Integrations → Webhooks", - Sensitive: true, - }, - }, - Tools: []string{"send_google_chat_message"}, - DocsURL: "https://developers.google.com/chat/how-tos/webhooks", - }, - - "zoom": { - ID: "zoom", - Name: "Zoom", - Description: "Create and manage Zoom meetings, handle registrations, and schedule video conferences", - Icon: "zoom", - Category: "communication", - Fields: []IntegrationField{ - { - Key: "account_id", - Label: "Account ID", - Type: "text", - Required: true, - Placeholder: "Your Zoom Account ID", - HelpText: "Find in Zoom Marketplace: Your app → App Credentials → Account ID", - Sensitive: false, - }, - { - Key: "client_id", - Label: "Client ID", - Type: "api_key", - Required: true, - Placeholder: "Your Zoom Client ID", - HelpText: "Find in Zoom Marketplace: Your app → App Credentials → Client ID", - Sensitive: true, - }, - { - Key: "client_secret", - Label: "Client Secret", - Type: "api_key", - Required: true, - Placeholder: "Your Zoom Client Secret", - HelpText: "Find in Zoom Marketplace: Your app → App Credentials → Client Secret", - Sensitive: true, - }, - }, - Tools: []string{"zoom_meeting"}, - DocsURL: "https://developers.zoom.us/docs/internal-apps/s2s-oauth/", - }, - - "twilio": { - ID: "twilio", - Name: "Twilio", - Description: "Send SMS, MMS, and WhatsApp messages via Twilio API", - Icon: "twilio", - Category: "communication", - Fields: []IntegrationField{ - { - Key: "account_sid", - Label: "Account SID", - Type: "text", - Required: true, - Placeholder: "ACxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx", - HelpText: "Find in Twilio Console: Account → Account SID", - Sensitive: false, - }, - { - Key: "auth_token", - Label: "Auth Token", - Type: "api_key", - Required: true, - Placeholder: "Your Twilio Auth Token", - HelpText: "Find in Twilio Console: Account → Auth Token", - Sensitive: true, - }, - { - Key: "from_number", - Label: "Default From Number", - Type: "text", - Required: false, - Placeholder: "+1234567890", - HelpText: "Default phone number to send messages from (must be a Twilio number)", - Sensitive: false, - }, - }, - Tools: []string{"twilio_send_sms", "twilio_send_whatsapp"}, - DocsURL: "https://www.twilio.com/docs/sms/api", - }, - - "referralmonk": { - ID: "referralmonk", - Name: "ReferralMonk", - Description: "Send WhatsApp messages via ReferralMonk with template support for campaigns and nurture flows", - Icon: "message-square", - Category: "communication", - Fields: []IntegrationField{ - { - Key: "api_token", - Label: "API Token", - Type: "api_key", - Required: true, - Placeholder: "Your ReferralMonk API Token", - HelpText: "Get your API credentials from ReferralMonk dashboard (AhaGuru instance)", - Sensitive: true, - }, - { - Key: "api_secret", - Label: "API Secret", - Type: "api_key", - Required: true, - Placeholder: "Your ReferralMonk API Secret", - HelpText: "Your API secret key from ReferralMonk dashboard", - Sensitive: true, - }, - }, - Tools: []string{"referralmonk_whatsapp"}, - DocsURL: "https://ahaguru.referralmonk.com/", - }, - - // ============================================ - // PRODUCTIVITY - // ============================================ - "notion": { - ID: "notion", - Name: "Notion", - Description: "Read and write to Notion databases and pages", - Icon: "notion", - Category: "productivity", - Fields: []IntegrationField{ - { - Key: "api_key", - Label: "Integration Token", - Type: "api_key", - Required: true, - Placeholder: "secret_...", - HelpText: "Create an integration at notion.so/my-integrations and share pages with it", - Sensitive: true, - }, - }, - Tools: []string{"notion_search", "notion_query_database", "notion_create_page", "notion_update_page"}, - DocsURL: "https://developers.notion.com/docs/getting-started", - }, - - "airtable": { - ID: "airtable", - Name: "Airtable", - Description: "Read and write to Airtable bases", - Icon: "airtable", - Category: "productivity", - Fields: []IntegrationField{ - { - Key: "api_key", - Label: "Personal Access Token", - Type: "api_key", - Required: true, - Placeholder: "pat...", - HelpText: "Create a Personal Access Token in your Airtable account settings", - Sensitive: true, - }, - { - Key: "base_id", - Label: "Base ID", - Type: "text", - Required: false, - Placeholder: "appXXXXXXXXXXXXXX", - HelpText: "Optional: Default base ID (can be overridden per request)", - Sensitive: false, - }, - }, - Tools: []string{"airtable_list", "airtable_read", "airtable_create", "airtable_update"}, - DocsURL: "https://airtable.com/developers/web/api/introduction", - }, - - "trello": { - ID: "trello", - Name: "Trello", - Description: "Manage Trello boards, lists, and cards", - Icon: "trello", - Category: "productivity", - Fields: []IntegrationField{ - { - Key: "api_key", - Label: "API Key", - Type: "api_key", - Required: true, - Placeholder: "Your Trello API key", - HelpText: "Get your API key from trello.com/app-key", - Sensitive: true, - }, - { - Key: "token", - Label: "Token", - Type: "token", - Required: true, - Placeholder: "Your Trello token", - HelpText: "Generate a token using your API key", - Sensitive: true, - }, - }, - Tools: []string{"trello_boards", "trello_lists", "trello_cards", "trello_create_card"}, - DocsURL: "https://developer.atlassian.com/cloud/trello/rest/", - }, - - "clickup": { - ID: "clickup", - Name: "ClickUp", - Description: "Manage ClickUp tasks, lists, and spaces", - Icon: "clickup", - Category: "productivity", - Fields: []IntegrationField{ - { - Key: "api_key", - Label: "API Key", - Type: "api_key", - Required: true, - Placeholder: "pk_...", - HelpText: "Get your API key from ClickUp: Settings → Apps → API Token", - Sensitive: true, - }, - }, - Tools: []string{"clickup_tasks", "clickup_create_task", "clickup_update_task"}, - DocsURL: "https://clickup.com/api", - }, - - "calendly": { - ID: "calendly", - Name: "Calendly", - Description: "Manage Calendly events, scheduling links, and invitees", - Icon: "calendly", - Category: "productivity", - Fields: []IntegrationField{ - { - Key: "api_key", - Label: "Personal Access Token", - Type: "api_key", - Required: true, - Placeholder: "eyJraW...", - HelpText: "Get your token from Calendly: Integrations → API & Webhooks → Personal Access Tokens", - Sensitive: true, - }, - }, - Tools: []string{"calendly_events", "calendly_event_types", "calendly_invitees"}, - DocsURL: "https://developer.calendly.com/api-docs", - }, - - "composio_googlesheets": { - ID: "composio_googlesheets", - Name: "Google Sheets", - Description: "Complete Google Sheets integration via Composio OAuth - no GCP setup required. Create, read, write, search, and manage spreadsheets.", - Icon: "file-spreadsheet", - Category: "productivity", - Fields: []IntegrationField{ - { - Key: "composio_entity_id", - Label: "Entity ID", - Type: "text", - Required: true, - Placeholder: "Automatically filled after OAuth", - HelpText: "Connect your Google account via Composio OAuth (managed by ClaraVerse)", - Sensitive: false, - }, - }, - Tools: []string{ - "googlesheets_read", - "googlesheets_write", - "googlesheets_append", - "googlesheets_create", - "googlesheets_get_info", - "googlesheets_list_sheets", - "googlesheets_search", - "googlesheets_clear", - "googlesheets_add_sheet", - "googlesheets_delete_sheet", - "googlesheets_find_replace", - "googlesheets_upsert_rows", - }, - DocsURL: "https://docs.composio.dev/toolkits/googlesheets", - }, - - "composio_gmail": { - ID: "composio_gmail", - Name: "Gmail", - Description: "Complete Gmail integration via Composio OAuth - no GCP setup required. Send, fetch, reply, manage drafts, and organize emails.", - Icon: "mail", - Category: "communication", - Fields: []IntegrationField{ - { - Key: "composio_entity_id", - Label: "Entity ID", - Type: "text", - Required: true, - Placeholder: "Automatically filled after OAuth", - HelpText: "Connect your Gmail account via Composio OAuth (managed by ClaraVerse)", - Sensitive: false, - }, - }, - Tools: []string{ - "gmail_send_email", - "gmail_fetch_emails", - "gmail_get_message", - "gmail_reply_to_thread", - "gmail_create_draft", - "gmail_send_draft", - "gmail_list_drafts", - "gmail_add_label", - "gmail_list_labels", - "gmail_move_to_trash", - }, - DocsURL: "https://docs.composio.dev/toolkits/gmail", - }, - - // ============================================ - // DEVELOPMENT - // ============================================ - "github": { - ID: "github", - Name: "GitHub", - Description: "Access GitHub repositories, issues, and pull requests", - Icon: "github", - Category: "development", - Fields: []IntegrationField{ - { - Key: "personal_access_token", - Label: "Personal Access Token", - Type: "api_key", - Required: true, - Placeholder: "ghp_...", - HelpText: "Create a PAT at github.com/settings/tokens with required scopes", - Sensitive: true, - }, - }, - Tools: []string{"github_create_issue", "github_list_issues", "github_get_repo", "github_add_comment"}, - DocsURL: "https://docs.github.com/en/rest", - }, - - "gitlab": { - ID: "gitlab", - Name: "GitLab", - Description: "Access GitLab projects, issues, and merge requests", - Icon: "gitlab", - Category: "development", - Fields: []IntegrationField{ - { - Key: "personal_access_token", - Label: "Personal Access Token", - Type: "api_key", - Required: true, - Placeholder: "glpat-...", - HelpText: "Create a PAT in GitLab: Settings → Access Tokens", - Sensitive: true, - }, - { - Key: "base_url", - Label: "GitLab URL", - Type: "text", - Required: false, - Placeholder: "https://gitlab.com", - HelpText: "Leave empty for gitlab.com, or enter your self-hosted URL", - Default: "https://gitlab.com", - Sensitive: false, - }, - }, - Tools: []string{"gitlab_projects", "gitlab_issues", "gitlab_mrs"}, - DocsURL: "https://docs.gitlab.com/ee/api/", - }, - - "linear": { - ID: "linear", - Name: "Linear", - Description: "Manage Linear issues and projects", - Icon: "linear", - Category: "development", - Fields: []IntegrationField{ - { - Key: "api_key", - Label: "API Key", - Type: "api_key", - Required: true, - Placeholder: "lin_api_...", - HelpText: "Create an API key in Linear: Settings → API → Personal API keys", - Sensitive: true, - }, - }, - Tools: []string{"linear_issues", "linear_create_issue", "linear_update_issue"}, - DocsURL: "https://developers.linear.app/docs/graphql/working-with-the-graphql-api", - }, - - "jira": { - ID: "jira", - Name: "Jira", - Description: "Manage Jira issues and projects", - Icon: "jira", - Category: "development", - Fields: []IntegrationField{ - { - Key: "email", - Label: "Email", - Type: "text", - Required: true, - Placeholder: "your@email.com", - HelpText: "Your Atlassian account email", - Sensitive: false, - }, - { - Key: "api_token", - Label: "API Token", - Type: "api_key", - Required: true, - Placeholder: "Your Jira API token", - HelpText: "Create an API token at id.atlassian.com/manage-profile/security/api-tokens", - Sensitive: true, - }, - { - Key: "domain", - Label: "Jira Domain", - Type: "text", - Required: true, - Placeholder: "your-company.atlassian.net", - HelpText: "Your Jira Cloud domain (without https://)", - Sensitive: false, - }, - }, - Tools: []string{"jira_issues", "jira_create_issue", "jira_update_issue"}, - DocsURL: "https://developer.atlassian.com/cloud/jira/platform/rest/v3/intro/", - }, - - // ============================================ - // CRM / SALES - // ============================================ - "hubspot": { - ID: "hubspot", - Name: "HubSpot", - Description: "Access HubSpot CRM contacts, deals, and companies", - Icon: "hubspot", - Category: "crm", - Fields: []IntegrationField{ - { - Key: "access_token", - Label: "Private App Access Token", - Type: "api_key", - Required: true, - Placeholder: "pat-...", - HelpText: "Create a Private App in HubSpot: Settings → Integrations → Private Apps", - Sensitive: true, - }, - }, - Tools: []string{"hubspot_contacts", "hubspot_deals", "hubspot_companies"}, - DocsURL: "https://developers.hubspot.com/docs/api/overview", - }, - - "leadsquared": { - ID: "leadsquared", - Name: "LeadSquared", - Description: "Access LeadSquared CRM leads and activities", - Icon: "leadsquared", - Category: "crm", - Fields: []IntegrationField{ - { - Key: "access_key", - Label: "Access Key", - Type: "api_key", - Required: true, - Placeholder: "Your LeadSquared Access Key", - HelpText: "Find in LeadSquared: Settings → API & Webhooks → Access Keys", - Sensitive: true, - }, - { - Key: "secret_key", - Label: "Secret Key", - Type: "api_key", - Required: true, - Placeholder: "Your LeadSquared Secret Key", - HelpText: "Find alongside your Access Key", - Sensitive: true, - }, - { - Key: "host", - Label: "API Host", - Type: "text", - Required: true, - Placeholder: "api.leadsquared.com", - HelpText: "Your LeadSquared API host (varies by region)", - Default: "api.leadsquared.com", - Sensitive: false, - }, - }, - Tools: []string{"leadsquared_leads", "leadsquared_create_lead", "leadsquared_activities"}, - DocsURL: "https://apidocs.leadsquared.com/", - }, - - // ============================================ - // MARKETING / EMAIL - // ============================================ - "sendgrid": { - ID: "sendgrid", - Name: "SendGrid", - Description: "Send emails via SendGrid API. Supports HTML/text emails, multiple recipients, CC/BCC, and attachments.", - Icon: "sendgrid", - Category: "marketing", - Fields: []IntegrationField{ - { - Key: "api_key", - Label: "API Key", - Type: "api_key", - Required: true, - Placeholder: "SG...", - HelpText: "Create an API key in SendGrid: Settings → API Keys", - Sensitive: true, - }, - { - Key: "from_email", - Label: "Default From Email", - Type: "text", - Required: false, - Placeholder: "noreply@yourdomain.com", - HelpText: "Default sender email (must be verified in SendGrid)", - Sensitive: false, - }, - }, - Tools: []string{"send_email"}, - DocsURL: "https://docs.sendgrid.com/api-reference/mail-send/mail-send", - }, - - "brevo": { - ID: "brevo", - Name: "Brevo", - Description: "Send transactional and marketing emails via Brevo (formerly SendInBlue). Supports templates, tracking, and automation.", - Icon: "brevo", - Category: "marketing", - Fields: []IntegrationField{ - { - Key: "api_key", - Label: "API Key", - Type: "api_key", - Required: true, - Placeholder: "xkeysib-...", - HelpText: "Create an API key in Brevo: Settings → SMTP & API → API Keys", - Sensitive: true, - }, - { - Key: "from_email", - Label: "Default From Email", - Type: "text", - Required: false, - Placeholder: "noreply@yourdomain.com", - HelpText: "Default sender email (must be verified in Brevo)", - Sensitive: false, - }, - { - Key: "from_name", - Label: "Default From Name", - Type: "text", - Required: false, - Placeholder: "My Company", - HelpText: "Default sender display name", - Sensitive: false, - }, - }, - Tools: []string{"send_brevo_email"}, - DocsURL: "https://developers.brevo.com/docs/send-a-transactional-email", - }, - - "mailchimp": { - ID: "mailchimp", - Name: "Mailchimp", - Description: "Manage Mailchimp audiences and campaigns", - Icon: "mailchimp", - Category: "marketing", - Fields: []IntegrationField{ - { - Key: "api_key", - Label: "API Key", - Type: "api_key", - Required: true, - Placeholder: "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx-usX", - HelpText: "Create an API key in Mailchimp: Account → Extras → API keys", - Sensitive: true, - }, - }, - Tools: []string{"mailchimp_lists", "mailchimp_add_subscriber"}, - DocsURL: "https://mailchimp.com/developer/marketing/api/", - }, - - // ============================================ - // ANALYTICS - // ============================================ - "mixpanel": { - ID: "mixpanel", - Name: "Mixpanel", - Description: "Track events and analyze user behavior with Mixpanel", - Icon: "mixpanel", - Category: "analytics", - Fields: []IntegrationField{ - { - Key: "project_token", - Label: "Project Token", - Type: "api_key", - Required: true, - Placeholder: "Your Mixpanel Project Token", - HelpText: "Find in Mixpanel: Settings → Project Settings → Project Token", - Sensitive: true, - }, - { - Key: "api_secret", - Label: "API Secret", - Type: "api_key", - Required: false, - Placeholder: "Your Mixpanel API Secret", - HelpText: "Required for data export. Find in Project Settings → API Secret", - Sensitive: true, - }, - }, - Tools: []string{"mixpanel_track", "mixpanel_user_profile"}, - DocsURL: "https://developer.mixpanel.com/reference/overview", - }, - - "posthog": { - ID: "posthog", - Name: "PostHog", - Description: "Track events and analyze product usage with PostHog", - Icon: "posthog", - Category: "analytics", - Fields: []IntegrationField{ - { - Key: "api_key", - Label: "Project API Key", - Type: "api_key", - Required: true, - Placeholder: "phc_...", - HelpText: "Find in PostHog: Settings → Project → Project API Key", - Sensitive: true, - }, - { - Key: "host", - Label: "PostHog Host", - Type: "text", - Required: false, - Placeholder: "https://app.posthog.com", - HelpText: "Leave empty for PostHog Cloud, or enter your self-hosted URL", - Default: "https://app.posthog.com", - Sensitive: false, - }, - { - Key: "personal_api_key", - Label: "Personal API Key", - Type: "api_key", - Required: false, - Placeholder: "phx_...", - HelpText: "Required for querying data. Create at Settings → Personal API Keys", - Sensitive: true, - }, - }, - Tools: []string{"posthog_capture", "posthog_identify", "posthog_query"}, - DocsURL: "https://posthog.com/docs/api", - }, - - // ============================================ - // E-COMMERCE - // ============================================ - "shopify": { - ID: "shopify", - Name: "Shopify", - Description: "Manage Shopify products, orders, and customers", - Icon: "shopify", - Category: "ecommerce", - Fields: []IntegrationField{ - { - Key: "store_url", - Label: "Store URL", - Type: "text", - Required: true, - Placeholder: "your-store.myshopify.com", - HelpText: "Your Shopify store URL (without https://)", - Sensitive: false, - }, - { - Key: "access_token", - Label: "Admin API Access Token", - Type: "api_key", - Required: true, - Placeholder: "shpat_...", - HelpText: "Create in Shopify Admin: Settings → Apps → Develop apps → Create an app", - Sensitive: true, - }, - }, - Tools: []string{"shopify_products", "shopify_orders", "shopify_customers"}, - DocsURL: "https://shopify.dev/docs/api/admin-rest", - }, - - // ============================================ - // DEPLOYMENT - // ============================================ - "netlify": { - ID: "netlify", - Name: "Netlify", - Description: "Manage Netlify sites, deploys, and DNS settings", - Icon: "netlify", - Category: "deployment", - Fields: []IntegrationField{ - { - Key: "access_token", - Label: "Personal Access Token", - Type: "api_key", - Required: true, - Placeholder: "Your Netlify Personal Access Token", - HelpText: "Create at app.netlify.com/user/applications#personal-access-tokens", - Sensitive: true, - }, - }, - Tools: []string{"netlify_sites", "netlify_deploys", "netlify_trigger_build"}, - DocsURL: "https://docs.netlify.com/api/get-started/", - }, - - // ============================================ - // STORAGE - // ============================================ - "aws_s3": { - ID: "aws_s3", - Name: "AWS S3", - Description: "Access AWS S3 buckets for file storage", - Icon: "aws", - Category: "storage", - Fields: []IntegrationField{ - { - Key: "access_key_id", - Label: "Access Key ID", - Type: "api_key", - Required: true, - Placeholder: "AKIA...", - HelpText: "Your AWS Access Key ID", - Sensitive: true, - }, - { - Key: "secret_access_key", - Label: "Secret Access Key", - Type: "api_key", - Required: true, - Placeholder: "Your AWS Secret Access Key", - HelpText: "Your AWS Secret Access Key", - Sensitive: true, - }, - { - Key: "region", - Label: "Region", - Type: "text", - Required: true, - Placeholder: "us-east-1", - HelpText: "AWS region for your S3 bucket", - Default: "us-east-1", - Sensitive: false, - }, - { - Key: "bucket", - Label: "Default Bucket", - Type: "text", - Required: false, - Placeholder: "my-bucket", - HelpText: "Optional: Default S3 bucket name", - Sensitive: false, - }, - }, - Tools: []string{"s3_list", "s3_upload", "s3_download", "s3_delete"}, - DocsURL: "https://docs.aws.amazon.com/s3/", - }, - - // ============================================ - // SOCIAL MEDIA - // ============================================ - "x_twitter": { - ID: "x_twitter", - Name: "X (Twitter)", - Description: "Access X (Twitter) API v2 to post tweets, search posts, manage users, and interact with the platform programmatically.", - Icon: "twitter", - Category: "social", - Fields: []IntegrationField{ - { - Key: "bearer_token", - Label: "Bearer Token", - Type: "api_key", - Required: true, - Placeholder: "AAAAAAAAAAAAAAAAAAAAAA...", - HelpText: "Get your Bearer Token from developer.x.com portal", - Sensitive: true, - }, - { - Key: "api_key", - Label: "API Key (Consumer Key)", - Type: "api_key", - Required: false, - Placeholder: "Your API Key", - HelpText: "Required for posting tweets (OAuth 1.0a)", - Sensitive: true, - }, - { - Key: "api_secret", - Label: "API Secret (Consumer Secret)", - Type: "api_key", - Required: false, - Placeholder: "Your API Secret", - HelpText: "Required for posting tweets (OAuth 1.0a)", - Sensitive: true, - }, - { - Key: "access_token", - Label: "Access Token", - Type: "api_key", - Required: false, - Placeholder: "Your Access Token", - HelpText: "Required for posting tweets on behalf of a user", - Sensitive: true, - }, - { - Key: "access_token_secret", - Label: "Access Token Secret", - Type: "api_key", - Required: false, - Placeholder: "Your Access Token Secret", - HelpText: "Required for posting tweets on behalf of a user", - Sensitive: true, - }, - }, - Tools: []string{"x_search_posts", "x_post_tweet", "x_get_user", "x_get_user_posts"}, - DocsURL: "https://docs.x.com/x-api/getting-started/about-x-api", - }, - - // ============================================ - // CUSTOM - // ============================================ - "custom_webhook": { - ID: "custom_webhook", - Name: "Custom Webhook", - Description: "Send data to any HTTP endpoint", - Icon: "webhook", - Category: "custom", - Fields: []IntegrationField{ - { - Key: "url", - Label: "Webhook URL", - Type: "webhook_url", - Required: true, - Placeholder: "https://your-endpoint.com/webhook", - HelpText: "The URL to send webhook requests to", - Sensitive: true, - }, - { - Key: "method", - Label: "HTTP Method", - Type: "select", - Required: true, - Options: []string{"POST", "PUT", "PATCH"}, - Default: "POST", - HelpText: "HTTP method for the webhook request", - Sensitive: false, - }, - { - Key: "auth_type", - Label: "Authentication Type", - Type: "select", - Required: false, - Options: []string{"none", "bearer", "basic", "api_key"}, - Default: "none", - HelpText: "Type of authentication to use", - Sensitive: false, - }, - { - Key: "auth_value", - Label: "Auth Token/Key", - Type: "api_key", - Required: false, - Placeholder: "Your authentication token or key", - HelpText: "The authentication value (token, API key, or user:pass for basic)", - Sensitive: true, - }, - { - Key: "headers", - Label: "Custom Headers (JSON)", - Type: "json", - Required: false, - Placeholder: `{"X-Custom-Header": "value"}`, - HelpText: "Additional headers as JSON object", - Sensitive: false, - }, - }, - Tools: []string{"send_webhook"}, - DocsURL: "", - }, - - "rest_api": { - ID: "rest_api", - Name: "REST API", - Description: "Connect to any REST API endpoint", - Icon: "api", - Category: "custom", - Fields: []IntegrationField{ - { - Key: "base_url", - Label: "Base URL", - Type: "text", - Required: true, - Placeholder: "https://api.example.com/v1", - HelpText: "Base URL for the API (endpoints will be appended)", - Sensitive: false, - }, - { - Key: "auth_type", - Label: "Authentication Type", - Type: "select", - Required: false, - Options: []string{"none", "bearer", "basic", "api_key_header", "api_key_query"}, - Default: "none", - HelpText: "Type of authentication to use", - Sensitive: false, - }, - { - Key: "auth_value", - Label: "Auth Token/Key", - Type: "api_key", - Required: false, - Placeholder: "Your authentication token or key", - HelpText: "The authentication value", - Sensitive: true, - }, - { - Key: "auth_header_name", - Label: "API Key Header Name", - Type: "text", - Required: false, - Placeholder: "X-API-Key", - HelpText: "Header name for API key authentication", - Default: "X-API-Key", - Sensitive: false, - }, - { - Key: "headers", - Label: "Default Headers (JSON)", - Type: "json", - Required: false, - Placeholder: `{"Accept": "application/json"}`, - HelpText: "Default headers to include in all requests", - Sensitive: false, - }, - }, - Tools: []string{"api_request"}, - DocsURL: "", - }, - - // ============================================ - // DATABASE - // ============================================ - "mongodb": { - ID: "mongodb", - Name: "MongoDB", - Description: "Query and write to MongoDB databases. Supports find, insert, update operations (no delete for safety).", - Icon: "database", - Category: "database", - Fields: []IntegrationField{ - { - Key: "connection_string", - Label: "Connection String", - Type: "api_key", - Required: true, - Placeholder: "mongodb+srv://user:password@cluster.mongodb.net", - HelpText: "MongoDB connection URI (SRV or standard format)", - Sensitive: true, - }, - { - Key: "database", - Label: "Database Name", - Type: "text", - Required: true, - Placeholder: "myDatabase", - HelpText: "The database to connect to", - Sensitive: false, - }, - }, - Tools: []string{"mongodb_query", "mongodb_write"}, - DocsURL: "https://www.mongodb.com/docs/drivers/go/current/", - }, - - "redis": { - ID: "redis", - Name: "Redis", - Description: "Read and write to Redis key-value store. Supports strings, hashes, lists, sets, and sorted sets (no delete for safety).", - Icon: "database", - Category: "database", - Fields: []IntegrationField{ - { - Key: "host", - Label: "Host", - Type: "text", - Required: true, - Placeholder: "localhost", - HelpText: "Redis server hostname or IP", - Default: "localhost", - Sensitive: false, - }, - { - Key: "port", - Label: "Port", - Type: "text", - Required: false, - Placeholder: "6379", - HelpText: "Redis server port (default: 6379)", - Default: "6379", - Sensitive: false, - }, - { - Key: "password", - Label: "Password", - Type: "api_key", - Required: false, - Placeholder: "Your Redis password", - HelpText: "Redis authentication password (leave empty if not required)", - Sensitive: true, - }, - { - Key: "database", - Label: "Database Number", - Type: "text", - Required: false, - Placeholder: "0", - HelpText: "Redis database number (default: 0)", - Default: "0", - Sensitive: false, - }, - }, - Tools: []string{"redis_read", "redis_write"}, - DocsURL: "https://redis.io/docs/", - }, -} - -// IntegrationCategories defines the categories and their order -var IntegrationCategories = []IntegrationCategory{ - { - ID: "communication", - Name: "Communication", - Icon: "message-square", - }, - { - ID: "productivity", - Name: "Productivity", - Icon: "layout-grid", - }, - { - ID: "development", - Name: "Development", - Icon: "code", - }, - { - ID: "crm", - Name: "CRM / Sales", - Icon: "users", - }, - { - ID: "marketing", - Name: "Marketing / Email", - Icon: "mail", - }, - { - ID: "analytics", - Name: "Analytics", - Icon: "bar-chart-2", - }, - { - ID: "ecommerce", - Name: "E-Commerce", - Icon: "shopping-cart", - }, - { - ID: "deployment", - Name: "Deployment", - Icon: "rocket", - }, - { - ID: "storage", - Name: "Storage", - Icon: "hard-drive", - }, - { - ID: "database", - Name: "Database", - Icon: "database", - }, - { - ID: "social", - Name: "Social Media", - Icon: "share-2", - }, - { - ID: "custom", - Name: "Custom", - Icon: "settings", - }, -} - -// GetIntegration returns an integration by ID -func GetIntegration(id string) (Integration, bool) { - integration, exists := IntegrationRegistry[id] - return integration, exists -} - -// GetIntegrationsByCategory returns all integrations grouped by category -func GetIntegrationsByCategory() []IntegrationCategory { - result := make([]IntegrationCategory, len(IntegrationCategories)) - - for i, category := range IntegrationCategories { - result[i] = IntegrationCategory{ - ID: category.ID, - Name: category.Name, - Icon: category.Icon, - Integrations: []Integration{}, - } - - for _, integration := range IntegrationRegistry { - if integration.Category == category.ID { - result[i].Integrations = append(result[i].Integrations, integration) - } - } - } - - return result -} - -// ValidateCredentialData validates that the provided data matches the integration schema -func ValidateCredentialData(integrationType string, data map[string]interface{}) error { - integration, exists := IntegrationRegistry[integrationType] - if !exists { - return &CredentialValidationError{Field: "integrationType", Message: "unknown integration type"} - } - - for _, field := range integration.Fields { - value, hasValue := data[field.Key] - if field.Required && (!hasValue || value == nil || value == "") { - return &CredentialValidationError{Field: field.Key, Message: "required field is missing"} - } - } - - return nil -} - -// CredentialValidationError represents a credential validation error -type CredentialValidationError struct { - Field string `json:"field"` - Message string `json:"message"` -} - -func (e *CredentialValidationError) Error() string { - return e.Field + ": " + e.Message -} - -// MaskCredentialValue masks a sensitive value for display -// e.g., "sk-1234567890abcdef" -> "sk-...cdef" -func MaskCredentialValue(value string, fieldType string) string { - if value == "" { - return "" - } - - switch fieldType { - case "webhook_url": - // For URLs, show domain and mask the rest - if len(value) > 30 { - return value[:20] + "..." + value[len(value)-8:] - } - return value - - case "api_key", "token": - // For API keys, show prefix and last few chars - if len(value) > 12 { - return value[:6] + "..." + value[len(value)-4:] - } - if len(value) > 6 { - return value[:3] + "..." + value[len(value)-2:] - } - return "***" - - case "json": - return "[JSON data]" - - default: - // For other sensitive data, basic masking - if len(value) > 8 { - return value[:4] + "..." + value[len(value)-4:] - } - return "***" - } -} - -// GenerateMaskedPreview generates a masked preview for a credential -func GenerateMaskedPreview(integrationType string, data map[string]interface{}) string { - integration, exists := IntegrationRegistry[integrationType] - if !exists { - return "" - } - - // Find the primary field (first required sensitive field) - for _, field := range integration.Fields { - if field.Required && field.Sensitive { - if value, ok := data[field.Key].(string); ok { - return MaskCredentialValue(value, field.Type) - } - } - } - - // Fallback: mask first field - for _, field := range integration.Fields { - if value, ok := data[field.Key].(string); ok && value != "" { - return MaskCredentialValue(value, field.Type) - } - } - - return "" -} diff --git a/backend/internal/models/mcp.go b/backend/internal/models/mcp.go deleted file mode 100644 index 61cf7e7c..00000000 --- a/backend/internal/models/mcp.go +++ /dev/null @@ -1,69 +0,0 @@ -package models - -import "time" - -// MCPConnection represents an active MCP client connection -type MCPConnection struct { - ID string `json:"id"` - UserID string `json:"user_id"` - ClientID string `json:"client_id"` - ClientVersion string `json:"client_version"` - Platform string `json:"platform"` - ConnectedAt time.Time `json:"connected_at"` - LastHeartbeat time.Time `json:"last_heartbeat"` - IsActive bool `json:"is_active"` - Tools []MCPTool `json:"tools"` - WriteChan chan MCPServerMessage `json:"-"` - StopChan chan bool `json:"-"` - PendingResults map[string]chan MCPToolResult `json:"-"` // call_id -> result channel -} - -// MCPTool represents a tool registered by an MCP client -type MCPTool struct { - Name string `json:"name"` - Description string `json:"description"` - Parameters map[string]interface{} `json:"parameters"` // JSON Schema - Source string `json:"source"` // "mcp_local" - UserID string `json:"user_id"` -} - -// MCPClientMessage represents messages from MCP client to backend -type MCPClientMessage struct { - Type string `json:"type"` // "register_tools", "tool_result", "heartbeat", "disconnect" - Payload map[string]interface{} `json:"payload"` -} - -// MCPServerMessage represents messages from backend to MCP client -type MCPServerMessage struct { - Type string `json:"type"` // "tool_call", "ack", "error" - Payload map[string]interface{} `json:"payload"` -} - -// MCPToolRegistration represents the registration payload from client -type MCPToolRegistration struct { - ClientID string `json:"client_id"` - ClientVersion string `json:"client_version"` - Platform string `json:"platform"` - Tools []MCPTool `json:"tools"` -} - -// MCPToolCall represents a tool execution request to client -type MCPToolCall struct { - CallID string `json:"call_id"` - ToolName string `json:"tool_name"` - Arguments map[string]interface{} `json:"arguments"` - Timeout int `json:"timeout"` // seconds -} - -// MCPToolResult represents a tool execution result from client -type MCPToolResult struct { - CallID string `json:"call_id"` - Success bool `json:"success"` - Result string `json:"result"` - Error string `json:"error,omitempty"` -} - -// MCPHeartbeat represents a heartbeat message -type MCPHeartbeat struct { - Timestamp time.Time `json:"timestamp"` -} diff --git a/backend/internal/models/memory.go b/backend/internal/models/memory.go deleted file mode 100644 index b5cb2a14..00000000 --- a/backend/internal/models/memory.go +++ /dev/null @@ -1,114 +0,0 @@ -package models - -import ( - "time" - - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// Memory represents a single memory extracted from conversations -type Memory struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - UserID string `bson:"userId" json:"user_id"` - ConversationID string `bson:"conversationId,omitempty" json:"conversation_id,omitempty"` // Source conversation (optional) - - // Memory Content (encrypted) - EncryptedContent string `bson:"encryptedContent" json:"-"` // AES-256-GCM encrypted memory text - ContentHash string `bson:"contentHash" json:"content_hash"` // SHA-256 hash for deduplication - - // Metadata (plaintext for querying) - Category string `bson:"category" json:"category"` // "personal_info", "preferences", "context", "fact", "instruction" - Tags []string `bson:"tags,omitempty" json:"tags,omitempty"` // Searchable tags (e.g., "coding", "music", "work") - - // PageRank-like Scoring - Score float64 `bson:"score" json:"score"` // Current relevance score (0.0-1.0) - AccessCount int64 `bson:"accessCount" json:"access_count"` // How many times memory was selected/used - LastAccessedAt *time.Time `bson:"lastAccessedAt,omitempty" json:"last_accessed_at,omitempty"` - - // Decay & Archival - IsArchived bool `bson:"isArchived" json:"is_archived"` // Decayed below threshold - ArchivedAt *time.Time `bson:"archivedAt,omitempty" json:"archived_at,omitempty"` - - // Engagement Metrics (for PageRank calculation) - SourceEngagement float64 `bson:"sourceEngagement" json:"source_engagement"` // Engagement score of conversation it came from - - // Timestamps - CreatedAt time.Time `bson:"createdAt" json:"created_at"` - UpdatedAt time.Time `bson:"updatedAt" json:"updated_at"` - - // Version (for deduplication/updates) - Version int64 `bson:"version" json:"version"` // Incremented on updates -} - -// DecryptedMemory represents a memory with decrypted content (for internal use only) -type DecryptedMemory struct { - Memory - DecryptedContent string `json:"content"` // Decrypted content -} - -// MemoryExtractionJob represents a pending extraction job -type MemoryExtractionJob struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - UserID string `bson:"userId" json:"user_id"` - ConversationID string `bson:"conversationId" json:"conversation_id"` - MessageCount int `bson:"messageCount" json:"message_count"` // Number of messages to process - EncryptedMessages string `bson:"encryptedMessages" json:"-"` // Encrypted message batch - - Status string `bson:"status" json:"status"` // "pending", "processing", "completed", "failed" - AttemptCount int `bson:"attemptCount" json:"attempt_count"` // For retry logic - ErrorMessage string `bson:"errorMessage,omitempty" json:"error_message,omitempty"` - - CreatedAt time.Time `bson:"createdAt" json:"created_at"` - ProcessedAt *time.Time `bson:"processedAt,omitempty" json:"processed_at,omitempty"` -} - -// ConversationEngagement tracks engagement for PageRank calculation -type ConversationEngagement struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - UserID string `bson:"userId" json:"user_id"` - ConversationID string `bson:"conversationId" json:"conversation_id"` - - MessageCount int `bson:"messageCount" json:"message_count"` // Total messages in conversation - UserMessageCount int `bson:"userMessageCount" json:"user_message_count"` // User's message count - AvgResponseLength int `bson:"avgResponseLength" json:"avg_response_length"` // Average user response length - - EngagementScore float64 `bson:"engagementScore" json:"engagement_score"` // Calculated engagement (0.0-1.0) - - CreatedAt time.Time `bson:"createdAt" json:"created_at"` - UpdatedAt time.Time `bson:"updatedAt" json:"updated_at"` -} - -// MemoryCategory constants -const ( - MemoryCategoryPersonalInfo = "personal_info" - MemoryCategoryPreferences = "preferences" - MemoryCategoryContext = "context" - MemoryCategoryFact = "fact" - MemoryCategoryInstruction = "instruction" -) - -// MemoryExtractionJobStatus constants -const ( - JobStatusPending = "pending" - JobStatusProcessing = "processing" - JobStatusCompleted = "completed" - JobStatusFailed = "failed" -) - -// Memory archive threshold (memories with score below this are archived) -const MemoryArchiveThreshold = 0.15 - -// ExtractedMemoryFromLLM represents the structured output from the extraction LLM -type ExtractedMemoryFromLLM struct { - Memories []struct { - Content string `json:"content"` - Category string `json:"category"` - Tags []string `json:"tags"` - } `json:"memories"` -} - -// SelectedMemoriesFromLLM represents the structured output from the selection LLM -type SelectedMemoriesFromLLM struct { - SelectedMemoryIDs []string `json:"selected_memory_ids"` - Reasoning string `json:"reasoning"` -} diff --git a/backend/internal/models/model.go b/backend/internal/models/model.go deleted file mode 100644 index 3e098d85..00000000 --- a/backend/internal/models/model.go +++ /dev/null @@ -1,43 +0,0 @@ -package models - -import "time" - -// Model represents an LLM model from a provider -type Model struct { - ID string `json:"id"` - ProviderID int `json:"provider_id"` - ProviderName string `json:"provider_name,omitempty"` - ProviderFavicon string `json:"provider_favicon,omitempty"` - Name string `json:"name"` - DisplayName string `json:"display_name,omitempty"` - Description string `json:"description,omitempty"` - ContextLength int `json:"context_length,omitempty"` - SupportsTools bool `json:"supports_tools"` - SupportsStreaming bool `json:"supports_streaming"` - SupportsVision bool `json:"supports_vision"` - SmartToolRouter bool `json:"smart_tool_router"` // If true, model can be used as tool predictor - AgentsEnabled bool `json:"agents_enabled"` // If true, model is available in agent builder - IsVisible bool `json:"is_visible"` - SystemPrompt string `json:"system_prompt,omitempty"` - FetchedAt time.Time `json:"fetched_at"` -} - -// ModelFilter represents a filter rule for showing/hiding models -type ModelFilter struct { - ID int `json:"id"` - ProviderID int `json:"provider_id"` - ModelPattern string `json:"model_pattern"` - Action string `json:"action"` // "include" or "exclude" - Priority int `json:"priority"` -} - -// OpenAIModelsResponse represents the response from OpenAI-compatible /v1/models endpoint -type OpenAIModelsResponse struct { - Object string `json:"object"` - Data []struct { - ID string `json:"id"` - Object string `json:"object"` - Created int64 `json:"created"` - OwnedBy string `json:"owned_by"` - } `json:"data"` -} diff --git a/backend/internal/models/provider.go b/backend/internal/models/provider.go deleted file mode 100644 index 783e95b0..00000000 --- a/backend/internal/models/provider.go +++ /dev/null @@ -1,100 +0,0 @@ -package models - -import "time" - -// Provider represents an AI API provider (OpenAI, Anthropic, etc.) -type Provider struct { - ID int `json:"id"` - Name string `json:"name"` - BaseURL string `json:"base_url"` - APIKey string `json:"api_key,omitempty"` // Omit from responses for security - Enabled bool `json:"enabled"` - AudioOnly bool `json:"audio_only,omitempty"` // If true, provider is only used for audio transcription (not shown in model list) - ImageOnly bool `json:"image_only,omitempty"` // If true, provider is only used for image generation (not shown in model list) - ImageEditOnly bool `json:"image_edit_only,omitempty"` // If true, provider is only used for image editing (not shown in model list) - Secure bool `json:"secure,omitempty"` // If true, provider doesn't store user data - DefaultModel string `json:"default_model,omitempty"` // Default model for image generation - SystemPrompt string `json:"system_prompt,omitempty"` - Favicon string `json:"favicon,omitempty"` // Optional favicon URL for the provider - CreatedAt time.Time `json:"created_at"` - UpdatedAt time.Time `json:"updated_at"` -} - -// ModelAlias represents a model alias with display name and description -type ModelAlias struct { - ActualModel string `json:"actual_model"` // The actual model name to use with the provider API - DisplayName string `json:"display_name"` // Human-readable name shown in the UI - Description string `json:"description,omitempty"` // Optional description for the model - SupportsVision *bool `json:"supports_vision,omitempty"` // Optional override for vision/image support - Agents *bool `json:"agents,omitempty"` // If true, model is available for agent builder. If false/nil, hidden from agents - SmartToolRouter *bool `json:"smart_tool_router,omitempty"` // If true, model can be used as tool predictor for chat - FreeTier *bool `json:"free_tier,omitempty"` // If true, model is available for anonymous/free tier users - StructuredOutputSupport string `json:"structured_output_support,omitempty"` // Structured output quality: "excellent", "good", "poor", "unknown" - StructuredOutputCompliance *int `json:"structured_output_compliance,omitempty"` // Compliance percentage (0-100) - StructuredOutputWarning string `json:"structured_output_warning,omitempty"` // Warning message about structured output - StructuredOutputSpeedMs *int `json:"structured_output_speed_ms,omitempty"` // Average response time in milliseconds - StructuredOutputBadge string `json:"structured_output_badge,omitempty"` // Badge label (e.g., "FASTEST") - MemoryExtractor *bool `json:"memory_extractor,omitempty"` // If true, model can extract memories from conversations - MemorySelector *bool `json:"memory_selector,omitempty"` // If true, model can select relevant memories for context -} - -// ModelAliasView represents a model alias from the database (includes DB metadata) -type ModelAliasView struct { - ID int `json:"id"` - AliasName string `json:"alias_name"` - ModelID string `json:"model_id"` - ProviderID int `json:"provider_id"` - DisplayName string `json:"display_name"` - Description *string `json:"description,omitempty"` - SupportsVision *bool `json:"supports_vision,omitempty"` - AgentsEnabled *bool `json:"agents_enabled,omitempty"` - SmartToolRouter *bool `json:"smart_tool_router,omitempty"` - FreeTier *bool `json:"free_tier,omitempty"` - StructuredOutputSupport *string `json:"structured_output_support,omitempty"` - StructuredOutputCompliance *int `json:"structured_output_compliance,omitempty"` - StructuredOutputWarning *string `json:"structured_output_warning,omitempty"` - StructuredOutputSpeedMs *int `json:"structured_output_speed_ms,omitempty"` - StructuredOutputBadge *string `json:"structured_output_badge,omitempty"` - MemoryExtractor *bool `json:"memory_extractor,omitempty"` - MemorySelector *bool `json:"memory_selector,omitempty"` - CreatedAt time.Time `json:"created_at"` - UpdatedAt time.Time `json:"updated_at"` -} - -// RecommendedModels represents recommended model tiers -type RecommendedModels struct { - Top string `json:"top"` // Best/most capable model - Medium string `json:"medium"` // Balanced model - Fastest string `json:"fastest"` // Fastest/cheapest model - New string `json:"new"` // Newly added model -} - -// ProvidersConfig represents the providers.json file structure -type ProvidersConfig struct { - Providers []ProviderConfig `json:"providers"` -} - -// ProviderConfig represents a provider configuration from JSON -type ProviderConfig struct { - Name string `json:"name"` - BaseURL string `json:"base_url"` - APIKey string `json:"api_key"` - Enabled bool `json:"enabled"` - Secure bool `json:"secure,omitempty"` // Indicates provider doesn't store user data - AudioOnly bool `json:"audio_only,omitempty"` // If true, provider is only used for audio transcription (not shown in model list) - ImageOnly bool `json:"image_only,omitempty"` // If true, provider is only used for image generation (not shown in model list) - ImageEditOnly bool `json:"image_edit_only,omitempty"` // If true, provider is only used for image editing (not shown in model list) - DefaultModel string `json:"default_model,omitempty"` // Default model for image generation - SystemPrompt string `json:"system_prompt,omitempty"` - Favicon string `json:"favicon,omitempty"` // Optional favicon URL - Filters []FilterConfig `json:"filters"` - ModelAliases map[string]ModelAlias `json:"model_aliases,omitempty"` // Maps frontend model names to actual model names with descriptions - RecommendedModels *RecommendedModels `json:"recommended_models,omitempty"` // Recommended model tiers -} - -// FilterConfig represents a filter configuration from JSON -type FilterConfig struct { - Pattern string `json:"pattern"` - Action string `json:"action"` // "include" or "exclude" - Priority int `json:"priority"` // Higher priority = applied first -} diff --git a/backend/internal/models/schedule.go b/backend/internal/models/schedule.go deleted file mode 100644 index 9467d8af..00000000 --- a/backend/internal/models/schedule.go +++ /dev/null @@ -1,171 +0,0 @@ -package models - -import ( - "time" - - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// Schedule represents a cron-based schedule for agent execution -type Schedule struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - AgentID string `bson:"agentId" json:"agentId"` - UserID string `bson:"userId" json:"userId"` - CronExpression string `bson:"cronExpression" json:"cronExpression"` - Timezone string `bson:"timezone" json:"timezone"` - Enabled bool `bson:"enabled" json:"enabled"` - InputTemplate map[string]any `bson:"inputTemplate,omitempty" json:"inputTemplate,omitempty"` - - // Tracking - NextRunAt *time.Time `bson:"nextRunAt,omitempty" json:"nextRunAt,omitempty"` - LastRunAt *time.Time `bson:"lastRunAt,omitempty" json:"lastRunAt,omitempty"` - - // Statistics - TotalRuns int64 `bson:"totalRuns" json:"totalRuns"` - SuccessfulRuns int64 `bson:"successfulRuns" json:"successfulRuns"` - FailedRuns int64 `bson:"failedRuns" json:"failedRuns"` - - // Timestamps - CreatedAt time.Time `bson:"createdAt" json:"createdAt"` - UpdatedAt time.Time `bson:"updatedAt" json:"updatedAt"` -} - -// CreateScheduleRequest represents a request to create a schedule -type CreateScheduleRequest struct { - CronExpression string `json:"cronExpression" validate:"required"` - Timezone string `json:"timezone" validate:"required"` - InputTemplate map[string]any `json:"inputTemplate,omitempty"` - Enabled *bool `json:"enabled,omitempty"` // Defaults to true -} - -// UpdateScheduleRequest represents a request to update a schedule -type UpdateScheduleRequest struct { - CronExpression *string `json:"cronExpression,omitempty"` - Timezone *string `json:"timezone,omitempty"` - InputTemplate map[string]any `json:"inputTemplate,omitempty"` - Enabled *bool `json:"enabled,omitempty"` -} - -// ScheduleResponse represents the API response for a schedule -type ScheduleResponse struct { - ID string `json:"id"` - AgentID string `json:"agentId"` - CronExpression string `json:"cronExpression"` - Timezone string `json:"timezone"` - Enabled bool `json:"enabled"` - InputTemplate map[string]any `json:"inputTemplate,omitempty"` - NextRunAt *time.Time `json:"nextRunAt,omitempty"` - LastRunAt *time.Time `json:"lastRunAt,omitempty"` - TotalRuns int64 `json:"totalRuns"` - SuccessfulRuns int64 `json:"successfulRuns"` - FailedRuns int64 `json:"failedRuns"` - CreatedAt time.Time `json:"createdAt"` - UpdatedAt time.Time `json:"updatedAt"` -} - -// ToResponse converts a Schedule to ScheduleResponse -func (s *Schedule) ToResponse() *ScheduleResponse { - return &ScheduleResponse{ - ID: s.ID.Hex(), - AgentID: s.AgentID, - CronExpression: s.CronExpression, - Timezone: s.Timezone, - Enabled: s.Enabled, - InputTemplate: s.InputTemplate, - NextRunAt: s.NextRunAt, - LastRunAt: s.LastRunAt, - TotalRuns: s.TotalRuns, - SuccessfulRuns: s.SuccessfulRuns, - FailedRuns: s.FailedRuns, - CreatedAt: s.CreatedAt, - UpdatedAt: s.UpdatedAt, - } -} - -// TierLimits defines rate limits and quotas per subscription tier -type TierLimits struct { - MaxSchedules int `json:"maxSchedules"` - MaxAPIKeys int `json:"maxApiKeys"` - RequestsPerMinute int64 `json:"requestsPerMinute"` - RequestsPerHour int64 `json:"requestsPerHour"` - RetentionDays int `json:"retentionDays"` - MaxExecutionsPerDay int64 `json:"maxExecutionsPerDay"` - - // Usage limits - MaxMessagesPerMonth int64 `json:"maxMessagesPerMonth"` // Monthly message count limit - MaxFileUploadsPerDay int64 `json:"maxFileUploadsPerDay"` // Daily file upload limit - MaxImageGensPerDay int64 `json:"maxImageGensPerDay"` // Daily image generation limit - MaxMemoryExtractionsPerDay int64 `json:"maxMemoryExtractionsPerDay"` // Daily memory extraction limit -} - -// DefaultTierLimits provides tier configurations -var DefaultTierLimits = map[string]TierLimits{ - "free": { - MaxSchedules: 5, - MaxAPIKeys: 3, - RequestsPerMinute: 60, - RequestsPerHour: 1000, - RetentionDays: 30, - MaxExecutionsPerDay: 100, - MaxMessagesPerMonth: 300, - MaxFileUploadsPerDay: 10, - MaxImageGensPerDay: 10, - MaxMemoryExtractionsPerDay: 15, // ~15 extractions/day for free tier - }, - "pro": { - MaxSchedules: 50, - MaxAPIKeys: 50, - RequestsPerMinute: 300, - RequestsPerHour: 5000, - RetentionDays: 30, - MaxExecutionsPerDay: 1000, - MaxMessagesPerMonth: 10000, - MaxFileUploadsPerDay: 50, - MaxImageGensPerDay: 50, - MaxMemoryExtractionsPerDay: 100, // ~100 extractions/day for pro - }, - "max": { - MaxSchedules: 100, - MaxAPIKeys: 100, - RequestsPerMinute: 500, - RequestsPerHour: 10000, - RetentionDays: 30, - MaxExecutionsPerDay: 2000, - MaxMessagesPerMonth: -1, // unlimited - MaxFileUploadsPerDay: -1, // unlimited - MaxImageGensPerDay: -1, // unlimited - MaxMemoryExtractionsPerDay: -1, // unlimited - }, - "enterprise": { - MaxSchedules: -1, // unlimited - MaxAPIKeys: -1, - RequestsPerMinute: 1000, - RequestsPerHour: -1, // unlimited - RetentionDays: 365, - MaxExecutionsPerDay: -1, // unlimited - MaxMessagesPerMonth: -1, // unlimited - MaxFileUploadsPerDay: -1, // unlimited - MaxImageGensPerDay: -1, // unlimited - MaxMemoryExtractionsPerDay: -1, // unlimited - }, - "legacy_unlimited": { - MaxSchedules: -1, // unlimited - MaxAPIKeys: -1, // unlimited - RequestsPerMinute: -1, // unlimited - RequestsPerHour: -1, // unlimited - RetentionDays: 365, - MaxExecutionsPerDay: -1, // unlimited - MaxMessagesPerMonth: -1, // unlimited - MaxFileUploadsPerDay: -1, // unlimited - MaxImageGensPerDay: -1, // unlimited - MaxMemoryExtractionsPerDay: -1, // unlimited - }, -} - -// GetTierLimits returns the limits for a given tier -func GetTierLimits(tier string) TierLimits { - if limits, ok := DefaultTierLimits[tier]; ok { - return limits - } - return DefaultTierLimits["free"] -} diff --git a/backend/internal/models/schedule_test.go b/backend/internal/models/schedule_test.go deleted file mode 100644 index 6e6f7f2d..00000000 --- a/backend/internal/models/schedule_test.go +++ /dev/null @@ -1,140 +0,0 @@ -package models - -import ( - "testing" - "time" - - "go.mongodb.org/mongo-driver/bson/primitive" -) - -func TestScheduleToResponse(t *testing.T) { - now := time.Now() - nextRun := now.Add(1 * time.Hour) - lastRun := now.Add(-1 * time.Hour) - - schedule := &Schedule{ - ID: primitive.NewObjectID(), - AgentID: "agent-123", - UserID: "user-456", - CronExpression: "0 9 * * *", - Timezone: "America/New_York", - Enabled: true, - InputTemplate: map[string]interface{}{"topic": "AI news"}, - NextRunAt: &nextRun, - LastRunAt: &lastRun, - TotalRuns: 10, - SuccessfulRuns: 8, - FailedRuns: 2, - CreatedAt: now, - UpdatedAt: now, - } - - resp := schedule.ToResponse() - - if resp.ID != schedule.ID.Hex() { - t.Errorf("Expected ID %s, got %s", schedule.ID.Hex(), resp.ID) - } - - if resp.AgentID != schedule.AgentID { - t.Errorf("Expected AgentID %s, got %s", schedule.AgentID, resp.AgentID) - } - - if resp.CronExpression != schedule.CronExpression { - t.Errorf("Expected CronExpression %s, got %s", schedule.CronExpression, resp.CronExpression) - } - - if resp.Timezone != schedule.Timezone { - t.Errorf("Expected Timezone %s, got %s", schedule.Timezone, resp.Timezone) - } - - if resp.Enabled != schedule.Enabled { - t.Errorf("Expected Enabled %v, got %v", schedule.Enabled, resp.Enabled) - } - - if resp.TotalRuns != schedule.TotalRuns { - t.Errorf("Expected TotalRuns %d, got %d", schedule.TotalRuns, resp.TotalRuns) - } -} - -func TestGetTierLimits(t *testing.T) { - tests := []struct { - tier string - maxSchedules int - maxAPIKeys int - }{ - {"free", 5, 3}, - {"pro", 50, 50}, - {"max", 100, 100}, - {"enterprise", -1, -1}, // unlimited - {"legacy_unlimited", -1, -1}, // unlimited - {"unknown", 5, 3}, // defaults to free - } - - for _, tt := range tests { - t.Run(tt.tier, func(t *testing.T) { - limits := GetTierLimits(tt.tier) - - if limits.MaxSchedules != tt.maxSchedules { - t.Errorf("Expected MaxSchedules %d for tier %s, got %d", tt.maxSchedules, tt.tier, limits.MaxSchedules) - } - - if limits.MaxAPIKeys != tt.maxAPIKeys { - t.Errorf("Expected MaxAPIKeys %d for tier %s, got %d", tt.maxAPIKeys, tt.tier, limits.MaxAPIKeys) - } - }) - } -} - -func TestDefaultTierLimits(t *testing.T) { - // Verify all expected tiers exist - expectedTiers := []string{"free", "pro", "max", "enterprise", "legacy_unlimited"} - - for _, tier := range expectedTiers { - if _, ok := DefaultTierLimits[tier]; !ok { - t.Errorf("Expected tier %s in DefaultTierLimits", tier) - } - } - - // Verify free tier has reasonable defaults - freeLimits := DefaultTierLimits["free"] - if freeLimits.MaxSchedules <= 0 { - t.Error("Free tier should have positive MaxSchedules limit") - } - if freeLimits.RequestsPerMinute <= 0 { - t.Error("Free tier should have positive RequestsPerMinute limit") - } - if freeLimits.RetentionDays <= 0 { - t.Error("Free tier should have positive RetentionDays") - } - - // Verify enterprise tier has unlimited (-1) for schedules and API keys - enterpriseLimits := DefaultTierLimits["enterprise"] - if enterpriseLimits.MaxSchedules != -1 { - t.Error("Enterprise tier should have unlimited MaxSchedules (-1)") - } - if enterpriseLimits.MaxAPIKeys != -1 { - t.Error("Enterprise tier should have unlimited MaxAPIKeys (-1)") - } -} - -func TestCreateScheduleRequest(t *testing.T) { - enabled := true - req := CreateScheduleRequest{ - CronExpression: "0 9 * * *", - Timezone: "UTC", - InputTemplate: map[string]interface{}{"key": "value"}, - Enabled: &enabled, - } - - if req.CronExpression != "0 9 * * *" { - t.Errorf("Expected CronExpression '0 9 * * *', got %s", req.CronExpression) - } - - if req.Timezone != "UTC" { - t.Errorf("Expected Timezone 'UTC', got %s", req.Timezone) - } - - if req.Enabled == nil || *req.Enabled != true { - t.Error("Expected Enabled to be true") - } -} diff --git a/backend/internal/models/subscription.go b/backend/internal/models/subscription.go deleted file mode 100644 index f9886f84..00000000 --- a/backend/internal/models/subscription.go +++ /dev/null @@ -1,195 +0,0 @@ -package models - -import ( - "time" - - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// Subscription status constants -const ( - SubStatusActive = "active" - SubStatusOnHold = "on_hold" // Payment failed, grace period - SubStatusPendingCancel = "pending_cancel" // Will cancel at period end - SubStatusCancelled = "cancelled" - SubStatusPaused = "paused" -) - -// Subscription tiers -const ( - TierFree = "free" - TierPro = "pro" - TierMax = "max" - TierEnterprise = "enterprise" - TierLegacyUnlimited = "legacy_unlimited" // For grandfathered users -) - -// Plan represents a subscription plan with pricing -type Plan struct { - ID string `json:"id"` - Name string `json:"name"` - Tier string `json:"tier"` - PriceMonthly int64 `json:"price_monthly"` // cents - DodoProductID string `json:"dodo_product_id"` - Features []string `json:"features"` - Limits TierLimits `json:"limits"` - ContactSales bool `json:"contact_sales"` // true for enterprise -} - -// Subscription tracks a user's subscription state -type Subscription struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - UserID string `bson:"userId" json:"user_id"` - DodoSubscriptionID string `bson:"dodoSubscriptionId,omitempty" json:"dodo_subscription_id,omitempty"` - DodoCustomerID string `bson:"dodoCustomerId,omitempty" json:"dodo_customer_id,omitempty"` - - // Current state - Tier string `bson:"tier" json:"tier"` - Status string `bson:"status" json:"status"` - - // Billing info - CurrentPeriodStart time.Time `bson:"currentPeriodStart,omitempty" json:"current_period_start,omitempty"` - CurrentPeriodEnd time.Time `bson:"currentPeriodEnd,omitempty" json:"current_period_end,omitempty"` - - // Scheduled changes (for downgrades/cancellations) - ScheduledTier string `bson:"scheduledTier,omitempty" json:"scheduled_tier,omitempty"` - ScheduledChangeAt *time.Time `bson:"scheduledChangeAt,omitempty" json:"scheduled_change_at,omitempty"` - CancelAtPeriodEnd bool `bson:"cancelAtPeriodEnd" json:"cancel_at_period_end"` - - // Timestamps - CreatedAt time.Time `bson:"createdAt" json:"created_at"` - UpdatedAt time.Time `bson:"updatedAt" json:"updated_at"` - CancelledAt *time.Time `bson:"cancelledAt,omitempty" json:"cancelled_at,omitempty"` -} - -// IsActive returns true if subscription is currently active (user has access) -func (s *Subscription) IsActive() bool { - switch s.Status { - case SubStatusActive, SubStatusOnHold, SubStatusPendingCancel: - return true - default: - return false - } -} - -// IsExpired returns true if subscription period has ended -func (s *Subscription) IsExpired() bool { - return !s.CurrentPeriodEnd.IsZero() && s.CurrentPeriodEnd.Before(time.Now()) -} - -// HasScheduledChange returns true if there's a scheduled tier change -func (s *Subscription) HasScheduledChange() bool { - return s.ScheduledTier != "" && s.ScheduledChangeAt != nil -} - -// SubscriptionEvent for audit logging -type SubscriptionEvent struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - UserID string `bson:"userId" json:"user_id"` - SubscriptionID string `bson:"subscriptionId" json:"subscription_id"` - EventType string `bson:"eventType" json:"event_type"` - FromTier string `bson:"fromTier,omitempty" json:"from_tier,omitempty"` - ToTier string `bson:"toTier,omitempty" json:"to_tier,omitempty"` - DodoEventID string `bson:"dodoEventId,omitempty" json:"dodo_event_id,omitempty"` - Metadata map[string]any `bson:"metadata,omitempty" json:"metadata,omitempty"` - CreatedAt time.Time `bson:"createdAt" json:"created_at"` -} - -// TierOrder defines the order of tiers for comparison -var TierOrder = map[string]int{ - TierFree: 0, - TierPro: 1, - TierMax: 2, - TierEnterprise: 3, - TierLegacyUnlimited: 4, // Highest tier -} - -// CompareTiers compares two tiers and returns: -// -1 if fromTier < toTier (upgrade) -// 0 if fromTier == toTier (same) -// 1 if fromTier > toTier (downgrade) -func CompareTiers(fromTier, toTier string) int { - fromOrder, fromOk := TierOrder[fromTier] - toOrder, toOk := TierOrder[toTier] - - if !fromOk || !toOk { - // Unknown tier, treat as same - return 0 - } - - if fromOrder < toOrder { - return -1 - } else if fromOrder > toOrder { - return 1 - } - return 0 -} - -// AvailablePlans returns all available subscription plans -var AvailablePlans = []Plan{ - { - ID: "free", - Name: "Free", - Tier: TierFree, - PriceMonthly: 0, - DodoProductID: "", - Features: []string{"Basic features", "Limited usage"}, - Limits: GetTierLimits(TierFree), - ContactSales: false, - }, - { - ID: "pro", - Name: "Pro", - Tier: TierPro, - PriceMonthly: 1499, // $14.99 in cents - configure DodoProductID when ready - DodoProductID: "pdt_0NVGlqj3fgVkEeAmygtuj", // Set when creating product in DodoPayments - Features: []string{"Advanced features", "Higher limits", "Priority support"}, - Limits: GetTierLimits(TierPro), - ContactSales: false, - }, - { - ID: "max", - Name: "Max", - Tier: TierMax, - PriceMonthly: 3999, // $39.99 in cents - configure DodoProductID when ready - DodoProductID: "pdt_0NVGm0KQk5F4a8NVoaQst", // Set when creating product in DodoPayments - Features: []string{"All Pro features", "Maximum limits", "Premium support"}, - Limits: GetTierLimits(TierMax), - ContactSales: false, - }, - { - ID: "enterprise", - Name: "Enterprise", - Tier: TierEnterprise, - PriceMonthly: 0, - DodoProductID: "", - Features: []string{"Custom features", "Unlimited usage", "Dedicated support"}, - Limits: GetTierLimits(TierEnterprise), - ContactSales: true, - }, -} - -// GetPlanByID returns a plan by its ID -func GetPlanByID(planID string) *Plan { - for i := range AvailablePlans { - if AvailablePlans[i].ID == planID { - return &AvailablePlans[i] - } - } - return nil -} - -// GetPlanByTier returns a plan by its tier -func GetPlanByTier(tier string) *Plan { - for i := range AvailablePlans { - if AvailablePlans[i].Tier == tier { - return &AvailablePlans[i] - } - } - return nil -} - -// GetAvailablePlans returns all available plans -func GetAvailablePlans() []Plan { - return AvailablePlans -} diff --git a/backend/internal/models/subscription_test.go b/backend/internal/models/subscription_test.go deleted file mode 100644 index 9bf29ed7..00000000 --- a/backend/internal/models/subscription_test.go +++ /dev/null @@ -1,168 +0,0 @@ -package models - -import ( - "testing" - "time" -) - -func TestPlanComparison(t *testing.T) { - tests := []struct { - name string - fromTier string - toTier string - expected int // -1 = upgrade, 0 = same, 1 = downgrade - }{ - {"free to pro is upgrade", TierFree, TierPro, -1}, - {"free to max is upgrade", TierFree, TierMax, -1}, - {"pro to max is upgrade", TierPro, TierMax, -1}, - {"max to pro is downgrade", TierMax, TierPro, 1}, - {"pro to free is downgrade", TierPro, TierFree, 1}, - {"max to free is downgrade", TierMax, TierFree, 1}, - {"same tier", TierPro, TierPro, 0}, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - result := CompareTiers(tt.fromTier, tt.toTier) - if result != tt.expected { - t.Errorf("CompareTiers(%s, %s) = %d, want %d", - tt.fromTier, tt.toTier, result, tt.expected) - } - }) - } -} - -func TestSubscriptionStatus_IsActive(t *testing.T) { - tests := []struct { - status string - expected bool - }{ - {SubStatusActive, true}, - {SubStatusOnHold, true}, // Still active during grace - {SubStatusPendingCancel, true}, // Active until period ends - {SubStatusCancelled, false}, - {SubStatusPaused, false}, - } - - for _, tt := range tests { - t.Run(tt.status, func(t *testing.T) { - sub := &Subscription{Status: tt.status} - if sub.IsActive() != tt.expected { - t.Errorf("IsActive() for status %s = %v, want %v", - tt.status, sub.IsActive(), tt.expected) - } - }) - } -} - -func TestSubscription_IsExpired(t *testing.T) { - now := time.Now() - - tests := []struct { - name string - periodEnd time.Time - expected bool - }{ - {"future date", now.Add(24 * time.Hour), false}, - {"past date", now.Add(-24 * time.Hour), true}, - {"just now", now, true}, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - sub := &Subscription{CurrentPeriodEnd: tt.periodEnd} - if sub.IsExpired() != tt.expected { - t.Errorf("IsExpired() = %v, want %v", sub.IsExpired(), tt.expected) - } - }) - } -} - -func TestSubscription_HasScheduledChange(t *testing.T) { - future := time.Now().Add(24 * time.Hour) - - tests := []struct { - name string - scheduledTier string - scheduledAt *time.Time - expected bool - }{ - {"no scheduled change", "", nil, false}, - {"has scheduled downgrade", TierPro, &future, true}, - {"empty tier with date", "", &future, false}, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - sub := &Subscription{ - ScheduledTier: tt.scheduledTier, - ScheduledChangeAt: tt.scheduledAt, - } - if sub.HasScheduledChange() != tt.expected { - t.Errorf("HasScheduledChange() = %v, want %v", - sub.HasScheduledChange(), tt.expected) - } - }) - } -} - -func TestPlan_GetByID(t *testing.T) { - plans := GetAvailablePlans() - - // Test finding each plan - for _, plan := range plans { - found := GetPlanByID(plan.ID) - if found == nil { - t.Errorf("GetPlanByID(%s) returned nil", plan.ID) - } - if found != nil && found.ID != plan.ID { - t.Errorf("GetPlanByID(%s) returned plan with ID %s", plan.ID, found.ID) - } - } - - // Test non-existent plan - if GetPlanByID("nonexistent") != nil { - t.Error("Expected nil for non-existent plan") - } -} - -func TestTierLimits_Max(t *testing.T) { - limits := GetTierLimits(TierMax) - - if limits.MaxSchedules != 100 { - t.Errorf("Expected MaxSchedules 100, got %d", limits.MaxSchedules) - } - if limits.MaxAPIKeys != 100 { - t.Errorf("Expected MaxAPIKeys 100, got %d", limits.MaxAPIKeys) - } -} - -func TestGetPlanByTier(t *testing.T) { - tests := []struct { - tier string - expected string - }{ - {TierFree, "free"}, - {TierPro, "pro"}, - {TierMax, "max"}, - {TierEnterprise, "enterprise"}, - {"invalid", ""}, - } - - for _, tt := range tests { - t.Run(tt.tier, func(t *testing.T) { - plan := GetPlanByTier(tt.tier) - if tt.expected == "" { - if plan != nil { - t.Errorf("Expected nil for invalid tier, got %v", plan) - } - } else { - if plan == nil { - t.Errorf("Expected plan for tier %s, got nil", tt.tier) - } else if plan.ID != tt.expected { - t.Errorf("Expected plan ID %s, got %s", tt.expected, plan.ID) - } - } - }) - } -} diff --git a/backend/internal/models/team.go b/backend/internal/models/team.go deleted file mode 100644 index 61b72a78..00000000 --- a/backend/internal/models/team.go +++ /dev/null @@ -1,169 +0,0 @@ -package models - -import ( - "time" - - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// Team represents a team for collaborative agent access -// NOTE: This is schema-only for now - team logic not yet implemented -type Team struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - Name string `bson:"name" json:"name"` - OwnerID string `bson:"ownerId" json:"ownerId"` // Supabase user ID - - // Team members - Members []TeamMember `bson:"members" json:"members"` - - // Team settings - Settings TeamSettings `bson:"settings" json:"settings"` - - CreatedAt time.Time `bson:"createdAt" json:"createdAt"` - UpdatedAt time.Time `bson:"updatedAt" json:"updatedAt"` -} - -// TeamMember represents a member of a team -type TeamMember struct { - UserID string `bson:"userId" json:"userId"` - Role string `bson:"role" json:"role"` // owner, admin, editor, viewer - AddedAt time.Time `bson:"addedAt" json:"addedAt"` - AddedBy string `bson:"addedBy" json:"addedBy"` -} - -// TeamSettings contains team configuration -type TeamSettings struct { - DefaultAgentVisibility string `bson:"defaultAgentVisibility" json:"defaultAgentVisibility"` // private, team -} - -// TeamRole constants -const ( - TeamRoleOwner = "owner" - TeamRoleAdmin = "admin" - TeamRoleEditor = "editor" - TeamRoleViewer = "viewer" -) - -// IsOwner checks if a user is the team owner -func (t *Team) IsOwner(userID string) bool { - return t.OwnerID == userID -} - -// GetMemberRole returns the role of a member (empty if not a member) -func (t *Team) GetMemberRole(userID string) string { - if t.OwnerID == userID { - return TeamRoleOwner - } - for _, m := range t.Members { - if m.UserID == userID { - return m.Role - } - } - return "" -} - -// HasMember checks if a user is a team member -func (t *Team) HasMember(userID string) bool { - return t.GetMemberRole(userID) != "" -} - -// CanManageTeam checks if a user can manage team settings -func (t *Team) CanManageTeam(userID string) bool { - role := t.GetMemberRole(userID) - return role == TeamRoleOwner || role == TeamRoleAdmin -} - -// CanEditAgents checks if a user can edit team agents -func (t *Team) CanEditAgents(userID string) bool { - role := t.GetMemberRole(userID) - return role == TeamRoleOwner || role == TeamRoleAdmin || role == TeamRoleEditor -} - -// CanViewAgents checks if a user can view team agents -func (t *Team) CanViewAgents(userID string) bool { - return t.HasMember(userID) -} - -// AgentPermission represents granular agent access permissions -// NOTE: This is schema-only for now - permission logic not yet implemented -type AgentPermission struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - AgentID string `bson:"agentId" json:"agentId"` - - // Who has access (one of these should be set) - TeamID primitive.ObjectID `bson:"teamId,omitempty" json:"teamId,omitempty"` // If shared with team - UserID string `bson:"userId,omitempty" json:"userId,omitempty"` // If shared with individual - - // What access level - Permission string `bson:"permission" json:"permission"` // view, execute, edit, admin - - GrantedBy string `bson:"grantedBy" json:"grantedBy"` - GrantedAt time.Time `bson:"grantedAt" json:"grantedAt"` -} - -// Permission level constants -const ( - PermissionView = "view" // Can view agent and executions - PermissionExecute = "execute" // Can execute agent - PermissionEdit = "edit" // Can edit agent workflow - PermissionAdmin = "admin" // Full control including sharing -) - -// CanView checks if the permission allows viewing -func (p *AgentPermission) CanView() bool { - return p.Permission == PermissionView || - p.Permission == PermissionExecute || - p.Permission == PermissionEdit || - p.Permission == PermissionAdmin -} - -// CanExecute checks if the permission allows execution -func (p *AgentPermission) CanExecute() bool { - return p.Permission == PermissionExecute || - p.Permission == PermissionEdit || - p.Permission == PermissionAdmin -} - -// CanEdit checks if the permission allows editing -func (p *AgentPermission) CanEdit() bool { - return p.Permission == PermissionEdit || - p.Permission == PermissionAdmin -} - -// CanAdmin checks if the permission allows admin access -func (p *AgentPermission) CanAdmin() bool { - return p.Permission == PermissionAdmin -} - -// AgentVisibility constants for agent model extension -const ( - VisibilityPrivate = "private" // Only owner - VisibilityTeam = "team" // Team members - VisibilityPublic = "public" // Anyone (future) -) - -// CreateTeamRequest is the request body for creating a team -type CreateTeamRequest struct { - Name string `json:"name"` -} - -// InviteMemberRequest is the request body for inviting a team member -type InviteMemberRequest struct { - Email string `json:"email"` // User email to invite - Role string `json:"role"` // Role to assign -} - -// ShareAgentRequest is the request body for sharing an agent -type ShareAgentRequest struct { - TeamID string `json:"teamId,omitempty"` // Share with team - UserID string `json:"userId,omitempty"` // Share with user - Permission string `json:"permission"` // Permission level -} - -// TeamListItem is a lightweight team representation -type TeamListItem struct { - ID string `json:"id"` - Name string `json:"name"` - Role string `json:"role"` // Current user's role - MemberCount int `json:"memberCount"` -} diff --git a/backend/internal/models/user.go b/backend/internal/models/user.go deleted file mode 100644 index b3ab99a8..00000000 --- a/backend/internal/models/user.go +++ /dev/null @@ -1,149 +0,0 @@ -package models - -import ( - "time" - - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// User represents a user with local JWT authentication -type User struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - SupabaseUserID string `bson:"supabaseUserId,omitempty" json:"supabase_user_id,omitempty"` // Legacy field (optional) - Email string `bson:"email" json:"email"` - CreatedAt time.Time `bson:"createdAt" json:"created_at"` - LastLoginAt time.Time `bson:"lastLoginAt" json:"last_login_at"` - - // Local authentication fields (v2.0) - PasswordHash string `bson:"passwordHash,omitempty" json:"-"` // Argon2id hash (never expose in API) - EmailVerified bool `bson:"emailVerified" json:"email_verified"` - RefreshTokenVersion int `bson:"refreshTokenVersion" json:"-"` // For token revocation (never expose) - Role string `bson:"role,omitempty" json:"role,omitempty"` // user, admin - - // Subscription fields - SubscriptionTier string `bson:"subscriptionTier,omitempty" json:"subscription_tier,omitempty"` - SubscriptionStatus string `bson:"subscriptionStatus,omitempty" json:"subscription_status,omitempty"` - SubscriptionExpiresAt *time.Time `bson:"subscriptionExpiresAt,omitempty" json:"subscription_expires_at,omitempty"` - - // Migration audit trail - MigratedToLegacyAt *time.Time `bson:"migratedToLegacyAt,omitempty" json:"migrated_to_legacy_at,omitempty"` - - // Manual tier override (set by superadmin) - TierOverride *string `bson:"tierOverride,omitempty" json:"tier_override,omitempty"` // Override tier (if set) - - // Granular limit overrides (set by superadmin) - per-feature overrides - LimitOverrides *TierLimits `bson:"limitOverrides,omitempty" json:"limit_overrides,omitempty"` // Custom limits per user - - // Audit trail for overrides - OverrideSetBy string `bson:"overrideSetBy,omitempty" json:"override_set_by,omitempty"` // Admin user ID - OverrideSetAt *time.Time `bson:"overrideSetAt,omitempty" json:"override_set_at,omitempty"` // When override was set - OverrideReason string `bson:"overrideReason,omitempty" json:"override_reason,omitempty"` // Why override was set - - // DodoPayments integration - DodoCustomerID string `bson:"dodoCustomerId,omitempty" json:"dodo_customer_id,omitempty"` - DodoSubscriptionID string `bson:"dodoSubscriptionId,omitempty" json:"-"` // Don't expose in API - - // User preferences - Preferences UserPreferences `bson:"preferences" json:"preferences"` - - // Onboarding state - HasSeenWelcomePopup bool `bson:"hasSeenWelcomePopup" json:"has_seen_welcome_popup"` -} - -// ChatPrivacyMode represents how chats are stored -type ChatPrivacyMode string - -const ( - ChatPrivacyModeLocal ChatPrivacyMode = "local" - ChatPrivacyModeCloud ChatPrivacyMode = "cloud" -) - -// UserPreferences holds user-specific settings -type UserPreferences struct { - StoreBuilderChatHistory bool `bson:"storeBuilderChatHistory" json:"store_builder_chat_history"` - DefaultModelID string `bson:"defaultModelId,omitempty" json:"default_model_id,omitempty"` - ToolPredictorModelID string `bson:"toolPredictorModelId,omitempty" json:"tool_predictor_model_id,omitempty"` - ChatPrivacyMode ChatPrivacyMode `bson:"chatPrivacyMode,omitempty" json:"chat_privacy_mode,omitempty"` - Theme string `bson:"theme,omitempty" json:"theme,omitempty"` - FontSize string `bson:"fontSize,omitempty" json:"font_size,omitempty"` - - // Memory system preferences - MemoryEnabled bool `bson:"memoryEnabled" json:"memory_enabled"` // Default: false (opt-in) - MemoryExtractionThreshold int `bson:"memoryExtractionThreshold,omitempty" json:"memory_extraction_threshold,omitempty"` // Default: 2 messages (for quick testing, range: 2-50) - MemoryMaxInjection int `bson:"memoryMaxInjection,omitempty" json:"memory_max_injection,omitempty"` // Default: 5 memories - MemoryExtractorModelID string `bson:"memoryExtractorModelId,omitempty" json:"memory_extractor_model_id,omitempty"` - MemorySelectorModelID string `bson:"memorySelectorModelId,omitempty" json:"memory_selector_model_id,omitempty"` -} - -// UpdateUserPreferencesRequest is the request body for updating preferences -type UpdateUserPreferencesRequest struct { - StoreBuilderChatHistory *bool `json:"store_builder_chat_history,omitempty"` - DefaultModelID *string `json:"default_model_id,omitempty"` - ToolPredictorModelID *string `json:"tool_predictor_model_id,omitempty"` - ChatPrivacyMode *ChatPrivacyMode `json:"chat_privacy_mode,omitempty"` - Theme *string `json:"theme,omitempty"` - FontSize *string `json:"font_size,omitempty"` - - // Memory system preferences - MemoryEnabled *bool `json:"memory_enabled,omitempty"` - MemoryExtractionThreshold *int `json:"memory_extraction_threshold,omitempty"` - MemoryMaxInjection *int `json:"memory_max_injection,omitempty"` - MemoryExtractorModelID *string `json:"memory_extractor_model_id,omitempty"` - MemorySelectorModelID *string `json:"memory_selector_model_id,omitempty"` -} - -// UserResponse is the API response for user data -type UserResponse struct { - ID string `json:"id"` - Email string `json:"email"` - Role string `json:"role"` - CreatedAt time.Time `json:"created_at"` - LastLoginAt time.Time `json:"last_login_at"` - SubscriptionTier string `json:"subscription_tier,omitempty"` - SubscriptionStatus string `json:"subscription_status,omitempty"` - SubscriptionExpiresAt *time.Time `json:"subscription_expires_at,omitempty"` - Preferences UserPreferences `json:"preferences"` - HasSeenWelcomePopup bool `json:"has_seen_welcome_popup"` -} - -// ToResponse converts User to UserResponse -func (u *User) ToResponse() UserResponse { - return UserResponse{ - ID: u.ID.Hex(), - Email: u.Email, - Role: u.Role, - CreatedAt: u.CreatedAt, - LastLoginAt: u.LastLoginAt, - SubscriptionTier: u.SubscriptionTier, - SubscriptionStatus: u.SubscriptionStatus, - SubscriptionExpiresAt: u.SubscriptionExpiresAt, - Preferences: u.Preferences, - HasSeenWelcomePopup: u.HasSeenWelcomePopup, - } -} - -// SetLimitOverridesRequest for admin to set granular limit overrides -type SetLimitOverridesRequest struct { - // Option 1: Set entire tier (simple) - Tier *string `json:"tier,omitempty"` // Set to a tier name - - // Option 2: Set custom limits (granular) - Limits *TierLimits `json:"limits,omitempty"` // Custom limits - - // Metadata - Reason string `json:"reason"` // Why override is being set -} - -// AdminUserResponse includes override information -type AdminUserResponse struct { - UserResponse // Embed normal user response - EffectiveTier string `json:"effective_tier"` // Tier being used - EffectiveLimits TierLimits `json:"effective_limits"` // Actual limits after overrides - HasTierOverride bool `json:"has_tier_override"` // Whether tier is overridden - HasLimitOverrides bool `json:"has_limit_overrides"` // Whether limits are overridden - TierOverride *string `json:"tier_override,omitempty"` // - LimitOverrides *TierLimits `json:"limit_overrides,omitempty"` // - OverrideSetBy string `json:"override_set_by,omitempty"` // - OverrideSetAt *time.Time `json:"override_set_at,omitempty"` // - OverrideReason string `json:"override_reason,omitempty"` // -} diff --git a/backend/internal/models/websocket.go b/backend/internal/models/websocket.go deleted file mode 100644 index 3b6956e9..00000000 --- a/backend/internal/models/websocket.go +++ /dev/null @@ -1,183 +0,0 @@ -package models - -import ( - "sync" - "time" - - "github.com/gofiber/contrib/websocket" -) - -// ClientMessage represents a message from the client -type ClientMessage struct { - Type string `json:"type"` // "chat_message", "new_conversation", "stop_generation", "resume_stream", or "interactive_prompt_response" - ConversationID string `json:"conversation_id"` - Content string `json:"content,omitempty"` - History []map[string]interface{} `json:"history,omitempty"` // Optional: Client provides conversation history - ModelID string `json:"model_id,omitempty"` // Option 1: Select from platform models - CustomConfig *CustomAPIConfig `json:"custom_config,omitempty"` // Option 2: Bring your own API key - SystemInstructions string `json:"system_instructions,omitempty"` // Optional: Custom system prompt override - Attachments []MessageAttachment `json:"attachments,omitempty"` // File attachments (images, documents) - DisableTools bool `json:"disable_tools,omitempty"` // Disable tools for this message (e.g., agent builder) - - // Interactive prompt response fields - PromptID string `json:"prompt_id,omitempty"` // ID of the prompt being responded to - Answers map[string]InteractiveAnswer `json:"answers,omitempty"` // Map of question_id -> answer - Skipped bool `json:"skipped,omitempty"` // True if user skipped/cancelled -} - -// MessageAttachment represents a file attachment in a message -type MessageAttachment struct { - Type string `json:"type"` // "image", "document", "audio" - FileID string `json:"file_id"` // UUID from upload endpoint - URL string `json:"url"` // File URL (e.g., "/uploads/uuid.jpg") - MimeType string `json:"mime_type"` // MIME type (e.g., "image/jpeg") - Size int64 `json:"size"` // File size in bytes - Filename string `json:"filename"` // Original filename -} - -// CustomAPIConfig allows users to bring their own API keys (BYOK) -type CustomAPIConfig struct { - BaseURL string `json:"base_url,omitempty"` - APIKey string `json:"api_key,omitempty"` - Model string `json:"model,omitempty"` -} - -// InteractiveQuestion represents a question in an interactive prompt -type InteractiveQuestion struct { - ID string `json:"id"` // Unique question ID - Type string `json:"type"` // "text", "select", "multi-select", "number", "checkbox" - Label string `json:"label"` // Question text - Placeholder string `json:"placeholder,omitempty"` // Placeholder for text inputs - Required bool `json:"required,omitempty"` // Whether answer is required - Options []string `json:"options,omitempty"` // Options for select/multi-select - AllowOther bool `json:"allow_other,omitempty"` // Enable "Other" option - DefaultValue interface{} `json:"default_value,omitempty"` // Default value - Validation *QuestionValidation `json:"validation,omitempty"` // Validation rules -} - -// QuestionValidation represents validation rules for a question -type QuestionValidation struct { - Min *float64 `json:"min,omitempty"` // Minimum value for number type - Max *float64 `json:"max,omitempty"` // Maximum value for number type - Pattern string `json:"pattern,omitempty"` // Regex pattern for text type - MinLength *int `json:"min_length,omitempty"` // Minimum length for text - MaxLength *int `json:"max_length,omitempty"` // Maximum length for text -} - -// InteractiveAnswer represents a user's answer to a question -type InteractiveAnswer struct { - QuestionID string `json:"question_id"` // ID of the question - Value interface{} `json:"value"` // Answer value (string, number, bool, or []string) - IsOther bool `json:"is_other,omitempty"` // True if "Other" option selected -} - -// ServerMessage represents a message sent to the client -type ServerMessage struct { - Type string `json:"type"` // "stream_chunk", "reasoning_chunk", "tool_call", "tool_result", "stream_end", "stream_resume", "stream_missed", "conversation_reset", "conversation_title", "context_optimizing", "interactive_prompt", "prompt_timeout", "prompt_validation_error", "error" - Content string `json:"content,omitempty"` - Title string `json:"title,omitempty"` // Auto-generated conversation title OR interactive prompt title - ToolName string `json:"tool_name,omitempty"` - ToolDisplayName string `json:"tool_display_name,omitempty"` // User-friendly tool name (e.g., "Search Web") - ToolIcon string `json:"tool_icon,omitempty"` // Lucide React icon name (e.g., "Calculator", "Search", "Clock") - ToolDescription string `json:"tool_description,omitempty"` // Human-readable tool description - Status string `json:"status,omitempty"` // "executing", "completed", "started" - Arguments map[string]interface{} `json:"arguments,omitempty"` - Result string `json:"result,omitempty"` - Plots []PlotData `json:"plots,omitempty"` // Visualization plots from E2B tools - ConversationID string `json:"conversation_id,omitempty"` - Tokens *TokenUsage `json:"tokens,omitempty"` - ErrorCode string `json:"code,omitempty"` - ErrorMessage string `json:"message,omitempty"` - IsComplete bool `json:"is_complete,omitempty"` // For stream_resume: whether generation completed - Reason string `json:"reason,omitempty"` // For stream_missed: "expired" or "not_found" - Progress int `json:"progress,omitempty"` // For context_optimizing: progress percentage (0-100) - - // Interactive prompt fields - PromptID string `json:"prompt_id,omitempty"` // Unique prompt ID - Description string `json:"description,omitempty"` // Optional prompt description - Questions []InteractiveQuestion `json:"questions,omitempty"` // Array of questions - AllowSkip *bool `json:"allow_skip,omitempty"` // Whether user can skip (pointer to distinguish false from unset) - Errors map[string]string `json:"errors,omitempty"` // Validation errors (question_id -> error message) -} - -// PlotData represents a visualization plot (chart/graph) from E2B tools -type PlotData struct { - Format string `json:"format"` // "png", "jpg", "svg" - Data string `json:"data"` // Base64-encoded image data -} - -// TokenUsage represents token consumption statistics -type TokenUsage struct { - Input int `json:"input"` - Output int `json:"output"` -} - -// PromptWaiterFunc is a function type that waits for a prompt response -// Returns (answers, skipped, error) -type PromptWaiterFunc func(promptID string, timeout time.Duration) (map[string]InteractiveAnswer, bool, error) - -// UserConnection represents a single WebSocket connection -type UserConnection struct { - ConnID string - UserID string // User ID from authentication - Conn *websocket.Conn - ConversationID string - Messages []map[string]interface{} - MessageCount int // Track number of messages for title generation - ModelID string // Selected model ID from platform - CustomConfig *CustomAPIConfig // OR user's custom API configuration (BYOK) - SystemInstructions string // Optional: User-provided system prompt override - DisableTools bool // Disable tools for this connection (e.g., agent builder) - CreatedAt time.Time - WriteChan chan ServerMessage - StopChan chan bool - Mutex sync.Mutex - closed bool // Track if connection is closed - PromptWaiter PromptWaiterFunc // Function to wait for interactive prompt responses -} - -// SafeSend sends a message to WriteChan safely, returning false if the channel is closed -func (uc *UserConnection) SafeSend(msg ServerMessage) bool { - uc.Mutex.Lock() - if uc.closed { - uc.Mutex.Unlock() - return false - } - uc.Mutex.Unlock() - - // Use defer/recover to handle panic from send on closed channel - defer func() { - if r := recover(); r != nil { - // Channel was closed, mark connection as closed - uc.Mutex.Lock() - uc.closed = true - uc.Mutex.Unlock() - } - }() - - uc.WriteChan <- msg - return true -} - -// MarkClosed marks the connection as closed -func (uc *UserConnection) MarkClosed() { - uc.Mutex.Lock() - uc.closed = true - uc.Mutex.Unlock() -} - -// IsClosed returns true if the connection has been marked as closed -func (uc *UserConnection) IsClosed() bool { - uc.Mutex.Lock() - defer uc.Mutex.Unlock() - return uc.closed -} - -// ChatRequest represents a request to OpenAI-compatible chat completion API -type ChatRequest struct { - Model string `json:"model"` - Messages []map[string]interface{} `json:"messages"` - Tools []map[string]interface{} `json:"tools,omitempty"` - Stream bool `json:"stream"` - Temperature float64 `json:"temperature,omitempty"` -} diff --git a/backend/internal/models/workflow.go b/backend/internal/models/workflow.go deleted file mode 100644 index c896c70b..00000000 --- a/backend/internal/models/workflow.go +++ /dev/null @@ -1,139 +0,0 @@ -package models - -// ConversationMessage represents a message in the conversation history -type ConversationMessage struct { - Role string `json:"role"` // "user" or "assistant" - Content string `json:"content"` // Message content -} - -// WorkflowGenerateRequest represents a request to generate or modify a workflow -type WorkflowGenerateRequest struct { - AgentID string `json:"agent_id"` - UserMessage string `json:"user_message"` - CurrentWorkflow *Workflow `json:"current_workflow,omitempty"` // For modifications - ModelID string `json:"model_id,omitempty"` // Optional model override - ConversationID string `json:"conversation_id,omitempty"` // For conversation persistence - ConversationHistory []ConversationMessage `json:"conversation_history,omitempty"` // Recent conversation context for better tool selection -} - -// WorkflowGenerateResponse represents the structured output from workflow generation -type WorkflowGenerateResponse struct { - Success bool `json:"success"` - Workflow *Workflow `json:"workflow,omitempty"` - Explanation string `json:"explanation"` - Action string `json:"action"` // "create" or "modify" - Error string `json:"error,omitempty"` - Version int `json:"version"` - Errors []ValidationError `json:"errors,omitempty"` - SuggestedName string `json:"suggested_name,omitempty"` // AI-generated agent name suggestion - SuggestedDescription string `json:"suggested_description,omitempty"` // AI-generated agent description -} - -// ValidationError represents a workflow validation error -type ValidationError struct { - Type string `json:"type"` // "schema", "cycle", "type_mismatch", "missing_input" - Message string `json:"message"` - BlockID string `json:"blockId,omitempty"` - ConnectionID string `json:"connectionId,omitempty"` -} - -// WorkflowJSONSchema returns the JSON schema for structured output -func WorkflowJSONSchema() map[string]interface{} { - return map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "blocks": map[string]interface{}{ - "type": "array", - "items": map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "id": map[string]interface{}{ - "type": "string", - "description": "Unique block ID in kebab-case format matching the block name", - }, - "type": map[string]interface{}{ - "type": "string", - "enum": []string{"llm_inference", "variable", "code_block"}, - "description": "Block type - llm_inference for AI agents, variable for inputs, code_block for direct tool execution", - }, - "name": map[string]interface{}{ - "type": "string", - "description": "Human-readable block name", - }, - "description": map[string]interface{}{ - "type": "string", - "description": "What this block does", - }, - "config": map[string]interface{}{ - "type": "object", - "description": "Block-specific configuration", - }, - "position": map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "x": map[string]interface{}{"type": "integer"}, - "y": map[string]interface{}{"type": "integer"}, - }, - "required": []string{"x", "y"}, - }, - "timeout": map[string]interface{}{ - "type": "integer", - "description": "Timeout in seconds (default 30, max 120 for LLM blocks)", - }, - }, - "required": []string{"id", "type", "name", "description", "config", "position", "timeout"}, - }, - }, - "connections": map[string]interface{}{ - "type": "array", - "items": map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "id": map[string]interface{}{ - "type": "string", - "description": "Unique connection ID", - }, - "sourceBlockId": map[string]interface{}{ - "type": "string", - "description": "ID of the source block", - }, - "sourceOutput": map[string]interface{}{ - "type": "string", - "description": "Output port name (usually 'output')", - }, - "targetBlockId": map[string]interface{}{ - "type": "string", - "description": "ID of the target block", - }, - "targetInput": map[string]interface{}{ - "type": "string", - "description": "Input port name (usually 'input')", - }, - }, - "required": []string{"id", "sourceBlockId", "sourceOutput", "targetBlockId", "targetInput"}, - }, - }, - "variables": map[string]interface{}{ - "type": "array", - "items": map[string]interface{}{"type": "object"}, - }, - "explanation": map[string]interface{}{ - "type": "string", - "description": "Brief explanation of the workflow or changes made", - }, - }, - "required": []string{"blocks", "connections", "variables", "explanation"}, - } -} - -// WorkflowExecuteResult contains the result of a workflow execution -type WorkflowExecuteResult struct { - Status string - Output map[string]interface{} - BlockStates map[string]*BlockState - Error string -} - -// WorkflowExecuteFunc is a function type for executing workflows -// This allows scheduler to call workflow engine without import cycle -type WorkflowExecuteFunc func(workflow *Workflow, inputs map[string]interface{}) (*WorkflowExecuteResult, error) diff --git a/backend/internal/preflight/checks.go b/backend/internal/preflight/checks.go deleted file mode 100644 index f6934ce4..00000000 --- a/backend/internal/preflight/checks.go +++ /dev/null @@ -1,204 +0,0 @@ -package preflight - -import ( - "claraverse/internal/database" - "fmt" - "log" - "os" -) - -// CheckResult represents the result of a preflight check -type CheckResult struct { - Name string - Status string // "pass", "fail", "warning" - Message string - Error error -} - -// Checker performs pre-flight checks before server starts -type Checker struct { - db *database.DB - requiredEnvars []string -} - -// NewChecker creates a new preflight checker -func NewChecker(db *database.DB) *Checker { - return &Checker{ - db: db, - requiredEnvars: []string{ - // Optional: Add required environment variables here - }, - } -} - -// RunAll runs all preflight checks and returns results -func (c *Checker) RunAll() []CheckResult { - log.Println("🔍 Running pre-flight checks...") - - results := []CheckResult{ - c.checkDatabaseConnection(), - c.checkDatabaseSchema(), - c.checkEnvironmentVariables(), - } - - // Print summary - passed := 0 - failed := 0 - warnings := 0 - - for _, result := range results { - switch result.Status { - case "pass": - log.Printf(" ✅ %s: %s", result.Name, result.Message) - passed++ - case "fail": - log.Printf(" ❌ %s: %s", result.Name, result.Message) - if result.Error != nil { - log.Printf(" Error: %v", result.Error) - } - failed++ - case "warning": - log.Printf(" ⚠️ %s: %s", result.Name, result.Message) - warnings++ - } - } - - log.Printf("\n📊 Pre-flight summary: %d passed, %d failed, %d warnings\n", passed, failed, warnings) - - return results -} - -// HasFailures returns true if any check failed -func HasFailures(results []CheckResult) bool { - for _, result := range results { - if result.Status == "fail" { - return true - } - } - return false -} - -// checkDatabaseConnection verifies database connectivity -func (c *Checker) checkDatabaseConnection() CheckResult { - if err := c.db.Ping(); err != nil { - return CheckResult{ - Name: "Database Connection", - Status: "fail", - Message: "Cannot connect to database", - Error: err, - } - } - - return CheckResult{ - Name: "Database Connection", - Status: "pass", - Message: "Database connection successful", - } -} - -// checkDatabaseSchema verifies all required tables exist -func (c *Checker) checkDatabaseSchema() CheckResult { - requiredTables := []string{ - "providers", - "models", - "provider_model_filters", - "model_capabilities", - "model_refresh_log", - } - - for _, table := range requiredTables { - var count int - // MySQL-compatible query using INFORMATION_SCHEMA - query := "SELECT COUNT(*) FROM INFORMATION_SCHEMA.TABLES WHERE TABLE_SCHEMA = DATABASE() AND TABLE_NAME = ?" - err := c.db.QueryRow(query, table).Scan(&count) - if err != nil || count == 0 { - return CheckResult{ - Name: "Database Schema", - Status: "fail", - Message: fmt.Sprintf("Required table '%s' not found", table), - Error: err, - } - } - } - - return CheckResult{ - Name: "Database Schema", - Status: "pass", - Message: fmt.Sprintf("All %d required tables exist", len(requiredTables)), - } -} - - -// checkEnvironmentVariables verifies required environment variables are set -func (c *Checker) checkEnvironmentVariables() CheckResult { - missing := []string{} - - for _, envar := range c.requiredEnvars { - if os.Getenv(envar) == "" { - missing = append(missing, envar) - } - } - - if len(missing) > 0 { - return CheckResult{ - Name: "Environment Variables", - Status: "warning", - Message: fmt.Sprintf("Missing environment variables: %v", missing), - } - } - - // Check optional but recommended variables - supabaseURL := os.Getenv("SUPABASE_URL") - supabaseKey := os.Getenv("SUPABASE_KEY") - - if supabaseURL == "" || supabaseKey == "" { - return CheckResult{ - Name: "Environment Variables", - Status: "warning", - Message: "Supabase authentication not configured (running in development mode)", - } - } - - return CheckResult{ - Name: "Environment Variables", - Status: "pass", - Message: "All environment variables configured", - } -} - -// checkProviderConnectivity tests if we can reach provider APIs (optional, can be slow) -func (c *Checker) checkProviderConnectivity() CheckResult { - // This is an optional check that could be added - // It would test actual connectivity to provider APIs - // For now, we'll skip it to keep startup fast - - return CheckResult{ - Name: "Provider Connectivity", - Status: "pass", - Message: "Skipped (optional check)", - } -} - -// QuickCheck runs minimal checks for fast startup -func (c *Checker) QuickCheck() []CheckResult { - log.Println("⚡ Running quick pre-flight checks...") - - results := []CheckResult{ - c.checkDatabaseConnection(), - } - - passed := 0 - failed := 0 - - for _, result := range results { - if result.Status == "pass" { - log.Printf(" ✅ %s", result.Name) - passed++ - } else if result.Status == "fail" { - log.Printf(" ❌ %s: %s", result.Name, result.Message) - failed++ - } - } - - return results -} diff --git a/backend/internal/preflight/checks_test.go b/backend/internal/preflight/checks_test.go deleted file mode 100644 index 30a506bf..00000000 --- a/backend/internal/preflight/checks_test.go +++ /dev/null @@ -1,204 +0,0 @@ -package preflight - -import ( - "claraverse/internal/database" - "os" - "testing" -) - -func setupPreflightTest(t *testing.T) (*database.DB, func()) { - t.Skip("SQLite tests are deprecated - preflight tests require MySQL DSN via DATABASE_URL") - tmpDB := "test_preflight.db" - - db, err := database.New(tmpDB) - if err != nil { - t.Fatalf("Failed to create test database: %v", err) - } - - if err := db.Initialize(); err != nil { - t.Fatalf("Failed to initialize test database: %v", err) - } - - cleanup := func() { - db.Close() - os.Remove(tmpDB) - } - - return db, cleanup -} - -func TestNewChecker(t *testing.T) { - db, cleanup := setupPreflightTest(t) - defer cleanup() - - checker := NewChecker(db) - if checker == nil { - t.Fatal("Expected non-nil checker") - } - - if checker.db != db { - t.Error("Checker database not set correctly") - } -} - -func TestCheckDatabaseConnection_Success(t *testing.T) { - db, cleanup := setupPreflightTest(t) - defer cleanup() - - checker := NewChecker(db) - result := checker.checkDatabaseConnection() - - if result.Status != "pass" { - t.Errorf("Expected status 'pass', got '%s'", result.Status) - } - - if result.Name != "Database Connection" { - t.Errorf("Expected name 'Database Connection', got '%s'", result.Name) - } -} - -func TestCheckDatabaseConnection_Failure(t *testing.T) { - db, cleanup := setupPreflightTest(t) - cleanup() // Close database immediately to simulate failure - - checker := NewChecker(db) - result := checker.checkDatabaseConnection() - - if result.Status != "fail" { - t.Errorf("Expected status 'fail', got '%s'", result.Status) - } - - if result.Error == nil { - t.Error("Expected error to be set") - } -} - -func TestCheckDatabaseSchema_Success(t *testing.T) { - db, cleanup := setupPreflightTest(t) - defer cleanup() - - checker := NewChecker(db) - result := checker.checkDatabaseSchema() - - if result.Status != "pass" { - t.Errorf("Expected status 'pass', got '%s': %s", result.Status, result.Message) - } -} - -func TestCheckDatabaseSchema_MissingTable(t *testing.T) { - t.Skip("SQLite tests are deprecated - preflight tests require MySQL DSN via DATABASE_URL") -} - -func TestCheckProvidersFile_Exists(t *testing.T) { - t.Skip("Provider file checks have been removed from preflight - providers are now managed via database") -} - -func TestCheckProvidersFile_Missing(t *testing.T) { - t.Skip("Provider file checks have been removed from preflight - providers are now managed via database") -} - -func TestCheckProvidersJSON_Valid(t *testing.T) { - t.Skip("Provider file checks have been removed from preflight - providers are now managed via database") -} - -func TestCheckProvidersJSON_InvalidJSON(t *testing.T) { - t.Skip("Provider file checks have been removed from preflight - providers are now managed via database") -} - -func TestCheckProvidersJSON_EmptyProviders(t *testing.T) { - t.Skip("Provider file checks have been removed from preflight - providers are now managed via database") -} - -func TestCheckProvidersJSON_MissingName(t *testing.T) { - t.Skip("Provider file checks have been removed from preflight - providers are now managed via database") -} - -func TestCheckProvidersJSON_MissingBaseURL(t *testing.T) { - t.Skip("Provider file checks have been removed from preflight - providers are now managed via database") -} - -func TestCheckProvidersJSON_MissingAPIKey(t *testing.T) { - t.Skip("Provider file checks have been removed from preflight - providers are now managed via database") -} - -func TestCheckEnvironmentVariables(t *testing.T) { - db, cleanup := setupPreflightTest(t) - defer cleanup() - - checker := NewChecker(db) - result := checker.checkEnvironmentVariables() - - // Should pass or warn, but not fail - if result.Status == "fail" { - t.Errorf("Expected status 'pass' or 'warning', got 'fail': %s", result.Message) - } -} - -func TestRunAll(t *testing.T) { - db, cleanup := setupPreflightTest(t) - defer cleanup() - - checker := NewChecker(db) - results := checker.RunAll() - - if len(results) == 0 { - t.Error("Expected results, got empty slice") - } - - // Verify all expected checks ran - expectedChecks := map[string]bool{ - "Database Connection": false, - "Database Schema": false, - "Environment Variables": false, - } - - for _, result := range results { - if _, exists := expectedChecks[result.Name]; exists { - expectedChecks[result.Name] = true - } - } - - for checkName, ran := range expectedChecks { - if !ran { - t.Errorf("Expected check '%s' to run", checkName) - } - } -} - -func TestHasFailures(t *testing.T) { - // Test with no failures - results := []CheckResult{ - {Status: "pass"}, - {Status: "pass"}, - {Status: "warning"}, - } - - if HasFailures(results) { - t.Error("Expected no failures") - } - - // Test with failures - results = append(results, CheckResult{Status: "fail"}) - - if !HasFailures(results) { - t.Error("Expected failures to be detected") - } -} - -func TestQuickCheck(t *testing.T) { - db, cleanup := setupPreflightTest(t) - defer cleanup() - - checker := NewChecker(db) - results := checker.QuickCheck() - - if len(results) == 0 { - t.Error("Expected results from quick check") - } - - // Quick check should run fewer checks than full check - fullResults := checker.RunAll() - if len(results) >= len(fullResults) { - t.Error("Expected quick check to run fewer checks than full check") - } -} diff --git a/backend/internal/presentation/service.go b/backend/internal/presentation/service.go deleted file mode 100644 index b4344aae..00000000 --- a/backend/internal/presentation/service.go +++ /dev/null @@ -1,317 +0,0 @@ -package presentation - -import ( - "bytes" - "context" - "fmt" - "html" - "log" - "os" - "path/filepath" - "strings" - "sync" - "time" - - "github.com/chromedp/cdproto/page" - "github.com/chromedp/chromedp" - "github.com/google/uuid" -) - -// Slide represents a single slide/page in the presentation -// Each slide contains a complete standalone HTML document -type Slide struct { - HTML string `json:"html"` // Complete HTML document (must include , , and tags) -} - -// PresentationConfig holds the presentation configuration -type PresentationConfig struct { - Title string `json:"title"` - Slides []Slide `json:"slides"` // Array of slides, each with custom HTML content -} - -// GeneratedPresentation represents a generated presentation file -type GeneratedPresentation struct { - PresentationID string - UserID string - ConversationID string - Filename string - FilePath string - Size int64 - ContentType string - CreatedAt time.Time -} - -// Service handles presentation generation -type Service struct { - outputDir string - presentations map[string]*GeneratedPresentation - mu sync.RWMutex -} - -var ( - serviceInstance *Service - serviceOnce sync.Once -) - -// GetService returns the singleton presentation service -func GetService() *Service { - serviceOnce.Do(func() { - outputDir := "./generated" - if err := os.MkdirAll(outputDir, 0700); err != nil { - log.Printf("⚠️ Warning: Could not create generated directory: %v", err) - } - serviceInstance = &Service{ - outputDir: outputDir, - presentations: make(map[string]*GeneratedPresentation), - } - }) - return serviceInstance -} - -// GeneratePresentation creates a PDF presentation with custom HTML pages -// Each page is rendered in 16:9 landscape format with complete creative freedom -func (s *Service) GeneratePresentation(config PresentationConfig, userID, conversationID string) (*GeneratedPresentation, error) { - // Validate - if config.Title == "" { - config.Title = "Presentation" - } - if len(config.Slides) == 0 { - return nil, fmt.Errorf("presentation must have at least one slide") - } - - // Generate multi-page HTML with CSS page breaks - htmlContent := s.generateMultiPageHTML(config) - - // Generate unique ID and filename - presentationID := uuid.New().String() - safeFilename := sanitizeFilename(config.Title) + ".pdf" - filePath := filepath.Join(s.outputDir, presentationID+".pdf") - - // Convert to PDF using chromedp (16:9 landscape) - if err := s.generatePDFLandscape(htmlContent, filePath); err != nil { - return nil, fmt.Errorf("failed to generate PDF: %w", err) - } - - // Get file size - fileInfo, err := os.Stat(filePath) - if err != nil { - return nil, fmt.Errorf("failed to get file info: %w", err) - } - - // Create record - pres := &GeneratedPresentation{ - PresentationID: presentationID, - UserID: userID, - ConversationID: conversationID, - Filename: safeFilename, - FilePath: filePath, - Size: fileInfo.Size(), - ContentType: "application/pdf", - CreatedAt: time.Now(), - } - - // Store - s.mu.Lock() - s.presentations[presentationID] = pres - s.mu.Unlock() - - log.Printf("🎯 [PRESENTATION-SERVICE] Generated PDF presentation: %s (%d bytes, %d pages)", safeFilename, fileInfo.Size(), len(config.Slides)) - - return pres, nil -} - -// generateMultiPageHTML creates HTML document with each slide as a separate page -// Extracts content from each slide's HTML and renders directly (no iframes) -func (s *Service) generateMultiPageHTML(config PresentationConfig) string { - var pagesHTML bytes.Buffer - var allStyles bytes.Buffer - - for i, slide := range config.Slides { - slideHTML := slide.HTML - - // Extract styles from (everything between ) - styleStart := strings.Index(strings.ToLower(slideHTML), "") - var styles string - if styleStart != -1 && styleEnd != -1 { - // Find the end of the opening - - -%s - -`, html.EscapeString(config.Title), allStyles.String(), pagesHTML.String()) -} - -// generatePDFLandscape converts HTML to PDF using chromedp with 16:9 landscape format -func (s *Service) generatePDFLandscape(htmlContent, outputPath string) error { - // Create allocator options for headless Chrome - opts := append(chromedp.DefaultExecAllocatorOptions[:], - chromedp.ExecPath("/usr/bin/chromium-browser"), - chromedp.NoSandbox, - chromedp.DisableGPU, - chromedp.Flag("disable-dev-shm-usage", true), - chromedp.Flag("no-first-run", true), - chromedp.Flag("no-default-browser-check", true), - ) - - // Create allocator context - allocCtx, allocCancel := chromedp.NewExecAllocator(context.Background(), opts...) - defer allocCancel() - - // Create context - ctx, cancel := chromedp.NewContext(allocCtx) - defer cancel() - - // Set timeout - ctx, cancel = context.WithTimeout(ctx, 60*time.Second) - defer cancel() - - // Generate PDF in 16:9 landscape format - var pdfBuffer []byte - if err := chromedp.Run(ctx, - chromedp.Navigate("about:blank"), - chromedp.ActionFunc(func(ctx context.Context) error { - frameTree, err := page.GetFrameTree().Do(ctx) - if err != nil { - return err - } - return page.SetDocumentContent(frameTree.Frame.ID, htmlContent).Do(ctx) - }), - chromedp.Sleep(3*time.Second), // Wait for fonts, images, and custom styles to load - chromedp.ActionFunc(func(ctx context.Context) error { - var err error - // Use PreferCSSPageSize to ensure Chrome respects CSS page-break properties - pdfBuffer, _, err = page.PrintToPDF(). - WithPrintBackground(true). - WithDisplayHeaderFooter(false). - WithMarginTop(0). - WithMarginBottom(0). - WithMarginLeft(0). - WithMarginRight(0). - WithPaperWidth(10.67). // 16:9 landscape width - WithPaperHeight(6). // 16:9 landscape height - WithPreferCSSPageSize(true). // CRITICAL: Enables CSS page-break properties - Do(ctx) - return err - }), - ); err != nil { - return err - } - - // Write PDF to file - if err := os.WriteFile(outputPath, pdfBuffer, 0600); err != nil { - return err - } - - return nil -} - -// sanitizeFilename removes invalid characters from filename -func sanitizeFilename(name string) string { - // Replace invalid characters with underscore - invalid := []string{"/", "\\", ":", "*", "?", "\"", "<", ">", "|"} - result := name - for _, char := range invalid { - result = strings.ReplaceAll(result, char, "_") - } - // Limit length - if len(result) > 50 { - result = result[:50] - } - if result == "" { - result = "presentation" - } - return result -} - -// GetPresentation retrieves a presentation by ID -func (s *Service) GetPresentation(presentationID string) (*GeneratedPresentation, bool) { - s.mu.RLock() - defer s.mu.RUnlock() - pres, exists := s.presentations[presentationID] - return pres, exists -} - diff --git a/backend/internal/securefile/service.go b/backend/internal/securefile/service.go deleted file mode 100644 index ca6f8d70..00000000 --- a/backend/internal/securefile/service.go +++ /dev/null @@ -1,374 +0,0 @@ -package securefile - -import ( - "crypto/rand" - "crypto/sha256" - "encoding/hex" - "fmt" - "log" - "os" - "path/filepath" - "sync" - "time" - - "github.com/google/uuid" - "github.com/patrickmn/go-cache" -) - -// File represents a file stored with access code protection -type File struct { - ID string `json:"id"` - UserID string `json:"user_id"` - Filename string `json:"filename"` - MimeType string `json:"mime_type"` - Size int64 `json:"size"` - FilePath string `json:"-"` // Not exposed in JSON - AccessCodeHash string `json:"-"` // SHA256 hash, not exposed - CreatedAt time.Time `json:"created_at"` - ExpiresAt time.Time `json:"expires_at"` -} - -// Result is returned when creating a file -type Result struct { - ID string `json:"id"` - Filename string `json:"filename"` - DownloadURL string `json:"download_url"` - AccessCode string `json:"access_code"` // Only returned once at creation - Size int64 `json:"size"` - MimeType string `json:"mime_type"` - ExpiresAt time.Time `json:"expires_at"` -} - -// Service manages files with access code protection -type Service struct { - cache *cache.Cache - storageDir string - mu sync.RWMutex -} - -var ( - instance *Service - once sync.Once -) - -// GetService returns the singleton secure file service -func GetService() *Service { - once.Do(func() { - instance = NewService("./secure_files") - }) - return instance -} - -// NewService creates a new secure file service -func NewService(storageDir string) *Service { - // Create storage directory if it doesn't exist - if err := os.MkdirAll(storageDir, 0700); err != nil { - log.Printf("⚠️ [SECURE-FILE] Failed to create storage directory: %v", err) - } - - // 30-day default expiration, cleanup every hour - c := cache.New(30*24*time.Hour, 1*time.Hour) - - // Set eviction handler to delete files when they expire - c.OnEvicted(func(key string, value interface{}) { - if file, ok := value.(*File); ok { - log.Printf("🗑️ [SECURE-FILE] Expiring file %s (%s)", file.ID, file.Filename) - if file.FilePath != "" { - if err := os.Remove(file.FilePath); err != nil && !os.IsNotExist(err) { - log.Printf("⚠️ [SECURE-FILE] Failed to delete expired file: %v", err) - } - } - } - }) - - svc := &Service{ - cache: c, - storageDir: storageDir, - } - - // Run startup cleanup - go svc.cleanupExpiredFiles() - - return svc -} - -// generateAccessCode generates a cryptographically secure access code -func generateAccessCode() (string, error) { - bytes := make([]byte, 16) // 16 bytes = 32 hex characters - if _, err := rand.Read(bytes); err != nil { - return "", err - } - return hex.EncodeToString(bytes), nil -} - -// hashAccessCode creates a SHA256 hash of the access code -func hashAccessCode(code string) string { - hash := sha256.Sum256([]byte(code)) - return hex.EncodeToString(hash[:]) -} - -// CreateFile stores content as a secure file with access code -func (s *Service) CreateFile(userID string, content []byte, filename, mimeType string) (*Result, error) { - s.mu.Lock() - defer s.mu.Unlock() - - // Generate unique ID - fileID := uuid.New().String() - - // Generate access code - accessCode, err := generateAccessCode() - if err != nil { - return nil, fmt.Errorf("failed to generate access code: %w", err) - } - - // Hash the access code for storage - accessCodeHash := hashAccessCode(accessCode) - - // Determine file extension from filename or mime type - ext := filepath.Ext(filename) - if ext == "" { - ext = getExtensionFromMimeType(mimeType) - } - - // Create file path - storedFilename := fmt.Sprintf("%s%s", fileID, ext) - filePath := filepath.Join(s.storageDir, storedFilename) - - // Write file to disk - if err := os.WriteFile(filePath, content, 0600); err != nil { - return nil, fmt.Errorf("failed to write file: %w", err) - } - - // Set expiration (30 days) - now := time.Now() - expiresAt := now.Add(30 * 24 * time.Hour) - - // Create secure file record - secureFile := &File{ - ID: fileID, - UserID: userID, - Filename: filename, - MimeType: mimeType, - Size: int64(len(content)), - FilePath: filePath, - AccessCodeHash: accessCodeHash, - CreatedAt: now, - ExpiresAt: expiresAt, - } - - // Store in cache - s.cache.Set(fileID, secureFile, 30*24*time.Hour) - - log.Printf("✅ [SECURE-FILE] Created file %s (%s) for user %s, expires %s", - fileID, filename, userID, expiresAt.Format(time.RFC3339)) - - // Build download URL with full backend URL for LLM tools - // LLM agents pass this URL to other tools (Discord, SendGrid, etc.) which need full URLs - backendURL := os.Getenv("BACKEND_URL") - if backendURL == "" { - backendURL = "http://localhost:3001" // Default fallback for development - } - downloadURL := fmt.Sprintf("%s/api/files/%s?code=%s", backendURL, fileID, accessCode) - - return &Result{ - ID: fileID, - Filename: filename, - DownloadURL: downloadURL, - AccessCode: accessCode, // Only returned once - Size: int64(len(content)), - MimeType: mimeType, - ExpiresAt: expiresAt, - }, nil -} - -// GetFile retrieves a file if the access code is valid -func (s *Service) GetFile(fileID, accessCode string) (*File, []byte, error) { - s.mu.RLock() - defer s.mu.RUnlock() - - // Get file from cache - value, found := s.cache.Get(fileID) - if !found { - return nil, nil, fmt.Errorf("file not found or expired") - } - - file, ok := value.(*File) - if !ok { - return nil, nil, fmt.Errorf("invalid file data") - } - - // Verify access code - providedHash := hashAccessCode(accessCode) - if providedHash != file.AccessCodeHash { - log.Printf("🚫 [SECURE-FILE] Invalid access code for file %s", fileID) - return nil, nil, fmt.Errorf("invalid access code") - } - - // Read file content - content, err := os.ReadFile(file.FilePath) - if err != nil { - log.Printf("❌ [SECURE-FILE] Failed to read file %s: %v", fileID, err) - return nil, nil, fmt.Errorf("failed to read file") - } - - log.Printf("✅ [SECURE-FILE] File %s accessed successfully", fileID) - return file, content, nil -} - -// GetFileInfo returns file metadata without content (requires access code) -func (s *Service) GetFileInfo(fileID, accessCode string) (*File, error) { - s.mu.RLock() - defer s.mu.RUnlock() - - value, found := s.cache.Get(fileID) - if !found { - return nil, fmt.Errorf("file not found or expired") - } - - file, ok := value.(*File) - if !ok { - return nil, fmt.Errorf("invalid file data") - } - - // Verify access code - providedHash := hashAccessCode(accessCode) - if providedHash != file.AccessCodeHash { - return nil, fmt.Errorf("invalid access code") - } - - return file, nil -} - -// DeleteFile removes a file (requires ownership) -func (s *Service) DeleteFile(fileID, userID string) error { - s.mu.Lock() - defer s.mu.Unlock() - - value, found := s.cache.Get(fileID) - if !found { - return fmt.Errorf("file not found") - } - - file, ok := value.(*File) - if !ok { - return fmt.Errorf("invalid file data") - } - - // Verify ownership - if file.UserID != userID { - return fmt.Errorf("access denied") - } - - // Delete from disk - if file.FilePath != "" { - if err := os.Remove(file.FilePath); err != nil && !os.IsNotExist(err) { - log.Printf("⚠️ [SECURE-FILE] Failed to delete file from disk: %v", err) - } - } - - // Delete from cache - s.cache.Delete(fileID) - - log.Printf("✅ [SECURE-FILE] Deleted file %s", fileID) - return nil -} - -// ListUserFiles returns all files for a user -func (s *Service) ListUserFiles(userID string) []*File { - s.mu.RLock() - defer s.mu.RUnlock() - - var files []*File - for _, item := range s.cache.Items() { - if file, ok := item.Object.(*File); ok { - if file.UserID == userID { - files = append(files, file) - } - } - } - return files -} - -// cleanupExpiredFiles removes files that have expired -func (s *Service) cleanupExpiredFiles() { - // Scan storage directory for orphaned files - entries, err := os.ReadDir(s.storageDir) - if err != nil { - log.Printf("⚠️ [SECURE-FILE] Failed to read storage directory: %v", err) - return - } - - now := time.Now() - orphanedCount := 0 - - for _, entry := range entries { - if entry.IsDir() { - continue - } - - filePath := filepath.Join(s.storageDir, entry.Name()) - info, err := entry.Info() - if err != nil { - continue - } - - // Delete files older than 31 days (1 day buffer) - if now.Sub(info.ModTime()) > 31*24*time.Hour { - log.Printf("🗑️ [SECURE-FILE] Deleting orphaned/expired file: %s", entry.Name()) - if err := os.Remove(filePath); err != nil { - log.Printf("⚠️ [SECURE-FILE] Failed to delete: %v", err) - } else { - orphanedCount++ - } - } - } - - if orphanedCount > 0 { - log.Printf("✅ [SECURE-FILE] Cleaned up %d orphaned files", orphanedCount) - } -} - -// GetStats returns service statistics -func (s *Service) GetStats() map[string]interface{} { - s.mu.RLock() - defer s.mu.RUnlock() - - items := s.cache.Items() - totalSize := int64(0) - - for _, item := range items { - if file, ok := item.Object.(*File); ok { - totalSize += file.Size - } - } - - return map[string]interface{}{ - "total_files": len(items), - "total_size": totalSize, - "storage_dir": s.storageDir, - } -} - -// getExtensionFromMimeType returns a file extension for common mime types -func getExtensionFromMimeType(mimeType string) string { - extensions := map[string]string{ - "application/pdf": ".pdf", - "application/json": ".json", - "text/plain": ".txt", - "text/csv": ".csv", - "text/html": ".html", - "text/css": ".css", - "text/javascript": ".js", - "application/xml": ".xml", - "text/xml": ".xml", - "text/yaml": ".yaml", - "application/x-yaml": ".yaml", - "text/markdown": ".md", - "application/octet-stream": ".bin", - } - - if ext, ok := extensions[mimeType]; ok { - return ext - } - return ".bin" -} diff --git a/backend/internal/securefile/service_test.go b/backend/internal/securefile/service_test.go deleted file mode 100644 index 46d5f429..00000000 --- a/backend/internal/securefile/service_test.go +++ /dev/null @@ -1,452 +0,0 @@ -package securefile - -import ( - "os" - "path/filepath" - "testing" - "time" -) - -// TestNewService tests service creation -func TestNewService(t *testing.T) { - tempDir := t.TempDir() - svc := NewService(tempDir) - - if svc == nil { - t.Fatal("NewService should return non-nil service") - } - - if svc.storageDir != tempDir { - t.Errorf("Expected storage dir %s, got %s", tempDir, svc.storageDir) - } -} - -// TestCreateFile tests file creation with access code -func TestCreateFile(t *testing.T) { - tempDir := t.TempDir() - svc := NewService(tempDir) - - content := []byte("test file content") - userID := "user-123" - filename := "test.txt" - mimeType := "text/plain" - - result, err := svc.CreateFile(userID, content, filename, mimeType) - if err != nil { - t.Fatalf("CreateFile failed: %v", err) - } - - // Verify result fields - if result.ID == "" { - t.Error("File ID should not be empty") - } - if result.Filename != filename { - t.Errorf("Expected filename %s, got %s", filename, result.Filename) - } - if result.AccessCode == "" { - t.Error("Access code should not be empty") - } - if len(result.AccessCode) != 32 { - t.Errorf("Access code should be 32 chars, got %d", len(result.AccessCode)) - } - if result.Size != int64(len(content)) { - t.Errorf("Expected size %d, got %d", len(content), result.Size) - } - if result.MimeType != mimeType { - t.Errorf("Expected mime type %s, got %s", mimeType, result.MimeType) - } - if result.ExpiresAt.Before(time.Now().Add(29 * 24 * time.Hour)) { - t.Error("Expiration should be ~30 days from now") - } - - // Verify download URL format (should contain the API path and code) - expectedURLSuffix := "/api/files/" + result.ID + "?code=" + result.AccessCode - if len(result.DownloadURL) < len(expectedURLSuffix) { - t.Errorf("Download URL too short: %s", result.DownloadURL) - } - // Check that URL ends with the expected path (ignoring the host prefix) - if result.DownloadURL[len(result.DownloadURL)-len(expectedURLSuffix):] != expectedURLSuffix { - t.Errorf("Download URL format incorrect: %s (expected to end with %s)", result.DownloadURL, expectedURLSuffix) - } -} - -// TestGetFile tests retrieving file with valid access code -func TestGetFile(t *testing.T) { - tempDir := t.TempDir() - svc := NewService(tempDir) - - content := []byte("secret content") - userID := "user-456" - filename := "secret.txt" - - result, err := svc.CreateFile(userID, content, filename, "text/plain") - if err != nil { - t.Fatalf("CreateFile failed: %v", err) - } - - // Retrieve with valid access code - file, retrievedContent, err := svc.GetFile(result.ID, result.AccessCode) - if err != nil { - t.Fatalf("GetFile failed: %v", err) - } - - if file.ID != result.ID { - t.Errorf("Expected file ID %s, got %s", result.ID, file.ID) - } - if file.Filename != filename { - t.Errorf("Expected filename %s, got %s", filename, file.Filename) - } - if string(retrievedContent) != string(content) { - t.Errorf("Content mismatch: expected %s, got %s", content, retrievedContent) - } -} - -// TestGetFileInvalidAccessCode tests access code validation -func TestGetFileInvalidAccessCode(t *testing.T) { - tempDir := t.TempDir() - svc := NewService(tempDir) - - content := []byte("protected content") - result, _ := svc.CreateFile("user-789", content, "protected.txt", "text/plain") - - // Try with wrong access code - _, _, err := svc.GetFile(result.ID, "wrong-access-code") - if err == nil { - t.Error("Expected error for invalid access code") - } - if err.Error() != "invalid access code" { - t.Errorf("Expected 'invalid access code' error, got: %v", err) - } -} - -// TestGetFileNotFound tests file not found error -func TestGetFileNotFound(t *testing.T) { - tempDir := t.TempDir() - svc := NewService(tempDir) - - _, _, err := svc.GetFile("non-existent-id", "some-code") - if err == nil { - t.Error("Expected error for non-existent file") - } - if err.Error() != "file not found or expired" { - t.Errorf("Expected 'file not found or expired' error, got: %v", err) - } -} - -// TestGetFileInfo tests retrieving file metadata -func TestGetFileInfo(t *testing.T) { - tempDir := t.TempDir() - svc := NewService(tempDir) - - content := []byte("info test content") - filename := "info-test.json" - mimeType := "application/json" - - result, _ := svc.CreateFile("user-info", content, filename, mimeType) - - // Get info with valid access code - file, err := svc.GetFileInfo(result.ID, result.AccessCode) - if err != nil { - t.Fatalf("GetFileInfo failed: %v", err) - } - - if file.Filename != filename { - t.Errorf("Expected filename %s, got %s", filename, file.Filename) - } - if file.MimeType != mimeType { - t.Errorf("Expected mime type %s, got %s", mimeType, file.MimeType) - } - if file.Size != int64(len(content)) { - t.Errorf("Expected size %d, got %d", len(content), file.Size) - } -} - -// TestGetFileInfoInvalidCode tests GetFileInfo with invalid access code -func TestGetFileInfoInvalidCode(t *testing.T) { - tempDir := t.TempDir() - svc := NewService(tempDir) - - result, _ := svc.CreateFile("user-x", []byte("data"), "file.txt", "text/plain") - - _, err := svc.GetFileInfo(result.ID, "bad-code") - if err == nil { - t.Error("Expected error for invalid access code") - } -} - -// TestDeleteFile tests file deletion with ownership check -func TestDeleteFile(t *testing.T) { - tempDir := t.TempDir() - svc := NewService(tempDir) - - userID := "owner-user" - content := []byte("deletable content") - - result, _ := svc.CreateFile(userID, content, "delete-me.txt", "text/plain") - - // Verify file exists on disk - files, _ := os.ReadDir(tempDir) - initialCount := len(files) - - // Delete with correct owner - err := svc.DeleteFile(result.ID, userID) - if err != nil { - t.Fatalf("DeleteFile failed: %v", err) - } - - // Verify file removed from disk - files, _ = os.ReadDir(tempDir) - if len(files) != initialCount-1 { - t.Error("File should be deleted from disk") - } - - // Verify file removed from cache - _, _, err = svc.GetFile(result.ID, result.AccessCode) - if err == nil { - t.Error("File should not be accessible after deletion") - } -} - -// TestDeleteFileWrongOwner tests deletion by non-owner -func TestDeleteFileWrongOwner(t *testing.T) { - tempDir := t.TempDir() - svc := NewService(tempDir) - - result, _ := svc.CreateFile("owner-a", []byte("data"), "owned.txt", "text/plain") - - // Try to delete with wrong owner - err := svc.DeleteFile(result.ID, "owner-b") - if err == nil { - t.Error("Expected error for non-owner deletion") - } - if err.Error() != "access denied" { - t.Errorf("Expected 'access denied' error, got: %v", err) - } -} - -// TestDeleteFileNotFound tests deleting non-existent file -func TestDeleteFileNotFound(t *testing.T) { - tempDir := t.TempDir() - svc := NewService(tempDir) - - err := svc.DeleteFile("non-existent", "user-1") - if err == nil { - t.Error("Expected error for non-existent file") - } -} - -// TestListUserFiles tests listing files for a user -func TestListUserFiles(t *testing.T) { - tempDir := t.TempDir() - svc := NewService(tempDir) - - userA := "user-a" - userB := "user-b" - - // Create files for user A - svc.CreateFile(userA, []byte("file1"), "file1.txt", "text/plain") - svc.CreateFile(userA, []byte("file2"), "file2.txt", "text/plain") - - // Create file for user B - svc.CreateFile(userB, []byte("file3"), "file3.txt", "text/plain") - - // List user A's files - filesA := svc.ListUserFiles(userA) - if len(filesA) != 2 { - t.Errorf("Expected 2 files for user A, got %d", len(filesA)) - } - - // List user B's files - filesB := svc.ListUserFiles(userB) - if len(filesB) != 1 { - t.Errorf("Expected 1 file for user B, got %d", len(filesB)) - } - - // List non-existent user's files - filesC := svc.ListUserFiles("user-c") - if len(filesC) != 0 { - t.Errorf("Expected 0 files for user C, got %d", len(filesC)) - } -} - -// TestGetStats tests service statistics -func TestGetStats(t *testing.T) { - tempDir := t.TempDir() - svc := NewService(tempDir) - - // Create some files - svc.CreateFile("user-1", []byte("content 1"), "file1.txt", "text/plain") - svc.CreateFile("user-2", []byte("content 2 longer"), "file2.txt", "text/plain") - - stats := svc.GetStats() - - totalFiles, ok := stats["total_files"].(int) - if !ok || totalFiles != 2 { - t.Errorf("Expected 2 total files, got %v", stats["total_files"]) - } - - totalSize, ok := stats["total_size"].(int64) - if !ok || totalSize != int64(len("content 1")+len("content 2 longer")) { - t.Errorf("Expected total size %d, got %v", len("content 1")+len("content 2 longer"), stats["total_size"]) - } - - storageDir, ok := stats["storage_dir"].(string) - if !ok || storageDir != tempDir { - t.Errorf("Expected storage dir %s, got %v", tempDir, stats["storage_dir"]) - } -} - -// TestAccessCodeGeneration tests access code is cryptographically random -func TestAccessCodeGeneration(t *testing.T) { - codes := make(map[string]bool) - - // Generate many codes and check uniqueness - for i := 0; i < 100; i++ { - code, err := generateAccessCode() - if err != nil { - t.Fatalf("generateAccessCode failed: %v", err) - } - - if len(code) != 32 { - t.Errorf("Access code length should be 32, got %d", len(code)) - } - - if codes[code] { - t.Error("Duplicate access code generated") - } - codes[code] = true - } -} - -// TestAccessCodeHashing tests SHA256 hashing -func TestAccessCodeHashing(t *testing.T) { - code := "test-access-code-12345678" - hash := hashAccessCode(code) - - // SHA256 produces 64-char hex string - if len(hash) != 64 { - t.Errorf("Hash length should be 64, got %d", len(hash)) - } - - // Same input should produce same hash - hash2 := hashAccessCode(code) - if hash != hash2 { - t.Error("Same input should produce same hash") - } - - // Different input should produce different hash - hash3 := hashAccessCode("different-code") - if hash == hash3 { - t.Error("Different inputs should produce different hashes") - } -} - -// TestMimeTypeExtensionMapping tests extension detection from MIME type -func TestMimeTypeExtensionMapping(t *testing.T) { - testCases := []struct { - mimeType string - expected string - }{ - {"application/pdf", ".pdf"}, - {"application/json", ".json"}, - {"text/plain", ".txt"}, - {"text/csv", ".csv"}, - {"text/html", ".html"}, - {"unknown/type", ".bin"}, - } - - for _, tc := range testCases { - result := getExtensionFromMimeType(tc.mimeType) - if result != tc.expected { - t.Errorf("getExtensionFromMimeType(%s) = %s, expected %s", tc.mimeType, result, tc.expected) - } - } -} - -// TestFileExtensionFromFilename tests extension detection from filename -func TestFileExtensionFromFilename(t *testing.T) { - tempDir := t.TempDir() - svc := NewService(tempDir) - - // Create file with extension in filename - result, _ := svc.CreateFile("user", []byte("data"), "document.pdf", "application/pdf") - - // Verify file was created with correct extension - files, _ := filepath.Glob(filepath.Join(tempDir, result.ID+"*")) - if len(files) != 1 { - t.Fatal("Expected exactly one file") - } - - if filepath.Ext(files[0]) != ".pdf" { - t.Errorf("Expected .pdf extension, got %s", filepath.Ext(files[0])) - } -} - -// TestConcurrentAccess tests thread safety -func TestConcurrentAccess(t *testing.T) { - tempDir := t.TempDir() - svc := NewService(tempDir) - - done := make(chan bool, 10) - - // Create files concurrently - for i := 0; i < 10; i++ { - go func(idx int) { - _, err := svc.CreateFile("concurrent-user", []byte("data"), "concurrent.txt", "text/plain") - if err != nil { - t.Errorf("Concurrent CreateFile failed: %v", err) - } - done <- true - }(i) - } - - // Wait for all goroutines - for i := 0; i < 10; i++ { - <-done - } - - // Verify all files created - files := svc.ListUserFiles("concurrent-user") - if len(files) != 10 { - t.Errorf("Expected 10 files, got %d", len(files)) - } -} - -// Benchmark tests -func BenchmarkCreateFile(b *testing.B) { - tempDir := b.TempDir() - svc := NewService(tempDir) - content := []byte("benchmark content") - - b.ResetTimer() - for i := 0; i < b.N; i++ { - svc.CreateFile("bench-user", content, "bench.txt", "text/plain") - } -} - -func BenchmarkGetFile(b *testing.B) { - tempDir := b.TempDir() - svc := NewService(tempDir) - content := []byte("benchmark content") - - result, _ := svc.CreateFile("bench-user", content, "bench.txt", "text/plain") - - b.ResetTimer() - for i := 0; i < b.N; i++ { - svc.GetFile(result.ID, result.AccessCode) - } -} - -func BenchmarkAccessCodeGeneration(b *testing.B) { - for i := 0; i < b.N; i++ { - generateAccessCode() - } -} - -func BenchmarkAccessCodeHashing(b *testing.B) { - code := "test-access-code-12345678" - b.ResetTimer() - for i := 0; i < b.N; i++ { - hashAccessCode(code) - } -} diff --git a/backend/internal/security/crypto.go b/backend/internal/security/crypto.go deleted file mode 100644 index 7e934b2f..00000000 --- a/backend/internal/security/crypto.go +++ /dev/null @@ -1,184 +0,0 @@ -package security - -import ( - "crypto/aes" - "crypto/cipher" - "crypto/rand" - "fmt" - "io" - "os" -) - -// EncryptionKey represents a 32-byte AES-256 key -type EncryptionKey [32]byte - -// GenerateKey creates a cryptographically secure random encryption key -func GenerateKey() (*EncryptionKey, error) { - var key EncryptionKey - if _, err := io.ReadFull(rand.Reader, key[:]); err != nil { - return nil, fmt.Errorf("failed to generate encryption key: %w", err) - } - return &key, nil -} - -// EncryptFile encrypts a file using AES-256-GCM and saves it to destPath -// Returns the encryption key used (needed for decryption) -func EncryptFile(srcPath, destPath string) (*EncryptionKey, error) { - // Generate encryption key - key, err := GenerateKey() - if err != nil { - return nil, err - } - - // Read source file - plaintext, err := os.ReadFile(srcPath) - if err != nil { - return nil, fmt.Errorf("failed to read source file: %w", err) - } - - // Encrypt data - ciphertext, err := EncryptData(plaintext, key) - if err != nil { - return nil, err - } - - // Write encrypted file - if err := os.WriteFile(destPath, ciphertext, 0600); err != nil { - return nil, fmt.Errorf("failed to write encrypted file: %w", err) - } - - return key, nil -} - -// DecryptFile decrypts a file and returns the plaintext data -// File is NOT written to disk - returned in memory only -func DecryptFile(srcPath string, key *EncryptionKey) ([]byte, error) { - // Read encrypted file - ciphertext, err := os.ReadFile(srcPath) - if err != nil { - return nil, fmt.Errorf("failed to read encrypted file: %w", err) - } - - // Decrypt data - plaintext, err := DecryptData(ciphertext, key) - if err != nil { - return nil, err - } - - return plaintext, nil -} - -// EncryptData encrypts data using AES-256-GCM -func EncryptData(plaintext []byte, key *EncryptionKey) ([]byte, error) { - // Create cipher block - block, err := aes.NewCipher(key[:]) - if err != nil { - return nil, fmt.Errorf("failed to create cipher: %w", err) - } - - // Create GCM mode - gcm, err := cipher.NewGCM(block) - if err != nil { - return nil, fmt.Errorf("failed to create GCM: %w", err) - } - - // Generate nonce - nonce := make([]byte, gcm.NonceSize()) - if _, err := io.ReadFull(rand.Reader, nonce); err != nil { - return nil, fmt.Errorf("failed to generate nonce: %w", err) - } - - // Encrypt and authenticate - // Format: [nonce][ciphertext+tag] - ciphertext := gcm.Seal(nonce, nonce, plaintext, nil) - - return ciphertext, nil -} - -// DecryptData decrypts data using AES-256-GCM -func DecryptData(ciphertext []byte, key *EncryptionKey) ([]byte, error) { - // Create cipher block - block, err := aes.NewCipher(key[:]) - if err != nil { - return nil, fmt.Errorf("failed to create cipher: %w", err) - } - - // Create GCM mode - gcm, err := cipher.NewGCM(block) - if err != nil { - return nil, fmt.Errorf("failed to create GCM: %w", err) - } - - // Extract nonce - nonceSize := gcm.NonceSize() - if len(ciphertext) < nonceSize { - return nil, fmt.Errorf("ciphertext too short") - } - - nonce, ciphertext := ciphertext[:nonceSize], ciphertext[nonceSize:] - - // Decrypt and verify authentication tag - plaintext, err := gcm.Open(nil, nonce, ciphertext, nil) - if err != nil { - return nil, fmt.Errorf("decryption failed (tampered or wrong key): %w", err) - } - - return plaintext, nil -} - -// SecureDeleteFile deletes a file and attempts to overwrite its contents first -// Note: This is best-effort on modern filesystems with journaling/SSDs -func SecureDeleteFile(path string) error { - // Get file info - info, err := os.Stat(path) - if err != nil { - return err - } - - // Open file for writing - file, err := os.OpenFile(path, os.O_WRONLY, 0600) - if err != nil { - return err - } - - // Overwrite with zeros - zeros := make([]byte, info.Size()) - if _, err := file.Write(zeros); err != nil { - file.Close() - return err - } - - // Sync to disk - if err := file.Sync(); err != nil { - file.Close() - return err - } - - // Overwrite with random data - random := make([]byte, info.Size()) - if _, err := io.ReadFull(rand.Reader, random); err != nil { - file.Close() - return err - } - - if _, err := file.Seek(0, 0); err != nil { - file.Close() - return err - } - - if _, err := file.Write(random); err != nil { - file.Close() - return err - } - - // Final sync - if err := file.Sync(); err != nil { - file.Close() - return err - } - - file.Close() - - // Delete file - return os.Remove(path) -} diff --git a/backend/internal/security/hash.go b/backend/internal/security/hash.go deleted file mode 100644 index 0ba708fb..00000000 --- a/backend/internal/security/hash.go +++ /dev/null @@ -1,88 +0,0 @@ -package security - -import ( - "crypto/sha256" - "crypto/subtle" - "encoding/hex" - "fmt" - "io" - "os" -) - -// Hash represents a SHA-256 hash (32 bytes) -type Hash [32]byte - -// CalculateFileHash computes the SHA-256 hash of a file -func CalculateFileHash(path string) (*Hash, error) { - file, err := os.Open(path) - if err != nil { - return nil, fmt.Errorf("failed to open file: %w", err) - } - defer file.Close() - - hash := sha256.New() - if _, err := io.Copy(hash, file); err != nil { - return nil, fmt.Errorf("failed to read file: %w", err) - } - - var result Hash - copy(result[:], hash.Sum(nil)) - return &result, nil -} - -// CalculateDataHash computes the SHA-256 hash of byte data -func CalculateDataHash(data []byte) *Hash { - hashArray := sha256.Sum256(data) - hash := Hash(hashArray) - return &hash -} - -// String returns the hash as a hex string -func (h *Hash) String() string { - return hex.EncodeToString(h[:]) -} - -// Bytes returns the hash as a byte slice -func (h *Hash) Bytes() []byte { - return h[:] -} - -// Equal compares two hashes using constant-time comparison -// This prevents timing attacks -func (h *Hash) Equal(other *Hash) bool { - if other == nil { - return false - } - return subtle.ConstantTimeCompare(h[:], other[:]) == 1 -} - -// FromHexString creates a Hash from a hex string -func FromHexString(s string) (*Hash, error) { - bytes, err := hex.DecodeString(s) - if err != nil { - return nil, fmt.Errorf("invalid hex string: %w", err) - } - - if len(bytes) != 32 { - return nil, fmt.Errorf("invalid hash length: expected 32 bytes, got %d", len(bytes)) - } - - var hash Hash - copy(hash[:], bytes) - return &hash, nil -} - -// Verify checks if the given data matches the hash -func (h *Hash) Verify(data []byte) bool { - computed := CalculateDataHash(data) - return h.Equal(computed) -} - -// VerifyFile checks if the given file matches the hash -func (h *Hash) VerifyFile(path string) (bool, error) { - computed, err := CalculateFileHash(path) - if err != nil { - return false, err - } - return h.Equal(computed), nil -} diff --git a/backend/internal/security/memory.go b/backend/internal/security/memory.go deleted file mode 100644 index 040c6427..00000000 --- a/backend/internal/security/memory.go +++ /dev/null @@ -1,183 +0,0 @@ -package security - -import ( - "crypto/rand" - "runtime" - "sync" -) - -// SecureString holds sensitive string data that can be securely wiped -type SecureString struct { - data []byte - mu sync.Mutex -} - -// NewSecureString creates a new SecureString from a regular string -func NewSecureString(s string) *SecureString { - ss := &SecureString{ - data: []byte(s), - } - // Set finalizer to wipe memory when garbage collected - runtime.SetFinalizer(ss, func(s *SecureString) { - s.Wipe() - }) - return ss -} - -// String returns the string value (use sparingly) -func (s *SecureString) String() string { - s.mu.Lock() - defer s.mu.Unlock() - if s.data == nil { - return "" - } - return string(s.data) -} - -// Bytes returns the byte slice (use sparingly) -func (s *SecureString) Bytes() []byte { - s.mu.Lock() - defer s.mu.Unlock() - if s.data == nil { - return nil - } - // Return a copy to prevent external modification - result := make([]byte, len(s.data)) - copy(result, s.data) - return result -} - -// Len returns the length of the string -func (s *SecureString) Len() int { - s.mu.Lock() - defer s.mu.Unlock() - if s.data == nil { - return 0 - } - return len(s.data) -} - -// IsEmpty returns true if the string is empty or wiped -func (s *SecureString) IsEmpty() bool { - s.mu.Lock() - defer s.mu.Unlock() - return s.data == nil || len(s.data) == 0 -} - -// Wipe securely erases the string data from memory -func (s *SecureString) Wipe() { - s.mu.Lock() - defer s.mu.Unlock() - - if s.data == nil { - return - } - - // Three-pass wipe: zeros, random, zeros - // Pass 1: Overwrite with zeros - for i := range s.data { - s.data[i] = 0 - } - - // Pass 2: Overwrite with random data - if len(s.data) > 0 { - random := make([]byte, len(s.data)) - rand.Read(random) - copy(s.data, random) - // Wipe the random buffer too - for i := range random { - random[i] = 0 - } - } - - // Pass 3: Overwrite with zeros again - for i := range s.data { - s.data[i] = 0 - } - - // Clear the slice - s.data = nil -} - -// WipeBytes securely wipes a byte slice -func WipeBytes(data []byte) { - if data == nil { - return - } - - // Pass 1: Zeros - for i := range data { - data[i] = 0 - } - - // Pass 2: Random - if len(data) > 0 { - random := make([]byte, len(data)) - rand.Read(random) - copy(data, random) - // Wipe random buffer - for i := range random { - random[i] = 0 - } - } - - // Pass 3: Zeros - for i := range data { - data[i] = 0 - } -} - -// WipeString securely wipes a string by converting to byte slice -// Note: Strings in Go are immutable, so this creates a copy -// Use SecureString for better security -func WipeString(s *string) { - if s == nil || *s == "" { - return - } - data := []byte(*s) - WipeBytes(data) - *s = "" -} - -// SecureBuffer is a buffer that automatically wipes its contents -type SecureBuffer struct { - data []byte - mu sync.Mutex -} - -// NewSecureBuffer creates a new secure buffer with given size -func NewSecureBuffer(size int) *SecureBuffer { - sb := &SecureBuffer{ - data: make([]byte, size), - } - runtime.SetFinalizer(sb, func(b *SecureBuffer) { - b.Wipe() - }) - return sb -} - -// Write writes data to the buffer -func (b *SecureBuffer) Write(data []byte) { - b.mu.Lock() - defer b.mu.Unlock() - if len(data) <= len(b.data) { - copy(b.data, data) - } -} - -// Read reads data from the buffer -func (b *SecureBuffer) Read() []byte { - b.mu.Lock() - defer b.mu.Unlock() - result := make([]byte, len(b.data)) - copy(result, b.data) - return result -} - -// Wipe securely erases the buffer -func (b *SecureBuffer) Wipe() { - b.mu.Lock() - defer b.mu.Unlock() - WipeBytes(b.data) - b.data = nil -} diff --git a/backend/internal/security/oauth_state.go b/backend/internal/security/oauth_state.go deleted file mode 100644 index 10054b53..00000000 --- a/backend/internal/security/oauth_state.go +++ /dev/null @@ -1,106 +0,0 @@ -package security - -import ( - "crypto/rand" - "encoding/hex" - "errors" - "sync" - "time" -) - -// OAuthState represents a stored OAuth state token -type OAuthState struct { - UserID string - Service string // "gmail" or "googlesheets" - ExpiresAt time.Time -} - -// OAuthStateStore manages OAuth state tokens with expiration -type OAuthStateStore struct { - states map[string]*OAuthState - mutex sync.RWMutex -} - -// NewOAuthStateStore creates a new OAuth state store -func NewOAuthStateStore() *OAuthStateStore { - store := &OAuthStateStore{ - states: make(map[string]*OAuthState), - } - - // Start cleanup goroutine to remove expired states every minute - go store.cleanupExpired() - - return store -} - -// GenerateState generates a cryptographically secure random state token -func (s *OAuthStateStore) GenerateState(userID, service string) (string, error) { - // Generate 32 random bytes (256 bits) - randomBytes := make([]byte, 32) - if _, err := rand.Read(randomBytes); err != nil { - return "", err - } - - stateToken := hex.EncodeToString(randomBytes) - - s.mutex.Lock() - defer s.mutex.Unlock() - - // Store state with 10-minute expiration - s.states[stateToken] = &OAuthState{ - UserID: userID, - Service: service, - ExpiresAt: time.Now().Add(10 * time.Minute), - } - - return stateToken, nil -} - -// ValidateState validates a state token and returns the associated user ID -// The state token is consumed (one-time use) after validation -func (s *OAuthStateStore) ValidateState(stateToken string) (string, string, error) { - s.mutex.Lock() - defer s.mutex.Unlock() - - state, exists := s.states[stateToken] - if !exists { - return "", "", errors.New("invalid or expired state token") - } - - // Check expiration - if time.Now().After(state.ExpiresAt) { - delete(s.states, stateToken) - return "", "", errors.New("state token expired") - } - - // Delete state token (one-time use for CSRF protection) - userID := state.UserID - service := state.Service - delete(s.states, stateToken) - - return userID, service, nil -} - -// cleanupExpired removes expired state tokens every minute -func (s *OAuthStateStore) cleanupExpired() { - ticker := time.NewTicker(1 * time.Minute) - defer ticker.Stop() - - for range ticker.C { - s.mutex.Lock() - now := time.Now() - for token, state := range s.states { - if now.After(state.ExpiresAt) { - delete(s.states, token) - } - } - s.mutex.Unlock() - } -} - -// Count returns the number of active state tokens (for monitoring) -func (s *OAuthStateStore) Count() int { - s.mutex.RLock() - defer s.mutex.RUnlock() - return len(s.states) -} diff --git a/backend/internal/security/path_validation.go b/backend/internal/security/path_validation.go deleted file mode 100644 index a033b962..00000000 --- a/backend/internal/security/path_validation.go +++ /dev/null @@ -1,43 +0,0 @@ -package security - -import ( - "fmt" - "regexp" - "strings" -) - -// ValidateFileID validates that a file ID is a valid UUID and contains no path traversal sequences. -// This prevents path traversal attacks like "../../../etc/passwd" or absolute paths. -// -// Returns an error if the fileID: -// - Is empty -// - Contains path traversal sequences (.., /, \) -// - Is not a valid UUID format (xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx) -// -// This function should be called before using any user-provided file ID in file system operations. -func ValidateFileID(fileID string) error { - if fileID == "" { - return fmt.Errorf("file_id cannot be empty") - } - - // Check for path traversal sequences - if strings.Contains(fileID, "..") { - return fmt.Errorf("invalid file_id: path traversal attempt detected (..)") - } - if strings.Contains(fileID, "/") { - return fmt.Errorf("invalid file_id: path traversal attempt detected (/)") - } - if strings.Contains(fileID, "\\") { - return fmt.Errorf("invalid file_id: path traversal attempt detected (\\)") - } - - // Validate UUID format (standard UUID format: xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx) - // UUIDs are always 36 characters with hyphens at positions 8, 13, 18, 23 - // This regex matches UUID v1, v4, and other valid UUID formats - uuidPattern := regexp.MustCompile(`^[a-fA-F0-9]{8}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{12}$`) - if !uuidPattern.MatchString(fileID) { - return fmt.Errorf("invalid file_id format: expected UUID (got %q)", fileID) - } - - return nil -} diff --git a/backend/internal/security/ssrf.go b/backend/internal/security/ssrf.go deleted file mode 100644 index 97095f38..00000000 --- a/backend/internal/security/ssrf.go +++ /dev/null @@ -1,158 +0,0 @@ -package security - -import ( - "fmt" - "net" - "net/url" - "strings" -) - -// privateIPRanges contains CIDR ranges for private/internal networks -var privateIPRanges = []string{ - "127.0.0.0/8", // IPv4 loopback - "10.0.0.0/8", // RFC1918 private - "172.16.0.0/12", // RFC1918 private - "192.168.0.0/16", // RFC1918 private - "169.254.0.0/16", // Link-local - "::1/128", // IPv6 loopback - "fc00::/7", // IPv6 unique local - "fe80::/10", // IPv6 link-local - "0.0.0.0/8", // "This" network -} - -// blockedHostnames contains hostnames that should never be accessed -var blockedHostnames = []string{ - "localhost", - "localhost.localdomain", - "ip6-localhost", - "ip6-loopback", - "metadata.google.internal", // GCP metadata - "169.254.169.254", // AWS/GCP/Azure metadata endpoint - "metadata.google.internal.", // GCP metadata with trailing dot - "kubernetes.default.svc", // Kubernetes - "kubernetes.default", // Kubernetes -} - -var parsedCIDRs []*net.IPNet - -func init() { - // Pre-parse CIDR ranges for efficiency - for _, cidr := range privateIPRanges { - _, network, err := net.ParseCIDR(cidr) - if err == nil { - parsedCIDRs = append(parsedCIDRs, network) - } - } -} - -// IsPrivateIP checks if an IP address is in a private/internal range -func IsPrivateIP(ip net.IP) bool { - if ip == nil { - return true // Treat nil as blocked for safety - } - - for _, network := range parsedCIDRs { - if network.Contains(ip) { - return true - } - } - return false -} - -// IsBlockedHostname checks if a hostname is in the blocklist -func IsBlockedHostname(hostname string) bool { - hostname = strings.ToLower(strings.TrimSuffix(hostname, ".")) - - for _, blocked := range blockedHostnames { - if hostname == blocked { - return true - } - // Also check if it ends with the blocked hostname (subdomain matching) - if strings.HasSuffix(hostname, "."+blocked) { - return true - } - } - return false -} - -// ValidateURLForSSRF validates a URL to prevent SSRF attacks -// Returns an error if the URL points to a private/internal resource -func ValidateURLForSSRF(rawURL string) error { - parsedURL, err := url.Parse(rawURL) - if err != nil { - return fmt.Errorf("invalid URL format: %w", err) - } - - // Only allow http/https schemes - if parsedURL.Scheme != "http" && parsedURL.Scheme != "https" { - return fmt.Errorf("only http and https schemes are allowed") - } - - hostname := parsedURL.Hostname() - if hostname == "" { - return fmt.Errorf("URL must have a hostname") - } - - // Check against blocked hostnames - if IsBlockedHostname(hostname) { - return fmt.Errorf("access to internal hostname '%s' is not allowed", hostname) - } - - // Try to parse as IP address first - ip := net.ParseIP(hostname) - if ip != nil { - if IsPrivateIP(ip) { - return fmt.Errorf("access to private IP address '%s' is not allowed", hostname) - } - return nil - } - - // Resolve hostname to IP addresses - ips, err := net.LookupIP(hostname) - if err != nil { - // DNS resolution failed - allow the request to proceed - // The actual HTTP request will fail if the host is unreachable - return nil - } - - // Check all resolved IPs - for _, resolvedIP := range ips { - if IsPrivateIP(resolvedIP) { - return fmt.Errorf("hostname '%s' resolves to private IP address '%s'", hostname, resolvedIP.String()) - } - } - - return nil -} - -// ValidateURLForSSRFQuick performs a quick validation without DNS resolution -// Use this when DNS resolution overhead is unacceptable -func ValidateURLForSSRFQuick(rawURL string) error { - parsedURL, err := url.Parse(rawURL) - if err != nil { - return fmt.Errorf("invalid URL format: %w", err) - } - - // Only allow http/https schemes - if parsedURL.Scheme != "http" && parsedURL.Scheme != "https" { - return fmt.Errorf("only http and https schemes are allowed") - } - - hostname := parsedURL.Hostname() - if hostname == "" { - return fmt.Errorf("URL must have a hostname") - } - - // Check against blocked hostnames - if IsBlockedHostname(hostname) { - return fmt.Errorf("access to internal hostname '%s' is not allowed", hostname) - } - - // Check if hostname is an IP address - ip := net.ParseIP(hostname) - if ip != nil && IsPrivateIP(ip) { - return fmt.Errorf("access to private IP address '%s' is not allowed", hostname) - } - - return nil -} diff --git a/backend/internal/services/agent_service.go b/backend/internal/services/agent_service.go deleted file mode 100644 index 918a08e8..00000000 --- a/backend/internal/services/agent_service.go +++ /dev/null @@ -1,819 +0,0 @@ -package services - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "context" - "fmt" - "log" - "time" - - "github.com/google/uuid" - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/bson/primitive" - "go.mongodb.org/mongo-driver/mongo" - "go.mongodb.org/mongo-driver/mongo/options" -) - -// ============================================================================ -// MongoDB Records -// ============================================================================ - -// AgentRecord is the MongoDB representation of an agent -type AgentRecord struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"_id,omitempty"` - AgentID string `bson:"agentId" json:"agentId"` // String ID for API compatibility - UserID string `bson:"userId" json:"userId"` - Name string `bson:"name" json:"name"` - Description string `bson:"description,omitempty" json:"description,omitempty"` - Status string `bson:"status" json:"status"` - CreatedAt time.Time `bson:"createdAt" json:"createdAt"` - UpdatedAt time.Time `bson:"updatedAt" json:"updatedAt"` -} - -// ToModel converts AgentRecord to models.Agent -func (r *AgentRecord) ToModel() *models.Agent { - return &models.Agent{ - ID: r.AgentID, - UserID: r.UserID, - Name: r.Name, - Description: r.Description, - Status: r.Status, - CreatedAt: r.CreatedAt, - UpdatedAt: r.UpdatedAt, - } -} - -// WorkflowRecord is the MongoDB representation of a workflow -type WorkflowRecord struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"_id,omitempty"` - WorkflowID string `bson:"workflowId" json:"workflowId"` // String ID for API compatibility - AgentID string `bson:"agentId" json:"agentId"` - Blocks []models.Block `bson:"blocks" json:"blocks"` - Connections []models.Connection `bson:"connections" json:"connections"` - Variables []models.Variable `bson:"variables" json:"variables"` - Version int `bson:"version" json:"version"` - CreatedAt time.Time `bson:"createdAt" json:"createdAt"` - UpdatedAt time.Time `bson:"updatedAt" json:"updatedAt"` -} - -// ToModel converts WorkflowRecord to models.Workflow -func (r *WorkflowRecord) ToModel() *models.Workflow { - return &models.Workflow{ - ID: r.WorkflowID, - AgentID: r.AgentID, - Blocks: r.Blocks, - Connections: r.Connections, - Variables: r.Variables, - Version: r.Version, - CreatedAt: r.CreatedAt, - UpdatedAt: r.UpdatedAt, - } -} - -// WorkflowVersionRecord stores historical workflow versions -type WorkflowVersionRecord struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"_id,omitempty"` - AgentID string `bson:"agentId" json:"agentId"` - Version int `bson:"version" json:"version"` - Blocks []models.Block `bson:"blocks" json:"blocks"` - Connections []models.Connection `bson:"connections" json:"connections"` - Variables []models.Variable `bson:"variables" json:"variables"` - Description string `bson:"description,omitempty" json:"description,omitempty"` - CreatedAt time.Time `bson:"createdAt" json:"createdAt"` -} - -// WorkflowVersionResponse is the API response for workflow versions -type WorkflowVersionResponse struct { - Version int `json:"version"` - Description string `json:"description,omitempty"` - BlockCount int `json:"blockCount"` - CreatedAt time.Time `json:"createdAt"` -} - -// ============================================================================ -// AgentService -// ============================================================================ - -// AgentService handles agent and workflow operations using MongoDB -type AgentService struct { - mongoDB *database.MongoDB -} - -// NewAgentService creates a new agent service -func NewAgentService(mongoDB *database.MongoDB) *AgentService { - return &AgentService{mongoDB: mongoDB} -} - -// Collection helpers -func (s *AgentService) agentsCollection() *mongo.Collection { - return s.mongoDB.Database().Collection("agents") -} - -func (s *AgentService) workflowsCollection() *mongo.Collection { - return s.mongoDB.Database().Collection("workflows") -} - -func (s *AgentService) workflowVersionsCollection() *mongo.Collection { - return s.mongoDB.Database().Collection("workflow_versions") -} - -// ============================================================================ -// Agent CRUD Operations -// ============================================================================ - -// CreateAgent creates a new agent for a user with auto-generated ID -func (s *AgentService) CreateAgent(userID, name, description string) (*models.Agent, error) { - id := uuid.New().String() - return s.CreateAgentWithID(id, userID, name, description) -} - -// CreateAgentWithID creates a new agent with a specific ID (for frontend-generated IDs) -func (s *AgentService) CreateAgentWithID(id, userID, name, description string) (*models.Agent, error) { - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - now := time.Now() - - record := &AgentRecord{ - AgentID: id, - UserID: userID, - Name: name, - Description: description, - Status: "draft", - CreatedAt: now, - UpdatedAt: now, - } - - _, err := s.agentsCollection().InsertOne(ctx, record) - if err != nil { - return nil, fmt.Errorf("failed to create agent: %w", err) - } - - log.Printf("📝 [AGENT] Created agent %s for user %s", id, userID) - return record.ToModel(), nil -} - -// GetAgent retrieves an agent by ID for a specific user -func (s *AgentService) GetAgent(agentID, userID string) (*models.Agent, error) { - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - var record AgentRecord - err := s.agentsCollection().FindOne(ctx, bson.M{ - "agentId": agentID, - "userId": userID, - }).Decode(&record) - - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("agent not found") - } - if err != nil { - return nil, fmt.Errorf("failed to get agent: %w", err) - } - - agent := record.ToModel() - - // Also load the workflow if it exists - workflow, err := s.GetWorkflow(agentID) - if err == nil { - agent.Workflow = workflow - } - - return agent, nil -} - -// GetAgentByID retrieves an agent by ID only (for internal/scheduled use) -// WARNING: This bypasses user ownership check - use only for scheduled jobs -func (s *AgentService) GetAgentByID(agentID string) (*models.Agent, error) { - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - var record AgentRecord - err := s.agentsCollection().FindOne(ctx, bson.M{ - "agentId": agentID, - }).Decode(&record) - - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("agent not found") - } - if err != nil { - return nil, fmt.Errorf("failed to get agent: %w", err) - } - - agent := record.ToModel() - - // Also load the workflow if it exists - workflow, err := s.GetWorkflow(agentID) - if err == nil { - agent.Workflow = workflow - } - - return agent, nil -} - -// ListAgents returns all agents for a user -func (s *AgentService) ListAgents(userID string) ([]*models.Agent, error) { - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - cursor, err := s.agentsCollection().Find(ctx, bson.M{"userId": userID}, - options.Find().SetSort(bson.D{{Key: "updatedAt", Value: -1}})) - if err != nil { - return nil, fmt.Errorf("failed to list agents: %w", err) - } - defer cursor.Close(ctx) - - var records []AgentRecord - if err := cursor.All(ctx, &records); err != nil { - return nil, fmt.Errorf("failed to decode agents: %w", err) - } - - agents := make([]*models.Agent, len(records)) - for i, record := range records { - agents[i] = record.ToModel() - } - - return agents, nil -} - -// UpdateAgent updates an agent's metadata -func (s *AgentService) UpdateAgent(agentID, userID string, req *models.UpdateAgentRequest) (*models.Agent, error) { - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - // First check if agent exists and belongs to user - agent, err := s.GetAgent(agentID, userID) - if err != nil { - return nil, err - } - - // Build update document - updateFields := bson.M{"updatedAt": time.Now()} - if req.Name != "" { - updateFields["name"] = req.Name - agent.Name = req.Name - } - if req.Description != "" { - updateFields["description"] = req.Description - agent.Description = req.Description - } - if req.Status != "" { - updateFields["status"] = req.Status - agent.Status = req.Status - } - - _, err = s.agentsCollection().UpdateOne(ctx, - bson.M{"agentId": agentID, "userId": userID}, - bson.M{"$set": updateFields}) - if err != nil { - return nil, fmt.Errorf("failed to update agent: %w", err) - } - - agent.UpdatedAt = time.Now() - return agent, nil -} - -// DeleteAgent deletes an agent and its workflow/versions -func (s *AgentService) DeleteAgent(agentID, userID string) error { - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - // Delete agent - result, err := s.agentsCollection().DeleteOne(ctx, bson.M{ - "agentId": agentID, - "userId": userID, - }) - if err != nil { - return fmt.Errorf("failed to delete agent: %w", err) - } - if result.DeletedCount == 0 { - return fmt.Errorf("agent not found") - } - - // Cascade delete workflow - s.workflowsCollection().DeleteOne(ctx, bson.M{"agentId": agentID}) - - // Cascade delete workflow versions - s.workflowVersionsCollection().DeleteMany(ctx, bson.M{"agentId": agentID}) - - log.Printf("🗑️ [AGENT] Deleted agent %s and associated workflows", agentID) - return nil -} - -// DeleteAllByUser deletes all agents, workflows, and workflow versions for a user (GDPR compliance) -func (s *AgentService) DeleteAllByUser(ctx context.Context, userID string) (int64, error) { - if userID == "" { - return 0, fmt.Errorf("user ID is required") - } - - // Get all agent IDs for this user first (for cascade deletion) - cursor, err := s.agentsCollection().Find(ctx, bson.M{"userId": userID}) - if err != nil { - return 0, fmt.Errorf("failed to find user agents: %w", err) - } - defer cursor.Close(ctx) - - var agentIDs []string - for cursor.Next(ctx) { - var agent struct { - AgentID string `bson:"agentId"` - } - if err := cursor.Decode(&agent); err == nil { - agentIDs = append(agentIDs, agent.AgentID) - } - } - - // Delete all workflow versions for these agents - if len(agentIDs) > 0 { - _, err := s.workflowVersionsCollection().DeleteMany(ctx, bson.M{ - "agentId": bson.M{"$in": agentIDs}, - }) - if err != nil { - log.Printf("⚠️ [GDPR] Failed to delete workflow versions: %v", err) - } - - // Delete all workflows for these agents - _, err = s.workflowsCollection().DeleteMany(ctx, bson.M{ - "agentId": bson.M{"$in": agentIDs}, - }) - if err != nil { - log.Printf("⚠️ [GDPR] Failed to delete workflows: %v", err) - } - } - - // Delete all agents for this user - result, err := s.agentsCollection().DeleteMany(ctx, bson.M{"userId": userID}) - if err != nil { - return 0, fmt.Errorf("failed to delete agents: %w", err) - } - - log.Printf("🗑️ [GDPR] Deleted %d agents and associated data for user %s", result.DeletedCount, userID) - return result.DeletedCount, nil -} - -// ============================================================================ -// Workflow Operations -// ============================================================================ - -// SaveWorkflow creates or updates a workflow for an agent -func (s *AgentService) SaveWorkflow(agentID, userID string, req *models.SaveWorkflowRequest) (*models.Workflow, error) { - return s.SaveWorkflowWithDescription(agentID, userID, req, "") -} - -// SaveWorkflowWithDescription creates or updates a workflow with a version description -func (s *AgentService) SaveWorkflowWithDescription(agentID, userID string, req *models.SaveWorkflowRequest, description string) (*models.Workflow, error) { - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - // Verify agent exists and belongs to user - _, err := s.GetAgent(agentID, userID) - if err != nil { - return nil, err - } - - now := time.Now() - - // Check if workflow exists - var existingWorkflow WorkflowRecord - err = s.workflowsCollection().FindOne(ctx, bson.M{"agentId": agentID}).Decode(&existingWorkflow) - - var workflow *models.Workflow - var newVersion int - - if err == mongo.ErrNoDocuments { - // Create new workflow - workflowID := uuid.New().String() - newVersion = 1 - - record := &WorkflowRecord{ - WorkflowID: workflowID, - AgentID: agentID, - Blocks: req.Blocks, - Connections: req.Connections, - Variables: req.Variables, - Version: newVersion, - CreatedAt: now, - UpdatedAt: now, - } - - _, err = s.workflowsCollection().InsertOne(ctx, record) - if err != nil { - return nil, fmt.Errorf("failed to create workflow: %w", err) - } - - workflow = record.ToModel() - } else if err != nil { - return nil, fmt.Errorf("failed to check existing workflow: %w", err) - } else { - // Update existing workflow - newVersion = existingWorkflow.Version + 1 - - _, err = s.workflowsCollection().UpdateOne(ctx, - bson.M{"agentId": agentID}, - bson.M{"$set": bson.M{ - "blocks": req.Blocks, - "connections": req.Connections, - "variables": req.Variables, - "version": newVersion, - "updatedAt": now, - }}) - if err != nil { - return nil, fmt.Errorf("failed to update workflow: %w", err) - } - - workflow = &models.Workflow{ - ID: existingWorkflow.WorkflowID, - AgentID: agentID, - Blocks: req.Blocks, - Connections: req.Connections, - Variables: req.Variables, - Version: newVersion, - CreatedAt: existingWorkflow.CreatedAt, - UpdatedAt: now, - } - } - - // Only save workflow version snapshot when explicitly requested (e.g., when AI generates/modifies workflow) - if req.CreateVersion { - versionDescription := description - if req.VersionDescription != "" { - versionDescription = req.VersionDescription - } - - versionRecord := &WorkflowVersionRecord{ - AgentID: agentID, - Version: newVersion, - Blocks: req.Blocks, - Connections: req.Connections, - Variables: req.Variables, - Description: versionDescription, - CreatedAt: now, - } - - _, err = s.workflowVersionsCollection().InsertOne(ctx, versionRecord) - if err != nil { - log.Printf("⚠️ [WORKFLOW] Failed to save version snapshot: %v", err) - // Don't fail the whole operation for version snapshot failure - } else { - log.Printf("📸 [WORKFLOW] Saved version %d snapshot for agent %s", newVersion, agentID) - } - } - - // Update agent's updated_at - s.agentsCollection().UpdateOne(ctx, - bson.M{"agentId": agentID}, - bson.M{"$set": bson.M{"updatedAt": now}}) - - return workflow, nil -} - -// GetWorkflow retrieves a workflow for an agent -func (s *AgentService) GetWorkflow(agentID string) (*models.Workflow, error) { - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - var record WorkflowRecord - err := s.workflowsCollection().FindOne(ctx, bson.M{"agentId": agentID}).Decode(&record) - - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("workflow not found") - } - if err != nil { - return nil, fmt.Errorf("failed to get workflow: %w", err) - } - - return record.ToModel(), nil -} - -// ============================================================================ -// Workflow Version History -// ============================================================================ - -// ListWorkflowVersions returns all versions for an agent's workflow -func (s *AgentService) ListWorkflowVersions(agentID, userID string) ([]WorkflowVersionResponse, error) { - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - // Verify agent belongs to user - _, err := s.GetAgent(agentID, userID) - if err != nil { - return nil, err - } - - cursor, err := s.workflowVersionsCollection().Find(ctx, - bson.M{"agentId": agentID}, - options.Find().SetSort(bson.D{{Key: "version", Value: -1}})) - if err != nil { - return nil, fmt.Errorf("failed to list workflow versions: %w", err) - } - defer cursor.Close(ctx) - - var records []WorkflowVersionRecord - if err := cursor.All(ctx, &records); err != nil { - return nil, fmt.Errorf("failed to decode versions: %w", err) - } - - versions := make([]WorkflowVersionResponse, len(records)) - for i, record := range records { - versions[i] = WorkflowVersionResponse{ - Version: record.Version, - Description: record.Description, - BlockCount: len(record.Blocks), - CreatedAt: record.CreatedAt, - } - } - - return versions, nil -} - -// GetWorkflowVersion retrieves a specific workflow version -func (s *AgentService) GetWorkflowVersion(agentID, userID string, version int) (*models.Workflow, error) { - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - // Verify agent belongs to user - _, err := s.GetAgent(agentID, userID) - if err != nil { - return nil, err - } - - var record WorkflowVersionRecord - err = s.workflowVersionsCollection().FindOne(ctx, bson.M{ - "agentId": agentID, - "version": version, - }).Decode(&record) - - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("workflow version not found") - } - if err != nil { - return nil, fmt.Errorf("failed to get workflow version: %w", err) - } - - return &models.Workflow{ - AgentID: record.AgentID, - Blocks: record.Blocks, - Connections: record.Connections, - Variables: record.Variables, - Version: record.Version, - CreatedAt: record.CreatedAt, - }, nil -} - -// RestoreWorkflowVersion restores a workflow to a previous version -func (s *AgentService) RestoreWorkflowVersion(agentID, userID string, version int) (*models.Workflow, error) { - // Get the version to restore - versionWorkflow, err := s.GetWorkflowVersion(agentID, userID, version) - if err != nil { - return nil, err - } - - // Save as new version with description - restoring always creates a version snapshot - req := &models.SaveWorkflowRequest{ - Blocks: versionWorkflow.Blocks, - Connections: versionWorkflow.Connections, - Variables: versionWorkflow.Variables, - CreateVersion: true, // Always create version when restoring - VersionDescription: fmt.Sprintf("Restored from version %d", version), - } - - return s.SaveWorkflowWithDescription(agentID, userID, req, "") -} - -// ============================================================================ -// Pagination Methods -// ============================================================================ - -// ListAgentsPaginated returns agents with pagination support -func (s *AgentService) ListAgentsPaginated(userID string, limit, offset int) (*models.PaginatedAgentsResponse, error) { - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - if limit <= 0 { - limit = 20 - } - if limit > 100 { - limit = 100 - } - if offset < 0 { - offset = 0 - } - - // Get total count - total, err := s.agentsCollection().CountDocuments(ctx, bson.M{"userId": userID}) - if err != nil { - return nil, fmt.Errorf("failed to count agents: %w", err) - } - - // Get agents with pagination - cursor, err := s.agentsCollection().Find(ctx, - bson.M{"userId": userID}, - options.Find(). - SetSort(bson.D{{Key: "updatedAt", Value: -1}}). - SetSkip(int64(offset)). - SetLimit(int64(limit))) - if err != nil { - return nil, fmt.Errorf("failed to list agents: %w", err) - } - defer cursor.Close(ctx) - - var records []AgentRecord - if err := cursor.All(ctx, &records); err != nil { - return nil, fmt.Errorf("failed to decode agents: %w", err) - } - - // Build list items with workflow info - agents := make([]models.AgentListItem, len(records)) - for i, record := range records { - item := models.AgentListItem{ - ID: record.AgentID, - Name: record.Name, - Description: record.Description, - Status: record.Status, - CreatedAt: record.CreatedAt, - UpdatedAt: record.UpdatedAt, - } - - // Get workflow info - workflow, err := s.GetWorkflow(record.AgentID) - if err == nil { - item.HasWorkflow = true - item.BlockCount = len(workflow.Blocks) - } - - agents[i] = item - } - - return &models.PaginatedAgentsResponse{ - Agents: agents, - Total: int(total), - Limit: limit, - Offset: offset, - HasMore: offset+len(agents) < int(total), - }, nil -} - -// GetRecentAgents returns the 10 most recently updated agents for the landing page -func (s *AgentService) GetRecentAgents(userID string) (*models.RecentAgentsResponse, error) { - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - cursor, err := s.agentsCollection().Find(ctx, - bson.M{"userId": userID}, - options.Find(). - SetSort(bson.D{{Key: "updatedAt", Value: -1}}). - SetLimit(10)) - if err != nil { - return nil, fmt.Errorf("failed to get recent agents: %w", err) - } - defer cursor.Close(ctx) - - var records []AgentRecord - if err := cursor.All(ctx, &records); err != nil { - return nil, fmt.Errorf("failed to decode agents: %w", err) - } - - agents := make([]models.AgentListItem, len(records)) - for i, record := range records { - item := models.AgentListItem{ - ID: record.AgentID, - Name: record.Name, - Description: record.Description, - Status: record.Status, - CreatedAt: record.CreatedAt, - UpdatedAt: record.UpdatedAt, - } - - // Get workflow info - workflow, err := s.GetWorkflow(record.AgentID) - if err == nil { - item.HasWorkflow = true - item.BlockCount = len(workflow.Blocks) - } - - agents[i] = item - } - - return &models.RecentAgentsResponse{ - Agents: agents, - }, nil -} - -// ============================================================================ -// Sync Method (for first-message persistence) -// ============================================================================ - -// SyncAgent creates or updates an agent with its workflow in a single operation -// This is called when a user sends their first message to persist the local agent -func (s *AgentService) SyncAgent(agentID, userID string, req *models.SyncAgentRequest) (*models.Agent, *models.Workflow, error) { - ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) - defer cancel() - - now := time.Now() - - // Check if agent already exists - existingAgent, err := s.GetAgent(agentID, userID) - if err != nil && err.Error() != "agent not found" { - return nil, nil, fmt.Errorf("failed to check existing agent: %w", err) - } - - var agent *models.Agent - - if existingAgent != nil { - // Update existing agent - agent, err = s.UpdateAgent(agentID, userID, &models.UpdateAgentRequest{ - Name: req.Name, - Description: req.Description, - }) - if err != nil { - return nil, nil, fmt.Errorf("failed to update agent: %w", err) - } - } else { - // Create new agent with the provided ID - record := &AgentRecord{ - AgentID: agentID, - UserID: userID, - Name: req.Name, - Description: req.Description, - Status: "draft", - CreatedAt: now, - UpdatedAt: now, - } - - _, err = s.agentsCollection().InsertOne(ctx, record) - if err != nil { - return nil, nil, fmt.Errorf("failed to create agent: %w", err) - } - - agent = record.ToModel() - } - - // Save the workflow - workflow, err := s.SaveWorkflow(agentID, userID, &req.Workflow) - if err != nil { - return nil, nil, fmt.Errorf("failed to save workflow: %w", err) - } - - return agent, workflow, nil -} - -// ============================================================================ -// Index Initialization -// ============================================================================ - -// EnsureIndexes creates indexes for agents and workflows collections -func (s *AgentService) EnsureIndexes(ctx context.Context) error { - // Agents collection indexes - agentIndexes := []mongo.IndexModel{ - { - Keys: bson.D{{Key: "agentId", Value: 1}}, - Options: options.Index().SetUnique(true), - }, - { - Keys: bson.D{ - {Key: "userId", Value: 1}, - {Key: "updatedAt", Value: -1}, - }, - }, - { - Keys: bson.D{{Key: "status", Value: 1}}, - }, - } - - _, err := s.agentsCollection().Indexes().CreateMany(ctx, agentIndexes) - if err != nil { - return fmt.Errorf("failed to create agent indexes: %w", err) - } - - // Workflows collection indexes - workflowIndexes := []mongo.IndexModel{ - { - Keys: bson.D{{Key: "agentId", Value: 1}}, - Options: options.Index().SetUnique(true), - }, - } - - _, err = s.workflowsCollection().Indexes().CreateMany(ctx, workflowIndexes) - if err != nil { - return fmt.Errorf("failed to create workflow indexes: %w", err) - } - - // Workflow versions collection indexes - versionIndexes := []mongo.IndexModel{ - { - Keys: bson.D{ - {Key: "agentId", Value: 1}, - {Key: "version", Value: -1}, - }, - }, - } - - _, err = s.workflowVersionsCollection().Indexes().CreateMany(ctx, versionIndexes) - if err != nil { - return fmt.Errorf("failed to create workflow version indexes: %w", err) - } - - log.Println("✅ [AGENT] Ensured indexes for agents, workflows, and workflow_versions collections") - return nil -} diff --git a/backend/internal/services/analytics_service.go b/backend/internal/services/analytics_service.go deleted file mode 100644 index b12e1e02..00000000 --- a/backend/internal/services/analytics_service.go +++ /dev/null @@ -1,993 +0,0 @@ -package services - -import ( - "context" - "encoding/json" - "log" - "os" - "path/filepath" - "time" - - "claraverse/internal/database" - "claraverse/internal/models" - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/bson/primitive" - "go.mongodb.org/mongo-driver/mongo" -) - -// AnalyticsService handles minimal usage tracking (non-invasive) -type AnalyticsService struct { - mongoDB *database.MongoDB -} - -// NewAnalyticsService creates a new analytics service -func NewAnalyticsService(mongoDB *database.MongoDB) *AnalyticsService { - return &AnalyticsService{ - mongoDB: mongoDB, - } -} - -// ChatSessionAnalytics stores minimal chat session data -type ChatSessionAnalytics struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - UserID string `bson:"userId" json:"userId"` - ConversationID string `bson:"conversationId" json:"conversationId"` - SessionID string `bson:"sessionId" json:"sessionId"` // WebSocket connection ID - - // Minimal metrics - MessageCount int `bson:"messageCount" json:"messageCount"` - StartedAt time.Time `bson:"startedAt" json:"startedAt"` - EndedAt *time.Time `bson:"endedAt,omitempty" json:"endedAt,omitempty"` - DurationMs int64 `bson:"durationMs,omitempty" json:"durationMs,omitempty"` - - // Optional context (if available) - ModelID string `bson:"modelId,omitempty" json:"modelId,omitempty"` - DisabledTools bool `bson:"disabledTools,omitempty" json:"disabledTools,omitempty"` -} - -// AgentUsageAnalytics stores minimal agent execution context -type AgentUsageAnalytics struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - UserID string `bson:"userId" json:"userId"` - AgentID string `bson:"agentId" json:"agentId"` - ExecutionID primitive.ObjectID `bson:"executionId" json:"executionId"` - - TriggerType string `bson:"triggerType" json:"triggerType"` // chat, api, scheduled - ExecutedAt time.Time `bson:"executedAt" json:"executedAt"` -} - -// TrackChatSessionStart records the start of a chat session -func (s *AnalyticsService) TrackChatSessionStart(ctx context.Context, sessionID, userID, conversationID string) error { - if s.mongoDB == nil { - return nil // Analytics disabled - } - - session := &ChatSessionAnalytics{ - UserID: userID, - ConversationID: conversationID, - SessionID: sessionID, - MessageCount: 0, - StartedAt: time.Now(), - } - - _, err := s.collection("chat_sessions").InsertOne(ctx, session) - if err != nil { - log.Printf("⚠️ [ANALYTICS] Failed to track session start: %v", err) - return err - } - - return nil -} - -// TrackChatSessionEnd records the end of a chat session -func (s *AnalyticsService) TrackChatSessionEnd(ctx context.Context, sessionID string, messageCount int) error { - if s.mongoDB == nil { - return nil - } - - now := time.Now() - - // Find the session - var session ChatSessionAnalytics - err := s.collection("chat_sessions").FindOne(ctx, bson.M{"sessionId": sessionID}).Decode(&session) - if err != nil { - if err == mongo.ErrNoDocuments { - // Session wasn't tracked (maybe analytics added after session started) - return nil - } - return err - } - - durationMs := now.Sub(session.StartedAt).Milliseconds() - - // Update the session - _, err = s.collection("chat_sessions").UpdateOne( - ctx, - bson.M{"sessionId": sessionID}, - bson.M{ - "$set": bson.M{ - "endedAt": now, - "messageCount": messageCount, - "durationMs": durationMs, - }, - }, - ) - - if err != nil { - log.Printf("⚠️ [ANALYTICS] Failed to track session end: %v", err) - } - - return err -} - -// UpdateChatSessionModel updates the model used in a session -func (s *AnalyticsService) UpdateChatSessionModel(ctx context.Context, sessionID, modelID string, disabledTools bool) error { - if s.mongoDB == nil { - return nil - } - - _, err := s.collection("chat_sessions").UpdateOne( - ctx, - bson.M{"sessionId": sessionID}, - bson.M{ - "$set": bson.M{ - "modelId": modelID, - "disabledTools": disabledTools, - }, - }, - ) - - return err -} - -// TrackAgentUsage records when an agent is used -func (s *AnalyticsService) TrackAgentUsage(ctx context.Context, userID, agentID string, executionID primitive.ObjectID, triggerType string) error { - if s.mongoDB == nil { - return nil - } - - usage := &AgentUsageAnalytics{ - UserID: userID, - AgentID: agentID, - ExecutionID: executionID, - TriggerType: triggerType, - ExecutedAt: time.Now(), - } - - _, err := s.collection("agent_usage").InsertOne(ctx, usage) - if err != nil { - log.Printf("⚠️ [ANALYTICS] Failed to track agent usage: %v", err) - } - - return err -} - -// collection returns a MongoDB collection -func (s *AnalyticsService) collection(name string) *mongo.Collection { - return s.mongoDB.Database().Collection(name) -} - -// loadProvidersConfig reads and parses the providers.json file -func (s *AnalyticsService) loadProvidersConfig() (*models.ProvidersConfig, error) { - // Get the path to providers.json (relative to backend root) - providersPath := filepath.Join("providers.json") - - // Read the file - data, err := os.ReadFile(providersPath) - if err != nil { - log.Printf("⚠️ [ANALYTICS] Failed to read providers.json: %v", err) - return nil, err - } - - // Parse JSON - var config models.ProvidersConfig - if err := json.Unmarshal(data, &config); err != nil { - log.Printf("⚠️ [ANALYTICS] Failed to parse providers.json: %v", err) - return nil, err - } - - return &config, nil -} - -// GetOverviewStats returns system overview statistics -func (s *AnalyticsService) GetOverviewStats(ctx context.Context) (map[string]interface{}, error) { - if s.mongoDB == nil { - return map[string]interface{}{ - "total_users": 0, - "active_chats": 0, - "total_messages": 0, - "api_calls_today": 0, - "active_providers": 0, - "total_models": 0, - "total_agents": 0, - "agent_executions": 0, - "agents_run_today": 0, - }, nil - } - - // Count total chat sessions (approximation of active chats) - activeChats, _ := s.collection("chat_sessions").CountDocuments(ctx, bson.M{"endedAt": bson.M{"$exists": false}}) - - // Count total messages from all sessions - pipeline := mongo.Pipeline{ - {{Key: "$group", Value: bson.M{"_id": nil, "totalMessages": bson.M{"$sum": "$messageCount"}}}}, - } - cursor, err := s.collection("chat_sessions").Aggregate(ctx, pipeline) - var totalMessages int64 - if err == nil { - var results []bson.M - if err := cursor.All(ctx, &results); err == nil && len(results) > 0 { - if count, ok := results[0]["totalMessages"].(int32); ok { - totalMessages = int64(count) - } else if count, ok := results[0]["totalMessages"].(int64); ok { - totalMessages = count - } - } - } - - // Count unique users - uniqueUsers, _ := s.collection("chat_sessions").Distinct(ctx, "userId", bson.M{}) - - // Count API calls today - today := time.Now().Truncate(24 * time.Hour) - apiCallsToday, _ := s.collection("chat_sessions").CountDocuments(ctx, bson.M{"startedAt": bson.M{"$gte": today}}) - - // Count total models from providers.json - totalModels := 0 - providersConfig, err := s.loadProvidersConfig() - if err == nil { - for _, provider := range providersConfig.Providers { - // Only count enabled providers - if provider.Enabled { - // Count models from ModelAliases map - totalModels += len(provider.ModelAliases) - } - } - } - - // Count active providers (providers that have been used in the last 30 days) - activeProviders := 0 - thirtyDaysAgo := time.Now().Add(-30 * 24 * time.Hour) - - // Get distinct model IDs used in the last 30 days - usedModels, err := s.collection("chat_sessions").Distinct(ctx, "modelId", bson.M{ - "modelId": bson.M{"$exists": true, "$ne": ""}, - "startedAt": bson.M{"$gte": thirtyDaysAgo}, - }) - - if err == nil && providersConfig != nil { - // Create a set of used model IDs for faster lookup - usedModelSet := make(map[string]bool) - for _, modelID := range usedModels { - if modelStr, ok := modelID.(string); ok { - usedModelSet[modelStr] = true - } - } - - // Check each provider to see if any of their models were used - for _, provider := range providersConfig.Providers { - if !provider.Enabled { - continue - } - - // Check if any model from this provider was used - for modelAlias := range provider.ModelAliases { - if usedModelSet[modelAlias] { - activeProviders++ - break // Count provider only once - } - } - } - } - - // Count agent metrics - totalAgentExecutions, _ := s.collection("agent_usage").CountDocuments(ctx, bson.M{}) - agentsRunToday, _ := s.collection("agent_usage").CountDocuments(ctx, bson.M{"executedAt": bson.M{"$gte": today}}) - - // Count unique agents - uniqueAgents, _ := s.collection("agent_usage").Distinct(ctx, "agentId", bson.M{}) - totalAgents := len(uniqueAgents) - - return map[string]interface{}{ - "total_users": len(uniqueUsers), - "active_chats": activeChats, - "total_messages": totalMessages, - "api_calls_today": apiCallsToday, - "active_providers": activeProviders, - "total_models": totalModels, - "total_agents": totalAgents, - "agent_executions": totalAgentExecutions, - "agents_run_today": agentsRunToday, - }, nil -} - -// GetProviderAnalytics returns usage analytics per provider -func (s *AnalyticsService) GetProviderAnalytics(ctx context.Context) ([]map[string]interface{}, error) { - if s.mongoDB == nil { - return []map[string]interface{}{}, nil - } - - // Group by model ID and aggregate usage - pipeline := mongo.Pipeline{ - {{Key: "$match", Value: bson.M{"modelId": bson.M{"$exists": true, "$ne": ""}}}}, - {{Key: "$group", Value: bson.M{ - "_id": "$modelId", - "total_requests": bson.M{"$sum": 1}, - "last_used_at": bson.M{"$max": "$startedAt"}, - }}}, - {{Key: "$sort", Value: bson.M{"total_requests": -1}}}, - } - - cursor, err := s.collection("chat_sessions").Aggregate(ctx, pipeline) - if err != nil { - return []map[string]interface{}{}, err - } - - var results []bson.M - if err := cursor.All(ctx, &results); err != nil { - return []map[string]interface{}{}, err - } - - analytics := make([]map[string]interface{}, 0, len(results)) - for _, result := range results { - analytics = append(analytics, map[string]interface{}{ - "provider_id": result["_id"], - "provider_name": result["_id"], // TODO: Resolve from providers.json - "total_requests": result["total_requests"], - "total_tokens": 0, // TODO: Track tokens - "estimated_cost": nil, - "active_models": []string{}, - "last_used_at": result["last_used_at"], - }) - } - - return analytics, nil -} - -// GetChatAnalytics returns chat usage statistics -func (s *AnalyticsService) GetChatAnalytics(ctx context.Context) (map[string]interface{}, error) { - if s.mongoDB == nil { - return map[string]interface{}{ - "total_chats": 0, - "active_chats": 0, - "total_messages": 0, - "avg_messages_per_chat": 0.0, - "chats_created_today": 0, - "messages_sent_today": 0, - "time_series": []map[string]interface{}{}, - }, nil - } - - totalChats, _ := s.collection("chat_sessions").CountDocuments(ctx, bson.M{}) - activeChats, _ := s.collection("chat_sessions").CountDocuments(ctx, bson.M{"endedAt": bson.M{"$exists": false}}) - - // Total messages - pipeline := mongo.Pipeline{ - {{Key: "$group", Value: bson.M{"_id": nil, "totalMessages": bson.M{"$sum": "$messageCount"}}}}, - } - cursor, _ := s.collection("chat_sessions").Aggregate(ctx, pipeline) - var totalMessages int64 - var results []bson.M - if cursor.All(ctx, &results) == nil && len(results) > 0 { - if count, ok := results[0]["totalMessages"].(int32); ok { - totalMessages = int64(count) - } else if count, ok := results[0]["totalMessages"].(int64); ok { - totalMessages = count - } - } - - avgMessages := 0.0 - if totalChats > 0 { - avgMessages = float64(totalMessages) / float64(totalChats) - } - - // Today's stats - today := time.Now().Truncate(24 * time.Hour) - chatsToday, _ := s.collection("chat_sessions").CountDocuments(ctx, bson.M{"startedAt": bson.M{"$gte": today}}) - - // Get time series data for the last 30 days - timeSeries, _ := s.getTimeSeriesData(ctx, 30) - - return map[string]interface{}{ - "total_chats": totalChats, - "active_chats": activeChats, - "total_messages": totalMessages, - "avg_messages_per_chat": avgMessages, - "chats_created_today": chatsToday, - "messages_sent_today": 0, // TODO: Track per-day messages - "time_series": timeSeries, - }, nil -} - -// getTimeSeriesData returns daily statistics for the specified number of days -func (s *AnalyticsService) getTimeSeriesData(ctx context.Context, days int) ([]map[string]interface{}, error) { - if s.mongoDB == nil { - return []map[string]interface{}{}, nil - } - - // Calculate start date - startDate := time.Now().Add(-time.Duration(days) * 24 * time.Hour).Truncate(24 * time.Hour) - - // Aggregate chats and messages by day - pipeline := mongo.Pipeline{ - {{Key: "$match", Value: bson.M{"startedAt": bson.M{"$gte": startDate}}}}, - {{Key: "$group", Value: bson.M{ - "_id": bson.M{ - "$dateToString": bson.M{ - "format": "%Y-%m-%d", - "date": "$startedAt", - }, - }, - "chat_count": bson.M{"$sum": 1}, - "message_count": bson.M{"$sum": "$messageCount"}, - "unique_users": bson.M{"$addToSet": "$userId"}, - }}}, - {{Key: "$sort", Value: bson.M{"_id": 1}}}, - } - - cursor, err := s.collection("chat_sessions").Aggregate(ctx, pipeline) - if err != nil { - log.Printf("⚠️ [ANALYTICS] Failed to aggregate time series: %v", err) - return []map[string]interface{}{}, err - } - - var results []bson.M - if err := cursor.All(ctx, &results); err != nil { - return []map[string]interface{}{}, err - } - - // Aggregate agent executions by day - agentPipeline := mongo.Pipeline{ - {{Key: "$match", Value: bson.M{"executedAt": bson.M{"$gte": startDate}}}}, - {{Key: "$group", Value: bson.M{ - "_id": bson.M{ - "$dateToString": bson.M{ - "format": "%Y-%m-%d", - "date": "$executedAt", - }, - }, - "agent_count": bson.M{"$sum": 1}, - }}}, - {{Key: "$sort", Value: bson.M{"_id": 1}}}, - } - - agentCursor, err := s.collection("agent_usage").Aggregate(ctx, agentPipeline) - var agentResults []bson.M - if err == nil { - agentCursor.All(ctx, &agentResults) - } - - // Create a map of agent counts by date for easy lookup - agentCountByDate := make(map[string]int64) - for _, result := range agentResults { - date, _ := result["_id"].(string) - count := int64(0) - if c, ok := result["agent_count"].(int32); ok { - count = int64(c) - } else if c, ok := result["agent_count"].(int64); ok { - count = c - } - agentCountByDate[date] = count - } - - // Convert to response format with agent data - timeSeries := make([]map[string]interface{}, 0, len(results)) - for _, result := range results { - date, _ := result["_id"].(string) - chatCount := int64(0) - messageCount := int64(0) - uniqueUsers := 0 - - if count, ok := result["chat_count"].(int32); ok { - chatCount = int64(count) - } else if count, ok := result["chat_count"].(int64); ok { - chatCount = count - } - - if count, ok := result["message_count"].(int32); ok { - messageCount = int64(count) - } else if count, ok := result["message_count"].(int64); ok { - messageCount = count - } - - if users, ok := result["unique_users"].(primitive.A); ok { - uniqueUsers = len(users) - } - - // Get agent count for this date - agentCount := agentCountByDate[date] - - timeSeries = append(timeSeries, map[string]interface{}{ - "date": date, - "chat_count": chatCount, - "message_count": messageCount, - "user_count": uniqueUsers, - "agent_count": agentCount, - }) - } - - // Fill in missing dates with zeros - filledSeries := s.fillMissingDates(timeSeries, startDate, days) - - return filledSeries, nil -} - -// fillMissingDates ensures all dates in the range have entries (with zeros if no data) -func (s *AnalyticsService) fillMissingDates(data []map[string]interface{}, startDate time.Time, days int) []map[string]interface{} { - // Create a map of existing dates - dataMap := make(map[string]map[string]interface{}) - for _, entry := range data { - if date, ok := entry["date"].(string); ok { - dataMap[date] = entry - } - } - - // Fill all dates - result := make([]map[string]interface{}, 0, days) - for i := 0; i < days; i++ { - date := startDate.Add(time.Duration(i) * 24 * time.Hour) - dateStr := date.Format("2006-01-02") - - if entry, exists := dataMap[dateStr]; exists { - result = append(result, entry) - } else { - result = append(result, map[string]interface{}{ - "date": dateStr, - "chat_count": 0, - "message_count": 0, - "user_count": 0, - "agent_count": 0, - }) - } - } - - return result -} - -// EnsureIndexes creates indexes for analytics collections -func (s *AnalyticsService) EnsureIndexes(ctx context.Context) error { - if s.mongoDB == nil { - return nil - } - - // Chat sessions indexes - _, err := s.collection("chat_sessions").Indexes().CreateMany(ctx, []mongo.IndexModel{ - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "startedAt", Value: -1}}}, - {Keys: bson.D{{Key: "sessionId", Value: 1}}}, - {Keys: bson.D{{Key: "startedAt", Value: -1}}}, - }) - if err != nil { - return err - } - - // Agent usage indexes - _, err = s.collection("agent_usage").Indexes().CreateMany(ctx, []mongo.IndexModel{ - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "executedAt", Value: -1}}}, - {Keys: bson.D{{Key: "agentId", Value: 1}, {Key: "executedAt", Value: -1}}}, - {Keys: bson.D{{Key: "executedAt", Value: -1}}}, - }) - - log.Println("✅ [ANALYTICS] Indexes created") - return err -} - -// MigrateChatSessionTimestamps fixes existing chat sessions that don't have proper startedAt timestamps -// Uses the MongoDB ObjectID creation time as the startedAt value -func (s *AnalyticsService) MigrateChatSessionTimestamps(ctx context.Context) (int, error) { - if s.mongoDB == nil { - return 0, nil - } - - log.Println("🔄 [ANALYTICS MIGRATION] Starting chat session timestamp migration...") - - // Find all sessions without startedAt or with zero time - zeroTime := time.Time{} - cursor, err := s.collection("chat_sessions").Find(ctx, bson.M{ - "$or": []bson.M{ - {"startedAt": bson.M{"$exists": false}}, - {"startedAt": zeroTime}, - {"startedAt": nil}, - }, - }) - if err != nil { - log.Printf("❌ [ANALYTICS MIGRATION] Failed to query sessions: %v", err) - return 0, err - } - defer cursor.Close(ctx) - - updatedCount := 0 - for cursor.Next(ctx) { - var session ChatSessionAnalytics - if err := cursor.Decode(&session); err != nil { - log.Printf("⚠️ [ANALYTICS MIGRATION] Failed to decode session: %v", err) - continue - } - - // Extract timestamp from MongoDB ObjectID - // ObjectID first 4 bytes are Unix timestamp - createdAt := session.ID.Timestamp() - - // Update the session with the extracted timestamp - _, err := s.collection("chat_sessions").UpdateOne( - ctx, - bson.M{"_id": session.ID}, - bson.M{"$set": bson.M{"startedAt": createdAt}}, - ) - if err != nil { - log.Printf("⚠️ [ANALYTICS MIGRATION] Failed to update session %s: %v", session.ID.Hex(), err) - continue - } - - updatedCount++ - } - - if err := cursor.Err(); err != nil { - log.Printf("❌ [ANALYTICS MIGRATION] Cursor error: %v", err) - return updatedCount, err - } - - log.Printf("✅ [ANALYTICS MIGRATION] Successfully migrated %d chat sessions with proper timestamps", updatedCount) - return updatedCount, nil -} - - -// GetAgentAnalytics returns comprehensive agent activity analytics -func (s *AnalyticsService) GetAgentAnalytics(ctx context.Context) (map[string]interface{}, error) { - if s.mongoDB == nil { - return map[string]interface{}{ - "total_agents": 0, - "deployed_agents": 0, - "total_executions": 0, - "active_schedules": 0, - "executions_today": 0, - "time_series": []map[string]interface{}{}, - }, nil - } - - // Count total agents - totalAgents, _ := s.collection("agents").CountDocuments(ctx, bson.M{}) - - // Count deployed agents (status = "deployed") - deployedAgents, _ := s.collection("agents").CountDocuments(ctx, bson.M{"status": "deployed"}) - - // Count total agent executions - totalExecutions, _ := s.collection("agent_usage").CountDocuments(ctx, bson.M{}) - - // Count active schedules - activeSchedules, _ := s.collection("schedules").CountDocuments(ctx, bson.M{"enabled": true}) - - // Count executions today - today := time.Now().Truncate(24 * time.Hour) - executionsToday, _ := s.collection("agent_usage").CountDocuments(ctx, bson.M{"executedAt": bson.M{"$gte": today}}) - - // Get time series data for the last 30 days - timeSeries, _ := s.getAgentTimeSeriesData(ctx, 30) - - return map[string]interface{}{ - "total_agents": totalAgents, - "deployed_agents": deployedAgents, - "total_executions": totalExecutions, - "active_schedules": activeSchedules, - "executions_today": executionsToday, - "time_series": timeSeries, - }, nil -} - -// getAgentTimeSeriesData returns daily agent activity statistics -func (s *AnalyticsService) getAgentTimeSeriesData(ctx context.Context, days int) ([]map[string]interface{}, error) { - if s.mongoDB == nil { - return []map[string]interface{}{}, nil - } - - startDate := time.Now().Add(-time.Duration(days) * 24 * time.Hour).Truncate(24 * time.Hour) - - // Aggregate agents created by day - agentsCreatedPipeline := mongo.Pipeline{ - {{Key: "$match", Value: bson.M{"createdAt": bson.M{"$gte": startDate}}}}, - {{Key: "$group", Value: bson.M{ - "_id": bson.M{ - "$dateToString": bson.M{ - "format": "%Y-%m-%d", - "date": "$createdAt", - }, - }, - "agents_created": bson.M{"$sum": 1}, - }}}, - {{Key: "$sort", Value: bson.M{"_id": 1}}}, - } - - agentsCreatedCursor, err := s.collection("agents").Aggregate(ctx, agentsCreatedPipeline) - var agentsCreatedResults []bson.M - if err == nil { - agentsCreatedCursor.All(ctx, &agentsCreatedResults) - } - - // Aggregate agents deployed by day (when updatedAt changed to deployed status) - agentsDeployedPipeline := mongo.Pipeline{ - {{Key: "$match", Value: bson.M{ - "status": "deployed", - "updatedAt": bson.M{"$gte": startDate}, - }}}, - {{Key: "$group", Value: bson.M{ - "_id": bson.M{ - "$dateToString": bson.M{ - "format": "%Y-%m-%d", - "date": "$updatedAt", - }, - }, - "agents_deployed": bson.M{"$sum": 1}, - }}}, - {{Key: "$sort", Value: bson.M{"_id": 1}}}, - } - - agentsDeployedCursor, err := s.collection("agents").Aggregate(ctx, agentsDeployedPipeline) - var agentsDeployedResults []bson.M - if err == nil { - agentsDeployedCursor.All(ctx, &agentsDeployedResults) - } - - // Aggregate agent executions by day - agentRunsPipeline := mongo.Pipeline{ - {{Key: "$match", Value: bson.M{"executedAt": bson.M{"$gte": startDate}}}}, - {{Key: "$group", Value: bson.M{ - "_id": bson.M{ - "$dateToString": bson.M{ - "format": "%Y-%m-%d", - "date": "$executedAt", - }, - }, - "agent_runs": bson.M{"$sum": 1}, - }}}, - {{Key: "$sort", Value: bson.M{"_id": 1}}}, - } - - agentRunsCursor, err := s.collection("agent_usage").Aggregate(ctx, agentRunsPipeline) - var agentRunsResults []bson.M - if err == nil { - agentRunsCursor.All(ctx, &agentRunsResults) - } - - // Aggregate schedules created by day - schedulesCreatedPipeline := mongo.Pipeline{ - {{Key: "$match", Value: bson.M{"createdAt": bson.M{"$gte": startDate}}}}, - {{Key: "$group", Value: bson.M{ - "_id": bson.M{ - "$dateToString": bson.M{ - "format": "%Y-%m-%d", - "date": "$createdAt", - }, - }, - "schedules_created": bson.M{"$sum": 1}, - }}}, - {{Key: "$sort", Value: bson.M{"_id": 1}}}, - } - - schedulesCreatedCursor, err := s.collection("schedules").Aggregate(ctx, schedulesCreatedPipeline) - var schedulesCreatedResults []bson.M - if err == nil { - schedulesCreatedCursor.All(ctx, &schedulesCreatedResults) - } - - // Create maps for easy lookup - agentsCreatedByDate := make(map[string]int64) - agentsDeployedByDate := make(map[string]int64) - agentRunsByDate := make(map[string]int64) - schedulesCreatedByDate := make(map[string]int64) - - for _, result := range agentsCreatedResults { - date, _ := result["_id"].(string) - count := extractInt64(result, "agents_created") - agentsCreatedByDate[date] = count - } - - for _, result := range agentsDeployedResults { - date, _ := result["_id"].(string) - count := extractInt64(result, "agents_deployed") - agentsDeployedByDate[date] = count - } - - for _, result := range agentRunsResults { - date, _ := result["_id"].(string) - count := extractInt64(result, "agent_runs") - agentRunsByDate[date] = count - } - - for _, result := range schedulesCreatedResults { - date, _ := result["_id"].(string) - count := extractInt64(result, "schedules_created") - schedulesCreatedByDate[date] = count - } - - // Fill all dates with data - timeSeries := make([]map[string]interface{}, 0, days) - for i := 0; i < days; i++ { - date := startDate.Add(time.Duration(i) * 24 * time.Hour) - dateStr := date.Format("2006-01-02") - - timeSeries = append(timeSeries, map[string]interface{}{ - "date": dateStr, - "agents_created": agentsCreatedByDate[dateStr], - "agents_deployed": agentsDeployedByDate[dateStr], - "agent_runs": agentRunsByDate[dateStr], - "schedules_created": schedulesCreatedByDate[dateStr], - }) - } - - return timeSeries, nil -} - -// extractInt64 safely extracts an int64 value from a bson.M result -func extractInt64(result bson.M, key string) int64 { - if count, ok := result[key].(int32); ok { - return int64(count) - } else if count, ok := result[key].(int64); ok { - return count - } - return 0 -} - -// UserListItemGDPR represents a GDPR-compliant user list item -type UserListItemGDPR struct { - UserID string `json:"user_id"` - EmailDomain string `json:"email_domain,omitempty"` - Tier string `json:"tier"` - CreatedAt time.Time `json:"created_at"` - LastActive *time.Time `json:"last_active,omitempty"` - TotalChats int64 `json:"total_chats"` - TotalMessages int64 `json:"total_messages"` - TotalAgentRuns int64 `json:"total_agent_runs"` - HasOverrides bool `json:"has_overrides"` -} - -// GetUserListGDPR returns a GDPR-compliant paginated user list -// Only includes aggregated analytics, no PII except anonymized user IDs and email domains -func (s *AnalyticsService) GetUserListGDPR(ctx context.Context, page, pageSize int, tierFilter, searchFilter string) ([]UserListItemGDPR, int64, error) { - if s.mongoDB == nil { - return []UserListItemGDPR{}, 0, nil - } - - // Get unique users from chat sessions with their activity - pipeline := mongo.Pipeline{ - {{Key: "$group", Value: bson.M{ - "_id": "$userId", - "total_chats": bson.M{"$sum": 1}, - "total_messages": bson.M{"$sum": "$messageCount"}, - "last_active": bson.M{"$max": "$startedAt"}, - "first_seen": bson.M{"$min": "$startedAt"}, - }}}, - } - - cursor, err := s.collection("chat_sessions").Aggregate(ctx, pipeline) - if err != nil { - return nil, 0, err - } - - var sessionStats []bson.M - if err := cursor.All(ctx, &sessionStats); err != nil { - return nil, 0, err - } - - // Get agent usage counts per user - agentPipeline := mongo.Pipeline{ - {{Key: "$group", Value: bson.M{ - "_id": "$userId", - "total_agent_runs": bson.M{"$sum": 1}, - }}}, - } - - agentCursor, err := s.collection("agent_usage").Aggregate(ctx, agentPipeline) - var agentStats []bson.M - agentCountByUser := make(map[string]int64) - if err == nil { - agentCursor.All(ctx, &agentStats) - for _, stat := range agentStats { - userID, _ := stat["_id"].(string) - count := extractInt64(stat, "total_agent_runs") - agentCountByUser[userID] = count - } - } - - // Build user list - users := make([]UserListItemGDPR, 0, len(sessionStats)) - for _, stat := range sessionStats { - userID, ok := stat["_id"].(string) - if !ok || userID == "" { - continue - } - - // Extract email domain from user ID if it's an email format - emailDomain := extractEmailDomain(userID) - - totalChats := extractInt64(stat, "total_chats") - totalMessages := extractInt64(stat, "total_messages") - totalAgentRuns := agentCountByUser[userID] - - var lastActive *time.Time - if lastActiveVal, ok := stat["last_active"].(primitive.DateTime); ok { - t := lastActiveVal.Time() - lastActive = &t - } - - var createdAt time.Time - if firstSeenVal, ok := stat["first_seen"].(primitive.DateTime); ok { - createdAt = firstSeenVal.Time() - } - - user := UserListItemGDPR{ - UserID: anonymizeUserID(userID), // Anonymize user ID - EmailDomain: emailDomain, - Tier: "free", // Default, would come from user service in production - CreatedAt: createdAt, - LastActive: lastActive, - TotalChats: totalChats, - TotalMessages: totalMessages, - TotalAgentRuns: totalAgentRuns, - HasOverrides: false, // Would check user service for overrides - } - - users = append(users, user) - } - - // Sort by last active (most recent first) - // Note: For production, this should be done in the database query - // Simple in-memory sort for now - totalCount := int64(len(users)) - - // Pagination - start := (page - 1) * pageSize - end := start + pageSize - if start >= len(users) { - return []UserListItemGDPR{}, totalCount, nil - } - if end > len(users) { - end = len(users) - } - - return users[start:end], totalCount, nil -} - -// extractEmailDomain extracts the domain from an email address -func extractEmailDomain(email string) string { - parts := splitString(email, "@") - if len(parts) == 2 { - return "@" + parts[1] - } - return "" -} - -// splitString is a simple string split helper -func splitString(s, sep string) []string { - result := []string{} - current := "" - sepLen := len(sep) - - for i := 0; i < len(s); i++ { - if i+sepLen <= len(s) && s[i:i+sepLen] == sep { - result = append(result, current) - current = "" - i += sepLen - 1 - } else { - current += string(s[i]) - } - } - result = append(result, current) - return result -} - -// anonymizeUserID creates a privacy-safe representation of a user ID -func anonymizeUserID(userID string) string { - // For emails, show first 3 chars + *** + domain - parts := splitString(userID, "@") - if len(parts) == 2 { - prefix := "***" - if len(parts[0]) > 3 { - prefix = parts[0][:3] + "***" - } - return prefix + "@" + parts[1] - } - // For non-email IDs, just show first 8 chars - if len(userID) > 8 { - return userID[:8] + "..." - } - return userID -} diff --git a/backend/internal/services/apikey_service.go b/backend/internal/services/apikey_service.go deleted file mode 100644 index 77e349dc..00000000 --- a/backend/internal/services/apikey_service.go +++ /dev/null @@ -1,349 +0,0 @@ -package services - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "context" - "crypto/rand" - "encoding/hex" - "fmt" - "log" - "time" - - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/bson/primitive" - "go.mongodb.org/mongo-driver/mongo" - "go.mongodb.org/mongo-driver/mongo/options" - "golang.org/x/crypto/bcrypt" -) - -const ( - // APIKeyPrefix is the prefix for all API keys - APIKeyPrefix = "clv_" - // APIKeyLength is the length of the random part of the key (32 bytes = 64 hex chars) - APIKeyLength = 32 - // APIKeyPrefixLength is how many chars to show as prefix (including "clv_") - APIKeyPrefixLength = 12 -) - -// APIKeyService manages API keys -type APIKeyService struct { - mongoDB *database.MongoDB - tierService *TierService -} - -// NewAPIKeyService creates a new API key service -func NewAPIKeyService(mongoDB *database.MongoDB, tierService *TierService) *APIKeyService { - return &APIKeyService{ - mongoDB: mongoDB, - tierService: tierService, - } -} - -// collection returns the api_keys collection -func (s *APIKeyService) collection() *mongo.Collection { - return s.mongoDB.Database().Collection("api_keys") -} - -// GenerateKey generates a new API key -func (s *APIKeyService) GenerateKey() (string, error) { - bytes := make([]byte, APIKeyLength) - if _, err := rand.Read(bytes); err != nil { - return "", fmt.Errorf("failed to generate random bytes: %w", err) - } - return APIKeyPrefix + hex.EncodeToString(bytes), nil -} - -// HashKey hashes an API key for storage -func (s *APIKeyService) HashKey(key string) (string, error) { - hash, err := bcrypt.GenerateFromPassword([]byte(key), bcrypt.DefaultCost) - if err != nil { - return "", fmt.Errorf("failed to hash key: %w", err) - } - return string(hash), nil -} - -// VerifyKey verifies an API key against a hash -func (s *APIKeyService) VerifyKey(key, hash string) bool { - err := bcrypt.CompareHashAndPassword([]byte(hash), []byte(key)) - return err == nil -} - -// Create creates a new API key -func (s *APIKeyService) Create(ctx context.Context, userID string, req *models.CreateAPIKeyRequest) (*models.CreateAPIKeyResponse, error) { - // Check API key limit - if s.tierService != nil { - count, err := s.CountByUser(ctx, userID) - if err != nil { - return nil, err - } - if !s.tierService.CheckAPIKeyLimit(ctx, userID, count) { - limits := s.tierService.GetLimits(ctx, userID) - return nil, fmt.Errorf("API key limit reached (%d/%d)", count, limits.MaxAPIKeys) - } - } - - // Validate scopes - for _, scope := range req.Scopes { - if !models.IsValidScope(scope) { - return nil, fmt.Errorf("invalid scope: %s", scope) - } - } - - // Generate key - key, err := s.GenerateKey() - if err != nil { - return nil, err - } - - // Hash key for storage - hash, err := s.HashKey(key) - if err != nil { - return nil, err - } - - // Calculate expiration if specified - var expiresAt *time.Time - if req.ExpiresIn > 0 { - exp := time.Now().Add(time.Duration(req.ExpiresIn) * 24 * time.Hour) - expiresAt = &exp - } - - now := time.Now() - apiKey := &models.APIKey{ - UserID: userID, - KeyPrefix: key[:APIKeyPrefixLength], - KeyHash: hash, - PlainKey: key, // TEMPORARY: Store plain key for early platform phase - Name: req.Name, - Description: req.Description, - Scopes: req.Scopes, - RateLimit: req.RateLimit, - ExpiresAt: expiresAt, - CreatedAt: now, - UpdatedAt: now, - } - - result, err := s.collection().InsertOne(ctx, apiKey) - if err != nil { - return nil, fmt.Errorf("failed to create API key: %w", err) - } - - apiKey.ID = result.InsertedID.(primitive.ObjectID) - - log.Printf("🔑 [APIKEY] Created API key %s for user %s (prefix: %s)", - apiKey.ID.Hex(), userID, apiKey.KeyPrefix) - - return &models.CreateAPIKeyResponse{ - ID: apiKey.ID.Hex(), - Key: key, // Full key - only returned once! - KeyPrefix: apiKey.KeyPrefix, - Name: apiKey.Name, - Scopes: apiKey.Scopes, - ExpiresAt: expiresAt, - CreatedAt: now, - }, nil -} - -// ValidateKey validates an API key and returns the key record -func (s *APIKeyService) ValidateKey(ctx context.Context, key string) (*models.APIKey, error) { - if len(key) < APIKeyPrefixLength { - return nil, fmt.Errorf("invalid API key format") - } - - // Extract prefix for lookup - prefix := key[:APIKeyPrefixLength] - - // Find by prefix (there could be multiple with same prefix, but unlikely) - cursor, err := s.collection().Find(ctx, bson.M{ - "keyPrefix": prefix, - "revokedAt": bson.M{"$exists": false}, // Not revoked - }) - if err != nil { - return nil, fmt.Errorf("failed to lookup API key: %w", err) - } - defer cursor.Close(ctx) - - // Check each matching key (usually just one) - for cursor.Next(ctx) { - var apiKey models.APIKey - if err := cursor.Decode(&apiKey); err != nil { - continue - } - - // Verify the hash - if s.VerifyKey(key, apiKey.KeyHash) { - // Check expiration - if apiKey.IsExpired() { - return nil, fmt.Errorf("API key has expired") - } - - // Update last used - go s.updateLastUsed(context.Background(), apiKey.ID) - - return &apiKey, nil - } - } - - return nil, fmt.Errorf("invalid API key") -} - -// updateLastUsed updates the last used timestamp -func (s *APIKeyService) updateLastUsed(ctx context.Context, keyID primitive.ObjectID) { - _, err := s.collection().UpdateByID(ctx, keyID, bson.M{ - "$set": bson.M{ - "lastUsedAt": time.Now(), - }, - }) - if err != nil { - log.Printf("⚠️ [APIKEY] Failed to update last used: %v", err) - } -} - -// ListByUser returns all API keys for a user (without hashes) -func (s *APIKeyService) ListByUser(ctx context.Context, userID string) ([]*models.APIKeyListItem, error) { - cursor, err := s.collection().Find(ctx, bson.M{ - "userId": userID, - }, options.Find().SetSort(bson.D{{Key: "createdAt", Value: -1}})) - if err != nil { - return nil, fmt.Errorf("failed to list API keys: %w", err) - } - defer cursor.Close(ctx) - - var keys []*models.APIKeyListItem - for cursor.Next(ctx) { - var key models.APIKey - if err := cursor.Decode(&key); err != nil { - continue - } - keys = append(keys, key.ToListItem()) - } - - return keys, nil -} - -// GetByID retrieves an API key by ID -func (s *APIKeyService) GetByID(ctx context.Context, keyID primitive.ObjectID) (*models.APIKey, error) { - var key models.APIKey - err := s.collection().FindOne(ctx, bson.M{"_id": keyID}).Decode(&key) - if err != nil { - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("API key not found") - } - return nil, fmt.Errorf("failed to get API key: %w", err) - } - return &key, nil -} - -// GetByIDAndUser retrieves an API key by ID ensuring user ownership -func (s *APIKeyService) GetByIDAndUser(ctx context.Context, keyID primitive.ObjectID, userID string) (*models.APIKey, error) { - var key models.APIKey - err := s.collection().FindOne(ctx, bson.M{ - "_id": keyID, - "userId": userID, - }).Decode(&key) - if err != nil { - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("API key not found") - } - return nil, fmt.Errorf("failed to get API key: %w", err) - } - return &key, nil -} - -// Revoke revokes an API key (soft delete) -func (s *APIKeyService) Revoke(ctx context.Context, keyID primitive.ObjectID, userID string) error { - result, err := s.collection().UpdateOne(ctx, bson.M{ - "_id": keyID, - "userId": userID, - }, bson.M{ - "$set": bson.M{ - "revokedAt": time.Now(), - "updatedAt": time.Now(), - }, - }) - if err != nil { - return fmt.Errorf("failed to revoke API key: %w", err) - } - - if result.MatchedCount == 0 { - return fmt.Errorf("API key not found") - } - - log.Printf("🔒 [APIKEY] Revoked API key %s for user %s", keyID.Hex(), userID) - return nil -} - -// Delete permanently deletes an API key -func (s *APIKeyService) Delete(ctx context.Context, keyID primitive.ObjectID, userID string) error { - result, err := s.collection().DeleteOne(ctx, bson.M{ - "_id": keyID, - "userId": userID, - }) - if err != nil { - return fmt.Errorf("failed to delete API key: %w", err) - } - - if result.DeletedCount == 0 { - return fmt.Errorf("API key not found") - } - - log.Printf("🗑️ [APIKEY] Deleted API key %s for user %s", keyID.Hex(), userID) - return nil -} - -// DeleteAllByUser deletes all API keys for a user (GDPR compliance) -func (s *APIKeyService) DeleteAllByUser(ctx context.Context, userID string) (int64, error) { - if userID == "" { - return 0, fmt.Errorf("user ID is required") - } - - result, err := s.collection().DeleteMany(ctx, bson.M{"userId": userID}) - if err != nil { - return 0, fmt.Errorf("failed to delete user API keys: %w", err) - } - - log.Printf("🗑️ [GDPR] Deleted %d API keys for user %s", result.DeletedCount, userID) - return result.DeletedCount, nil -} - -// CountByUser counts API keys for a user (non-revoked) -func (s *APIKeyService) CountByUser(ctx context.Context, userID string) (int64, error) { - count, err := s.collection().CountDocuments(ctx, bson.M{ - "userId": userID, - "revokedAt": bson.M{"$exists": false}, - }) - if err != nil { - return 0, fmt.Errorf("failed to count API keys: %w", err) - } - return count, nil -} - -// EnsureIndexes creates the necessary indexes for the api_keys collection -func (s *APIKeyService) EnsureIndexes(ctx context.Context) error { - indexes := []mongo.IndexModel{ - // User ID for listing - { - Keys: bson.D{{Key: "userId", Value: 1}}, - }, - // Key prefix for lookup during validation - { - Keys: bson.D{{Key: "keyPrefix", Value: 1}}, - }, - // Compound index for revoked check - { - Keys: bson.D{ - {Key: "keyPrefix", Value: 1}, - {Key: "revokedAt", Value: 1}, - }, - }, - } - - _, err := s.collection().Indexes().CreateMany(ctx, indexes) - if err != nil { - return fmt.Errorf("failed to create API key indexes: %w", err) - } - - log.Println("✅ [APIKEY] Ensured indexes for api_keys collection") - return nil -} diff --git a/backend/internal/services/apikey_service_test.go b/backend/internal/services/apikey_service_test.go deleted file mode 100644 index b443c0c4..00000000 --- a/backend/internal/services/apikey_service_test.go +++ /dev/null @@ -1,258 +0,0 @@ -package services - -import ( - "claraverse/internal/models" - "context" - "strings" - "testing" -) - -func TestNewAPIKeyService(t *testing.T) { - // Test creation without MongoDB (nil) - service := NewAPIKeyService(nil, nil) - if service == nil { - t.Fatal("Expected non-nil API key service") - } -} - -func TestAPIKeyService_GenerateKey(t *testing.T) { - service := NewAPIKeyService(nil, nil) - - key, err := service.GenerateKey() - if err != nil { - t.Fatalf("Failed to generate key: %v", err) - } - - // Check prefix - if !strings.HasPrefix(key, APIKeyPrefix) { - t.Errorf("Expected key to start with '%s', got '%s'", APIKeyPrefix, key[:len(APIKeyPrefix)]) - } - - // Check length (prefix + 64 hex chars) - expectedLen := len(APIKeyPrefix) + APIKeyLength*2 - if len(key) != expectedLen { - t.Errorf("Expected key length %d, got %d", expectedLen, len(key)) - } - - // Generate another key - should be different - key2, err := service.GenerateKey() - if err != nil { - t.Fatalf("Failed to generate second key: %v", err) - } - - if key == key2 { - t.Error("Generated keys should be unique") - } -} - -func TestAPIKeyService_HashAndVerify(t *testing.T) { - service := NewAPIKeyService(nil, nil) - - key, _ := service.GenerateKey() - - // Hash the key - hash, err := service.HashKey(key) - if err != nil { - t.Fatalf("Failed to hash key: %v", err) - } - - // Hash should not be empty - if hash == "" { - t.Error("Hash should not be empty") - } - - // Hash should not equal the key - if hash == key { - t.Error("Hash should not equal the original key") - } - - // Verify correct key - if !service.VerifyKey(key, hash) { - t.Error("VerifyKey should return true for correct key") - } - - // Verify wrong key - wrongKey := key + "x" - if service.VerifyKey(wrongKey, hash) { - t.Error("VerifyKey should return false for wrong key") - } -} - -func TestAPIKeyModel_Scopes(t *testing.T) { - tests := []struct { - name string - scopes []string - check string - expected bool - }{ - { - name: "exact match", - scopes: []string{"execute:*", "read:executions"}, - check: "execute:*", - expected: true, - }, - { - name: "wildcard execute", - scopes: []string{"execute:*"}, - check: "execute:agent-123", - expected: true, - }, - { - name: "specific agent", - scopes: []string{"execute:agent-123"}, - check: "execute:agent-123", - expected: true, - }, - { - name: "wrong agent", - scopes: []string{"execute:agent-123"}, - check: "execute:agent-456", - expected: false, - }, - { - name: "full access", - scopes: []string{"*"}, - check: "execute:agent-123", - expected: true, - }, - { - name: "no match", - scopes: []string{"read:executions"}, - check: "execute:*", - expected: false, - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - key := &models.APIKey{Scopes: tt.scopes} - result := key.HasScope(tt.check) - if result != tt.expected { - t.Errorf("HasScope(%s) = %v, expected %v", tt.check, result, tt.expected) - } - }) - } -} - -func TestAPIKeyModel_HasExecuteScope(t *testing.T) { - tests := []struct { - name string - scopes []string - agentID string - expected bool - }{ - { - name: "wildcard execute", - scopes: []string{"execute:*"}, - agentID: "agent-123", - expected: true, - }, - { - name: "specific agent match", - scopes: []string{"execute:agent-123"}, - agentID: "agent-123", - expected: true, - }, - { - name: "specific agent no match", - scopes: []string{"execute:agent-456"}, - agentID: "agent-123", - expected: false, - }, - { - name: "no execute scope", - scopes: []string{"read:executions"}, - agentID: "agent-123", - expected: false, - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - key := &models.APIKey{Scopes: tt.scopes} - result := key.HasExecuteScope(tt.agentID) - if result != tt.expected { - t.Errorf("HasExecuteScope(%s) = %v, expected %v", tt.agentID, result, tt.expected) - } - }) - } -} - -func TestAPIKeyModel_IsValid(t *testing.T) { - key := &models.APIKey{} - - // Should be valid by default (no revocation, no expiration) - if !key.IsValid() { - t.Error("New key should be valid") - } - - if key.IsRevoked() { - t.Error("New key should not be revoked") - } - - if key.IsExpired() { - t.Error("New key should not be expired") - } -} - -func TestAPIKeyListItem_Conversion(t *testing.T) { - key := &models.APIKey{ - KeyPrefix: "clv_test1234", - Name: "Test Key", - Description: "A test key", - Scopes: []string{"execute:*"}, - } - - item := key.ToListItem() - - if item.KeyPrefix != key.KeyPrefix { - t.Errorf("KeyPrefix mismatch: got %s, want %s", item.KeyPrefix, key.KeyPrefix) - } - if item.Name != key.Name { - t.Errorf("Name mismatch: got %s, want %s", item.Name, key.Name) - } - if item.Description != key.Description { - t.Errorf("Description mismatch: got %s, want %s", item.Description, key.Description) - } - if len(item.Scopes) != len(key.Scopes) { - t.Errorf("Scopes length mismatch: got %d, want %d", len(item.Scopes), len(key.Scopes)) - } -} - -func TestIsValidScope(t *testing.T) { - tests := []struct { - scope string - expected bool - }{ - {"execute:*", true}, - {"read:executions", true}, - {"read:*", true}, - {"*", true}, - {"execute:agent-123", true}, - {"invalid", false}, - {"write:*", false}, - {"delete:*", false}, - } - - for _, tt := range tests { - t.Run(tt.scope, func(t *testing.T) { - result := models.IsValidScope(tt.scope) - if result != tt.expected { - t.Errorf("IsValidScope(%s) = %v, expected %v", tt.scope, result, tt.expected) - } - }) - } -} - -func TestAPIKeyService_Integration(t *testing.T) { - if testing.Short() { - t.Skip("Skipping integration test in short mode") - } - - // This test would require MongoDB - _ = context.Background() - service := NewAPIKeyService(nil, nil) - if service == nil { - t.Fatal("Expected non-nil service") - } -} diff --git a/backend/internal/services/audio_init.go b/backend/internal/services/audio_init.go deleted file mode 100644 index 78acca4e..00000000 --- a/backend/internal/services/audio_init.go +++ /dev/null @@ -1,84 +0,0 @@ -package services - -import ( - "claraverse/internal/audio" - "fmt" - "log" - "sync" -) - -var audioInitOnce sync.Once - -// InitAudioService initializes the audio package with provider access -// Priority: Groq (cheaper) -> OpenAI (fallback) -func InitAudioService() { - if visionProviderSvc == nil { - log.Println("⚠️ [AUDIO-INIT] Provider service not set, audio service disabled") - return - } - - audioInitOnce.Do(func() { - // Groq provider getter callback (primary - much cheaper) - groqProviderGetter := func() (*audio.Provider, error) { - // Try to get Groq provider by name (try both cases) - provider, err := visionProviderSvc.GetByName("Groq") - if err != nil || provider == nil { - // Fallback to lowercase - provider, err = visionProviderSvc.GetByName("groq") - } - if err != nil { - return nil, fmt.Errorf("Groq provider not found: %w", err) - } - if provider == nil { - return nil, fmt.Errorf("Groq provider not configured") - } - if !provider.Enabled { - return nil, fmt.Errorf("Groq provider is disabled") - } - if provider.APIKey == "" { - return nil, fmt.Errorf("Groq API key not configured") - } - - return &audio.Provider{ - ID: provider.ID, - Name: provider.Name, - BaseURL: provider.BaseURL, - APIKey: provider.APIKey, - Enabled: provider.Enabled, - }, nil - } - - // OpenAI provider getter callback (fallback) - openaiProviderGetter := func() (*audio.Provider, error) { - // Try to get OpenAI provider by name (try both cases) - provider, err := visionProviderSvc.GetByName("OpenAI") - if err != nil || provider == nil { - // Fallback to lowercase - provider, err = visionProviderSvc.GetByName("openai") - } - if err != nil { - return nil, fmt.Errorf("OpenAI provider not found: %w", err) - } - if provider == nil { - return nil, fmt.Errorf("OpenAI provider not configured") - } - if !provider.Enabled { - return nil, fmt.Errorf("OpenAI provider is disabled") - } - if provider.APIKey == "" { - return nil, fmt.Errorf("OpenAI API key not configured") - } - - return &audio.Provider{ - ID: provider.ID, - Name: provider.Name, - BaseURL: provider.BaseURL, - APIKey: provider.APIKey, - Enabled: provider.Enabled, - }, nil - } - - audio.InitService(groqProviderGetter, openaiProviderGetter) - log.Printf("✅ [AUDIO-INIT] Audio service initialized (Groq primary, OpenAI fallback)") - }) -} diff --git a/backend/internal/services/builder_conversation_service.go b/backend/internal/services/builder_conversation_service.go deleted file mode 100644 index 087fbaa7..00000000 --- a/backend/internal/services/builder_conversation_service.go +++ /dev/null @@ -1,250 +0,0 @@ -package services - -import ( - "claraverse/internal/crypto" - "claraverse/internal/database" - "claraverse/internal/models" - "context" - "encoding/json" - "fmt" - "log" - "time" - - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/bson/primitive" - "go.mongodb.org/mongo-driver/mongo" - "go.mongodb.org/mongo-driver/mongo/options" -) - -// BuilderConversationService handles builder conversation operations with MongoDB -type BuilderConversationService struct { - db *database.MongoDB - collection *mongo.Collection - encryption *crypto.EncryptionService -} - -// NewBuilderConversationService creates a new builder conversation service -func NewBuilderConversationService(db *database.MongoDB, encryption *crypto.EncryptionService) *BuilderConversationService { - return &BuilderConversationService{ - db: db, - collection: db.Collection(database.CollectionBuilderConversations), - encryption: encryption, - } -} - -// CreateConversation creates a new builder conversation for an agent -func (s *BuilderConversationService) CreateConversation(ctx context.Context, agentID, userID, modelID string) (*models.ConversationResponse, error) { - now := time.Now() - - // Create encrypted conversation with string-based IDs - // AgentID is a timestamp-based string (e.g., "1765018813035-yplenlye1") - // UserID is a Supabase UUID string - conv := &models.EncryptedBuilderConversation{ - AgentID: agentID, - UserID: userID, - EncryptedMessages: "", // Empty at creation - ModelID: modelID, - MessageCount: 0, - CreatedAt: now, - UpdatedAt: now, - } - - result, err := s.collection.InsertOne(ctx, conv) - if err != nil { - return nil, fmt.Errorf("failed to create conversation: %w", err) - } - - conv.ID = result.InsertedID.(primitive.ObjectID) - - return &models.ConversationResponse{ - ID: conv.ID.Hex(), - AgentID: agentID, - ModelID: modelID, - Messages: []models.BuilderMessage{}, - CreatedAt: now, - UpdatedAt: now, - }, nil -} - -// GetConversation retrieves a conversation by ID and decrypts messages -func (s *BuilderConversationService) GetConversation(ctx context.Context, conversationID, userID string) (*models.ConversationResponse, error) { - convOID, err := primitive.ObjectIDFromHex(conversationID) - if err != nil { - return nil, fmt.Errorf("invalid conversation ID: %w", err) - } - - var encrypted models.EncryptedBuilderConversation - err = s.collection.FindOne(ctx, bson.M{"_id": convOID}).Decode(&encrypted) - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("conversation not found") - } - if err != nil { - return nil, fmt.Errorf("failed to get conversation: %w", err) - } - - // Decrypt messages - var messages []models.BuilderMessage - if encrypted.EncryptedMessages != "" { - decrypted, err := s.encryption.Decrypt(userID, encrypted.EncryptedMessages) - if err != nil { - log.Printf("⚠️ Failed to decrypt conversation %s: %v", conversationID, err) - // Return empty messages on decryption failure - messages = []models.BuilderMessage{} - } else { - if err := json.Unmarshal(decrypted, &messages); err != nil { - log.Printf("⚠️ Failed to unmarshal decrypted messages: %v", err) - messages = []models.BuilderMessage{} - } - } - } - - return &models.ConversationResponse{ - ID: conversationID, - AgentID: encrypted.AgentID, // AgentID is already a string - ModelID: encrypted.ModelID, - Messages: messages, - CreatedAt: encrypted.CreatedAt, - UpdatedAt: encrypted.UpdatedAt, - }, nil -} - -// GetConversationsByAgent retrieves all conversations for an agent -func (s *BuilderConversationService) GetConversationsByAgent(ctx context.Context, agentID, userID string) ([]models.ConversationListItem, error) { - opts := options.Find(). - SetSort(bson.M{"updatedAt": -1}). - SetLimit(50) - - // AgentID is a string (timestamp-based), not an ObjectID - cursor, err := s.collection.Find(ctx, bson.M{"agentId": agentID}, opts) - if err != nil { - return nil, fmt.Errorf("failed to list conversations: %w", err) - } - defer cursor.Close(ctx) - - var conversations []models.ConversationListItem - for cursor.Next(ctx) { - var encrypted models.EncryptedBuilderConversation - if err := cursor.Decode(&encrypted); err != nil { - log.Printf("⚠️ Failed to decode conversation: %v", err) - continue - } - conversations = append(conversations, encrypted.ToListItem()) - } - - return conversations, nil -} - -// AddMessage adds a message to a conversation -func (s *BuilderConversationService) AddMessage(ctx context.Context, conversationID, userID string, req *models.AddMessageRequest) (*models.BuilderMessage, error) { - // Get existing conversation - conv, err := s.GetConversation(ctx, conversationID, userID) - if err != nil { - return nil, err - } - - // Create new message - message := models.BuilderMessage{ - ID: fmt.Sprintf("msg-%d", time.Now().UnixNano()), - Role: req.Role, - Content: req.Content, - Timestamp: time.Now(), - WorkflowSnapshot: req.WorkflowSnapshot, - } - - // Add message to list - messages := append(conv.Messages, message) - - // Serialize and encrypt messages - messagesJSON, err := json.Marshal(messages) - if err != nil { - return nil, fmt.Errorf("failed to serialize messages: %w", err) - } - - encryptedMessages, err := s.encryption.Encrypt(userID, messagesJSON) - if err != nil { - return nil, fmt.Errorf("failed to encrypt messages: %w", err) - } - - // Update conversation - convOID, _ := primitive.ObjectIDFromHex(conversationID) - _, err = s.collection.UpdateOne(ctx, - bson.M{"_id": convOID}, - bson.M{ - "$set": bson.M{ - "encryptedMessages": encryptedMessages, - "messageCount": len(messages), - "updatedAt": time.Now(), - }, - }, - ) - if err != nil { - return nil, fmt.Errorf("failed to update conversation: %w", err) - } - - return &message, nil -} - -// DeleteConversation deletes a conversation -func (s *BuilderConversationService) DeleteConversation(ctx context.Context, conversationID, userID string) error { - convOID, err := primitive.ObjectIDFromHex(conversationID) - if err != nil { - return fmt.Errorf("invalid conversation ID: %w", err) - } - - result, err := s.collection.DeleteOne(ctx, bson.M{"_id": convOID}) - if err != nil { - return fmt.Errorf("failed to delete conversation: %w", err) - } - - if result.DeletedCount == 0 { - return fmt.Errorf("conversation not found") - } - - log.Printf("✅ Deleted builder conversation %s for user %s", conversationID, userID) - return nil -} - -// DeleteConversationsByAgent deletes all conversations for an agent -func (s *BuilderConversationService) DeleteConversationsByAgent(ctx context.Context, agentID string) error { - // AgentID is a string (timestamp-based), not an ObjectID - result, err := s.collection.DeleteMany(ctx, bson.M{"agentId": agentID}) - if err != nil { - return fmt.Errorf("failed to delete conversations: %w", err) - } - - log.Printf("✅ Deleted %d builder conversations for agent %s", result.DeletedCount, agentID) - return nil -} - -// DeleteConversationsByUser deletes all conversations for a user (GDPR) -func (s *BuilderConversationService) DeleteConversationsByUser(ctx context.Context, userID string) error { - // UserID is a Supabase UUID string, not an ObjectID - result, err := s.collection.DeleteMany(ctx, bson.M{"userId": userID}) - if err != nil { - return fmt.Errorf("failed to delete user conversations: %w", err) - } - - log.Printf("✅ [GDPR] Deleted %d builder conversations for user %s", result.DeletedCount, userID) - return nil -} - -// GetOrCreateConversation gets the most recent conversation for an agent, or creates one if none exists -func (s *BuilderConversationService) GetOrCreateConversation(ctx context.Context, agentID, userID, modelID string) (*models.ConversationResponse, error) { - // Try to find existing conversation - // AgentID is a string (timestamp-based), not an ObjectID - opts := options.FindOne().SetSort(bson.M{"updatedAt": -1}) - - var encrypted models.EncryptedBuilderConversation - err := s.collection.FindOne(ctx, bson.M{"agentId": agentID}, opts).Decode(&encrypted) - - if err == mongo.ErrNoDocuments { - // Create new conversation - return s.CreateConversation(ctx, agentID, userID, modelID) - } - if err != nil { - return nil, fmt.Errorf("failed to find conversation: %w", err) - } - - // Return existing conversation - return s.GetConversation(ctx, encrypted.ID.Hex(), userID) -} diff --git a/backend/internal/services/chat_service.go b/backend/internal/services/chat_service.go deleted file mode 100644 index d7e2966a..00000000 --- a/backend/internal/services/chat_service.go +++ /dev/null @@ -1,3043 +0,0 @@ -package services - -import ( - "bufio" - "bytes" - "context" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "strings" - "time" - - "claraverse/internal/database" - "claraverse/internal/models" - "claraverse/internal/tools" - - cache "github.com/patrickmn/go-cache" -) - -// truncateToolCallID ensures tool call IDs are max 40 chars (OpenAI API constraint) -// OpenAI may send IDs > 40 chars, but rejects them when sent back -func truncateToolCallID(id string) string { - if len(id) <= 40 { - return id - } - // Keep prefix (like "call_") and truncate to 40 chars - return id[:40] -} - -// ChatService handles chat operations -type ChatService struct { - db *database.DB - providerService *ProviderService - conversationCache *cache.Cache // TTL cache for conversation history (30 min) - summaryCache *cache.Cache // Cache for AI-generated context summaries - toolRegistry *tools.Registry - toolService *ToolService // Tool service for credential-filtered tools - mcpBridge *MCPBridgeService // MCP bridge for local client tools - modelAliases map[int]map[string]string // Provider ID -> (Frontend Model -> Actual Model) mapping - streamBuffer *StreamBufferService // Buffer for resumable streaming - usageLimiter *UsageLimiterService // Usage limiter for tier-based limits - toolPredictorService *ToolPredictorService // Tool predictor for dynamic tool selection - memoryExtractionService *MemoryExtractionService // Memory extraction service for extracting memories from chats - memorySelectionService *MemorySelectionService // Memory selection service for selecting relevant memories - userService *UserService // User service for getting user preferences -} - -// ContextSummary stores AI-generated summary of older messages -type ContextSummary struct { - Summary string // AI-generated summary text - SummarizedCount int // Number of messages that were summarized - LastMessageIndex int // Index of the last message that was summarized - CreatedAt time.Time // When this summary was created -} - -// ImageRegistryAdapter wraps ImageRegistryService to implement tools.ImageRegistryInterface -// This adapter is needed to avoid import cycles between services and tools packages -type ImageRegistryAdapter struct { - registry *ImageRegistryService -} - -// GetByHandle implements tools.ImageRegistryInterface -func (a *ImageRegistryAdapter) GetByHandle(conversationID, handle string) *tools.ImageRegistryEntry { - entry := a.registry.GetByHandle(conversationID, handle) - if entry == nil { - return nil - } - return &tools.ImageRegistryEntry{ - Handle: entry.Handle, - FileID: entry.FileID, - Filename: entry.Filename, - Source: entry.Source, - } -} - -// ListHandles implements tools.ImageRegistryInterface -func (a *ImageRegistryAdapter) ListHandles(conversationID string) []string { - return a.registry.ListHandles(conversationID) -} - -// RegisterGeneratedImage implements tools.ImageRegistryInterface -func (a *ImageRegistryAdapter) RegisterGeneratedImage(conversationID, fileID, prompt string) string { - return a.registry.RegisterGeneratedImage(conversationID, fileID, prompt) -} - -// RegisterEditedImage implements tools.ImageRegistryInterface -func (a *ImageRegistryAdapter) RegisterEditedImage(conversationID, fileID, sourceHandle, prompt string) string { - return a.registry.RegisterEditedImage(conversationID, fileID, sourceHandle, prompt) -} - -// NewChatService creates a new chat service -func NewChatService( - db *database.DB, - providerService *ProviderService, - mcpBridge *MCPBridgeService, - toolService *ToolService, -) *ChatService { - // Create conversation cache with eviction hook for file cleanup - conversationCache := cache.New(30*time.Minute, 10*time.Minute) - - // Create summary cache with longer TTL (1 hour) - summaries are expensive to regenerate - summaryCache := cache.New(1*time.Hour, 15*time.Minute) - - // Set up eviction handler to cleanup associated files - conversationCache.OnEvicted(func(key string, value interface{}) { - conversationID := key - log.Printf("🗑️ [CACHE-EVICT] Conversation %s expired, cleaning up associated files...", conversationID) - - // Get file cache service - fileCache := GetFileCacheService() - - // Delete all files for this conversation - fileCache.DeleteConversationFiles(conversationID) - - // Also clean up the summary cache - summaryCache.Delete(conversationID) - - log.Printf("✅ [CACHE-EVICT] Cleanup completed for conversation %s", conversationID) - }) - - return &ChatService{ - db: db, - providerService: providerService, - conversationCache: conversationCache, - summaryCache: summaryCache, - toolRegistry: tools.GetRegistry(), - toolService: toolService, - mcpBridge: mcpBridge, - modelAliases: make(map[int]map[string]string), - streamBuffer: NewStreamBufferService(), - } -} - -// SetToolService sets the tool service for credential-filtered tools -// This allows setting the tool service after initialization when there are circular dependencies -func (s *ChatService) SetToolService(toolService *ToolService) { - s.toolService = toolService - log.Println("✅ [CHAT-SERVICE] Tool service set for credential-filtered tools") -} - -// SetUsageLimiter sets the usage limiter for tier-based usage limits -func (s *ChatService) SetUsageLimiter(usageLimiter *UsageLimiterService) { - s.usageLimiter = usageLimiter - log.Println("✅ [CHAT-SERVICE] Usage limiter set for tier-based limits") -} - -// SetToolPredictorService sets the tool predictor service for dynamic tool selection -func (s *ChatService) SetToolPredictorService(predictor *ToolPredictorService) { - s.toolPredictorService = predictor - log.Println("✅ [CHAT-SERVICE] Tool predictor service set for smart tool routing") -} - -// SetMemoryExtractionService sets the memory extraction service for extracting memories from chats -func (s *ChatService) SetMemoryExtractionService(memoryExtraction *MemoryExtractionService) { - s.memoryExtractionService = memoryExtraction - log.Println("✅ [CHAT-SERVICE] Memory extraction service set for conversation memory extraction") -} - -// SetMemorySelectionService sets the memory selection service for selecting relevant memories -func (s *ChatService) SetMemorySelectionService(memorySelection *MemorySelectionService) { - s.memorySelectionService = memorySelection - log.Println("✅ [CHAT-SERVICE] Memory selection service set for memory injection") -} - -// SetUserService sets the user service for getting user preferences -func (s *ChatService) SetUserService(userService *UserService) { - s.userService = userService - log.Println("✅ [CHAT-SERVICE] User service set for preference checking") -} - -// GetStreamBuffer returns the stream buffer service for resume handling -func (s *ChatService) GetStreamBuffer() *StreamBufferService { - return s.streamBuffer -} - -// SetModelAliases sets model aliases for a provider -func (s *ChatService) SetModelAliases(providerID int, aliases map[string]models.ModelAlias) { - if aliases != nil && len(aliases) > 0 { - // Convert ModelAlias to string map for internal storage - stringAliases := make(map[string]string) - for frontend, alias := range aliases { - stringAliases[frontend] = alias.ActualModel - } - s.modelAliases[providerID] = stringAliases - - log.Printf("🔄 [MODEL-ALIAS] Loaded %d model aliases for provider %d", len(aliases), providerID) - for frontend, alias := range aliases { - if alias.Description != "" { - log.Printf(" %s -> %s (%s)", frontend, alias.ActualModel, alias.Description) - } else { - log.Printf(" %s -> %s", frontend, alias.ActualModel) - } - } - } -} - -// resolveModelName resolves a frontend model name to the actual model name using aliases -func (s *ChatService) resolveModelName(providerID int, modelName string) string { - if aliases, exists := s.modelAliases[providerID]; exists { - if actualModel, found := aliases[modelName]; found { - log.Printf("🔄 [MODEL-ALIAS] Resolved '%s' -> '%s' for provider %d", modelName, actualModel, providerID) - return actualModel - } - } - // No alias found, return original model name - return modelName -} - -// resolveModelAlias searches all providers for a model alias and returns the provider ID and actual model name -// Returns (providerID, actualModelName, found) -func (s *ChatService) resolveModelAlias(aliasName string) (int, string, bool) { - for providerID, aliases := range s.modelAliases { - if actualModel, found := aliases[aliasName]; found { - log.Printf("🔄 [MODEL-ALIAS] Resolved alias '%s' -> provider=%d, model='%s'", aliasName, providerID, actualModel) - return providerID, actualModel, true - } - } - return 0, "", false -} - -// ResolveModelAlias is the public version that returns provider and actual model name -// Returns (provider, actualModelName, found) -func (s *ChatService) ResolveModelAlias(aliasName string) (*models.Provider, string, bool) { - providerID, actualModel, found := s.resolveModelAlias(aliasName) - if !found { - return nil, "", false - } - - provider, err := s.providerService.GetByID(providerID) - if err != nil { - log.Printf("⚠️ [MODEL-ALIAS] Found alias but provider %d not found: %v", providerID, err) - return nil, "", false - } - - return provider, actualModel, true -} - -// GetDefaultProvider returns the first available enabled provider (for fallback) -func (s *ChatService) GetDefaultProvider() (*models.Provider, error) { - providers, err := s.providerService.GetAll() - if err != nil { - return nil, fmt.Errorf("failed to get providers: %w", err) - } - - if len(providers) == 0 { - return nil, fmt.Errorf("no providers configured") - } - - // Return first enabled provider - return &providers[0], nil -} - -// GetDefaultProviderWithModel returns the first available provider and a default model from it -func (s *ChatService) GetDefaultProviderWithModel() (*models.Provider, string, error) { - provider, err := s.GetDefaultProvider() - if err != nil { - return nil, "", err - } - - // Query for the first visible model from this provider - var modelID string - err = s.db.QueryRow(` - SELECT id FROM models - WHERE provider_id = ? AND is_visible = 1 - ORDER BY name - LIMIT 1 - `, provider.ID).Scan(&modelID) - - if err != nil { - // No models found, try without visibility filter - err = s.db.QueryRow(` - SELECT id FROM models - WHERE provider_id = ? - ORDER BY name - LIMIT 1 - `, provider.ID).Scan(&modelID) - - if err != nil { - return nil, "", fmt.Errorf("no models found for default provider %s: %w", provider.Name, err) - } - } - - log.Printf("🔧 [DEFAULT] Using provider '%s' with model '%s'", provider.Name, modelID) - return provider, modelID, nil -} - -// GetTextProviderWithModel returns a text-capable provider and model for internal use (metadata generation, etc.) -// It tries model aliases first, then falls back to finding any enabled text provider -// This filters out audio-only and image-only providers -func (s *ChatService) GetTextProviderWithModel() (*models.Provider, string, error) { - // Strategy 1: Try to use model aliases from config (these are known good text models) - configService := GetConfigService() - allAliases := configService.GetAllModelAliases() - - // Get image-only provider names to filter them out - imageProviderService := GetImageProviderService() - imageProviders := imageProviderService.GetAllProviders() - imageProviderNames := make(map[string]bool) - for _, ip := range imageProviders { - imageProviderNames[ip.Name] = true - } - - // Strategy 1.5: Query database for smallest/fastest available model (prefer smaller models for metadata) - // Try to get models with lower context length (usually faster/cheaper) - log.Printf("📋 [TEXT-PROVIDER] Querying database for optimal text model...") - var modelID string - var modelProviderID int - err := s.db.QueryRow(` - SELECT m.id, m.provider_id - FROM models m - JOIN providers p ON m.provider_id = p.id - WHERE m.is_visible = 1 - AND p.enabled = 1 - AND (p.audio_only = 0 OR p.audio_only IS NULL) - ORDER BY m.context_length ASC, m.id ASC - LIMIT 1 - `).Scan(&modelID, &modelProviderID) - - if err == nil { - provider, err := s.providerService.GetByID(modelProviderID) - if err == nil && !provider.AudioOnly && !imageProviderNames[provider.Name] { - log.Printf("📋 [TEXT-PROVIDER] Found optimal model from database: %s (provider: %s)", modelID, provider.Name) - return provider, modelID, nil - } - } - - // Strategy 2: Try any available model alias - for providerID, aliases := range allAliases { - for aliasName, aliasInfo := range aliases { - provider, err := s.providerService.GetByID(providerID) - if err != nil || !provider.Enabled || provider.AudioOnly { - continue - } - if imageProviderNames[provider.Name] { - continue - } - - log.Printf("📋 [TEXT-PROVIDER] Found via any alias: %s -> %s (provider: %s)", - aliasName, aliasInfo.ActualModel, provider.Name) - return provider, aliasInfo.ActualModel, nil - } - } - - // Strategy 3: Query database for any text-capable provider with models - log.Printf("📋 [TEXT-PROVIDER] No aliases found, querying database for text provider...") - - var providerID int - var providerName, baseURL, apiKey string - var systemPrompt, favicon *string - - // Find first enabled text provider (not audio_only) that has models - err = s.db.QueryRow(` - SELECT p.id, p.name, p.base_url, p.api_key, p.system_prompt, p.favicon - FROM providers p - WHERE p.enabled = 1 AND (p.audio_only = 0 OR p.audio_only IS NULL) - AND EXISTS (SELECT 1 FROM models m WHERE m.provider_id = p.id) - ORDER BY p.id ASC - LIMIT 1 - `).Scan(&providerID, &providerName, &baseURL, &apiKey, &systemPrompt, &favicon) - - if err != nil { - return nil, "", fmt.Errorf("no text-capable provider found: %w", err) - } - - // Check if this provider is an image-only provider - if imageProviderNames[providerName] { - // Try to find the next one that's not image-only - rows, err := s.db.Query(` - SELECT p.id, p.name, p.base_url, p.api_key, p.system_prompt, p.favicon - FROM providers p - WHERE p.enabled = 1 AND (p.audio_only = 0 OR p.audio_only IS NULL) - AND EXISTS (SELECT 1 FROM models m WHERE m.provider_id = p.id) - ORDER BY p.id ASC - `) - if err != nil { - return nil, "", fmt.Errorf("failed to query providers: %w", err) - } - defer rows.Close() - - found := false - for rows.Next() { - if err := rows.Scan(&providerID, &providerName, &baseURL, &apiKey, &systemPrompt, &favicon); err != nil { - continue - } - if !imageProviderNames[providerName] { - found = true - break - } - } - - if !found { - return nil, "", fmt.Errorf("no text-capable provider found (all are image-only)") - } - } - - // Get first model from this provider - modelID = "" // Reset modelID for this provider - err = s.db.QueryRow(` - SELECT id FROM models - WHERE provider_id = ? AND is_visible = 1 - ORDER BY name - LIMIT 1 - `, providerID).Scan(&modelID) - - if err != nil { - // Try without visibility filter - err = s.db.QueryRow(` - SELECT id FROM models - WHERE provider_id = ? - ORDER BY name - LIMIT 1 - `, providerID).Scan(&modelID) - - if err != nil { - return nil, "", fmt.Errorf("no models found for provider %s: %w", providerName, err) - } - } - - provider := &models.Provider{ - ID: providerID, - Name: providerName, - BaseURL: baseURL, - APIKey: apiKey, - Enabled: true, - } - if systemPrompt != nil { - provider.SystemPrompt = *systemPrompt - } - if favicon != nil { - provider.Favicon = *favicon - } - - log.Printf("📋 [TEXT-PROVIDER] Found via database: provider=%s, model=%s", providerName, modelID) - return provider, modelID, nil -} - -// getConversationMessages retrieves messages from cache -func (s *ChatService) getConversationMessages(conversationID string) []map[string]interface{} { - if cached, found := s.conversationCache.Get(conversationID); found { - if messages, ok := cached.([]map[string]interface{}); ok { - log.Printf("📖 [CACHE] Retrieved %d messages for conversation %s", len(messages), conversationID) - return messages - } - log.Printf("⚠️ [CACHE] Invalid cache data type for conversation %s", conversationID) - } - log.Printf("📖 [CACHE] No cached messages for conversation %s (starting fresh)", conversationID) - return []map[string]interface{}{} -} - -// GetConversationMessages retrieves messages from cache (public) -func (s *ChatService) GetConversationMessages(conversationID string) []map[string]interface{} { - return s.getConversationMessages(conversationID) -} - -// setConversationMessages stores messages in cache with TTL -func (s *ChatService) setConversationMessages(conversationID string, messages []map[string]interface{}) { - s.conversationCache.Set(conversationID, messages, cache.DefaultExpiration) - log.Printf("💾 [CACHE] Stored %d messages for conversation %s", len(messages), conversationID) -} - -// Context Window Management Constants -const ( - // Maximum tokens to send to the model (conservative limit for safety) - // Most models support 128K+, but we use 80K to leave room for response - MaxContextTokens = 80000 - - // Threshold to trigger summarization (70% of max) - SummarizationThreshold = 56000 - - // Approximate tokens per character (conservative estimate) - TokensPerChar = 0.25 - - // Number of recent messages to always keep verbatim (higher = more context preserved) - RecentMessagesToKeep = 20 - - // Maximum characters for a single message before truncation - MaxMessageChars = 50000 - - // Minimum messages before summarization kicks in - MinMessagesForSummary = 15 -) - -// estimateTokens provides a rough token count for a string -// Uses the conservative estimate of ~4 chars per token -func estimateTokens(s string) int { - return int(float64(len(s)) * TokensPerChar) -} - -// estimateMessagesTokens calculates approximate token count for messages -func estimateMessagesTokens(messages []map[string]interface{}) int { - total := 0 - for _, msg := range messages { - if content, ok := msg["content"].(string); ok { - total += estimateTokens(content) - } - // Account for role and structure overhead - total += 10 - } - return total -} - -// getContextSummary retrieves a cached context summary for a conversation -func (s *ChatService) getContextSummary(conversationID string) *ContextSummary { - if cached, found := s.summaryCache.Get(conversationID); found { - if summary, ok := cached.(*ContextSummary); ok { - return summary - } - } - return nil -} - -// setContextSummary stores a context summary in cache -func (s *ChatService) setContextSummary(conversationID string, summary *ContextSummary) { - s.summaryCache.Set(conversationID, summary, cache.DefaultExpiration) - log.Printf("💾 [SUMMARY] Stored context summary for %s (%d messages summarized)", conversationID, summary.SummarizedCount) -} - -// generateContextSummary uses AI to create a summary of older messages -// This runs asynchronously to not block the main conversation -func (s *ChatService) generateContextSummary(conversationID string, messages []map[string]interface{}, config *models.Config) string { - // Build the content to summarize - var contentToSummarize strings.Builder - for i, msg := range messages { - role, _ := msg["role"].(string) - content, _ := msg["content"].(string) - if role == "system" { - continue // Skip system messages - } - // Truncate very long messages for the summary (keep more context for technical conversations) - if len(content) > 8000 { - content = content[:4000] + "\n\n[... middle content truncated for summary ...]\n\n" + content[len(content)-2000:] - } - contentToSummarize.WriteString(fmt.Sprintf("[%s #%d]: %s\n\n", role, i+1, content)) - } - - // Create summarization prompt - optimized for technical conversations - summaryPrompt := []map[string]interface{}{ - { - "role": "system", - "content": `You are a technical conversation summarizer. Your job is to create a detailed context summary that preserves ALL important information needed to continue the conversation seamlessly. - -CRITICAL - You MUST preserve: -1. **FILE PATHS & CODE** - Every file path, function name, class name, variable name mentioned -2. **TECHNICAL DECISIONS** - Architecture choices, implementation approaches, why certain solutions were chosen/rejected -3. **BUGS & FIXES** - What was broken, what fixed it, error messages encountered -4. **CONFIGURATION** - Settings, thresholds, environment variables, API endpoints discussed -5. **CURRENT STATE** - What has been implemented, what's pending, what's blocked -6. **USER PREFERENCES** - Coding style, frameworks preferred, constraints mentioned -7. **SPECIFIC VALUES** - Numbers, dates, versions, exact strings that were important - -FORMAT YOUR SUMMARY AS: -## Project Context -[What is being built/modified] - -## Files Modified/Discussed -- path/to/file.ext - what was done -- path/to/another.ext - what was changed - -## Key Technical Details -[Specific implementations, code patterns, configurations] - -## Current Status -[What's done, what's in progress, what's next] - -## Important Decisions Made -[Why certain approaches were chosen] - -## Open Issues/Blockers -[Any unresolved problems] - -Be THOROUGH - it's better to include too much detail than to lose critical context. Max 1500 words.`, - }, - { - "role": "user", - "content": fmt.Sprintf("Create a detailed technical summary of this conversation that preserves all context needed to continue:\n\n%s", contentToSummarize.String()), - }, - } - - // Make a non-streaming request for summary - chatReq := models.ChatRequest{ - Model: config.Model, - Messages: summaryPrompt, - Stream: false, - Temperature: 0.3, // Low temperature for consistency - } - - reqBody, err := json.Marshal(chatReq) - if err != nil { - log.Printf("❌ [SUMMARY] Failed to marshal request: %v", err) - return "" - } - - req, err := http.NewRequest("POST", config.BaseURL+"/chat/completions", bytes.NewBuffer(reqBody)) - if err != nil { - log.Printf("❌ [SUMMARY] Failed to create request: %v", err) - return "" - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Authorization", "Bearer "+config.APIKey) - - client := &http.Client{Timeout: 60 * time.Second} - resp, err := client.Do(req) - if err != nil { - log.Printf("❌ [SUMMARY] Request failed: %v", err) - return "" - } - defer resp.Body.Close() - - if resp.StatusCode != http.StatusOK { - body, _ := io.ReadAll(resp.Body) - log.Printf("❌ [SUMMARY] API error (status %d): %s", resp.StatusCode, string(body)) - return "" - } - - var result struct { - Choices []struct { - Message struct { - Content string `json:"content"` - } `json:"message"` - } `json:"choices"` - } - - if err := json.NewDecoder(resp.Body).Decode(&result); err != nil { - log.Printf("❌ [SUMMARY] Failed to decode response: %v", err) - return "" - } - - if len(result.Choices) == 0 { - log.Printf("⚠️ [SUMMARY] No choices in response") - return "" - } - - summary := strings.TrimSpace(result.Choices[0].Message.Content) - log.Printf("✅ [SUMMARY] Generated summary for %s (%d chars)", conversationID, len(summary)) - return summary -} - -// optimizeContextWindow manages context to prevent exceeding token limits -// Uses AI-powered summarization for older messages, preserving context -func (s *ChatService) optimizeContextWindow(messages []map[string]interface{}, conversationID string, config *models.Config, writeChan chan models.ServerMessage) []map[string]interface{} { - totalTokens := estimateMessagesTokens(messages) - - // If within limits, return as-is - if totalTokens <= SummarizationThreshold { - return messages - } - - log.Printf("📊 [CONTEXT] Context optimization needed: %d tokens exceeds %d threshold", totalTokens, SummarizationThreshold) - - // Notify client that context optimization is starting - if writeChan != nil { - select { - case writeChan <- models.ServerMessage{ - Type: "context_optimizing", - Status: "started", - Progress: 0, - Content: "Compacting conversation to keep chatting...", - }: - default: - log.Printf("⚠️ [CONTEXT] WriteChan unavailable for optimization status") - } - } - - // Strategy 1: Truncate very long individual messages - for i := range messages { - if content, ok := messages[i]["content"].(string); ok { - if len(content) > MaxMessageChars { - keepFirst := MaxMessageChars / 2 - keepLast := MaxMessageChars / 4 - truncated := content[:keepFirst] + "\n\n[... content truncated ...]\n\n" + content[len(content)-keepLast:] - messages[i]["content"] = truncated - log.Printf("✂️ [CONTEXT] Truncated message %d from %d to %d chars", i, len(content), len(truncated)) - } - } - } - - // Recalculate after truncation - totalTokens = estimateMessagesTokens(messages) - if totalTokens <= SummarizationThreshold { - // Truncation was sufficient - notify completion before returning - if writeChan != nil { - select { - case writeChan <- models.ServerMessage{ - Type: "context_optimizing", - Status: "completed", - Progress: 100, - Content: "Context optimized via truncation", - }: - default: - } - } - return messages - } - - // Strategy 2: Use AI summary for older messages - // Separate system message from conversation - var systemMsg map[string]interface{} - nonSystemMessages := make([]map[string]interface{}, 0) - - for _, msg := range messages { - if role, ok := msg["role"].(string); ok && role == "system" { - systemMsg = msg - } else { - nonSystemMessages = append(nonSystemMessages, msg) - } - } - - // Need enough messages to summarize - if len(nonSystemMessages) < MinMessagesForSummary { - // Not enough messages for summarization - notify completion - if writeChan != nil { - select { - case writeChan <- models.ServerMessage{ - Type: "context_optimizing", - Status: "completed", - Progress: 100, - Content: "Context optimization complete", - }: - default: - } - } - return messages - } - - // Calculate how many messages to keep vs summarize - recentCount := RecentMessagesToKeep - if recentCount > len(nonSystemMessages) { - recentCount = len(nonSystemMessages) - } - - oldMessages := nonSystemMessages[:len(nonSystemMessages)-recentCount] - recentMessages := nonSystemMessages[len(nonSystemMessages)-recentCount:] - - // Check if we have a valid cached summary - existingSummary := s.getContextSummary(conversationID) - var summaryText string - - if existingSummary != nil && existingSummary.SummarizedCount >= len(oldMessages)-2 { - // Use existing summary if it covers most of the old messages - summaryText = existingSummary.Summary - log.Printf("📖 [CONTEXT] Using cached summary for %s (covers %d messages)", conversationID, existingSummary.SummarizedCount) - - // Quick progress update for cached summary - if writeChan != nil { - select { - case writeChan <- models.ServerMessage{ - Type: "context_optimizing", - Status: "completed", - Progress: 100, - Content: "Using cached summary...", - }: - default: - } - } - } else if config != nil { - // Generate new AI summary - send progress update - log.Printf("🤖 [CONTEXT] Generating AI summary for %d messages in %s", len(oldMessages), conversationID) - - if writeChan != nil { - select { - case writeChan <- models.ServerMessage{ - Type: "context_optimizing", - Status: "summarizing", - Progress: 30, - Content: "Summarizing older messages...", - }: - default: - } - } - - summaryText = s.generateContextSummary(conversationID, oldMessages, config) - - if summaryText != "" { - // Cache the summary - s.setContextSummary(conversationID, &ContextSummary{ - Summary: summaryText, - SummarizedCount: len(oldMessages), - LastMessageIndex: len(nonSystemMessages) - recentCount - 1, - CreatedAt: time.Now(), - }) - - // Summary complete - if writeChan != nil { - select { - case writeChan <- models.ServerMessage{ - Type: "context_optimizing", - Status: "completed", - Progress: 100, - Content: "Context optimized successfully", - }: - default: - } - } - } else { - // AI summary failed - still notify completion so modal closes - if writeChan != nil { - select { - case writeChan <- models.ServerMessage{ - Type: "context_optimizing", - Status: "completed", - Progress: 100, - Content: "Context trimmed (summary unavailable)", - }: - default: - } - } - } - } else { - // No cached summary and no config - just notify completion - if writeChan != nil { - select { - case writeChan <- models.ServerMessage{ - Type: "context_optimizing", - Status: "completed", - Progress: 100, - Content: "Context trimmed", - }: - default: - } - } - } - - // Build optimized context - result := make([]map[string]interface{}, 0) - - // Add system message first - if systemMsg != nil { - result = append(result, systemMsg) - } - - // Add summary as a system context message - if summaryText != "" { - summaryMsg := map[string]interface{}{ - "role": "system", - "content": fmt.Sprintf(`[Conversation Context Summary - %d earlier messages] -%s - -[End of summary - continuing with recent messages]`, len(oldMessages), summaryText), - } - result = append(result, summaryMsg) - } else { - // Fallback: just note that context was trimmed - summaryMsg := map[string]interface{}{ - "role": "system", - "content": fmt.Sprintf("[Note: %d earlier messages were condensed. Recent conversation continues below.]", len(oldMessages)), - } - result = append(result, summaryMsg) - } - - // Add recent messages - result = append(result, recentMessages...) - - newTokens := estimateMessagesTokens(result) - log.Printf("📉 [CONTEXT] Reduced from %d to %d tokens (kept %d messages + summary)", totalTokens, newTokens, len(recentMessages)) - - return result -} - -// optimizeContextAfterStream runs context optimization AFTER streaming completes -// This is called asynchronously so it doesn't block the user experience -func (s *ChatService) optimizeContextAfterStream(userConn *models.UserConnection) { - // Recover from panics (user may disconnect) - defer func() { - if r := recover(); r != nil { - log.Printf("⚠️ [CONTEXT] Recovered from panic during post-stream optimization: %v", r) - } - }() - - // Get current messages from cache - messages := s.getConversationMessages(userConn.ConversationID) - totalTokens := estimateMessagesTokens(messages) - - // Check if optimization is needed - if totalTokens <= SummarizationThreshold { - log.Printf("📊 [CONTEXT] Post-stream check: %d tokens, no optimization needed (threshold: %d)", - totalTokens, SummarizationThreshold) - return - } - - log.Printf("📊 [CONTEXT] Post-stream optimization starting: %d tokens exceeds %d threshold", - totalTokens, SummarizationThreshold) - - // Get config for summarization API call - config, err := s.GetEffectiveConfig(userConn, userConn.ModelID) - if err != nil { - log.Printf("❌ [CONTEXT] Failed to get config for optimization: %v", err) - return - } - - // Run the optimization (this will send UI notifications via WriteChan) - optimizedMessages := s.optimizeContextWindow(messages, userConn.ConversationID, config, userConn.WriteChan) - - // Save optimized messages back to cache - s.setConversationMessages(userConn.ConversationID, optimizedMessages) - - log.Printf("✅ [CONTEXT] Post-stream optimization complete for %s", userConn.ConversationID) -} - -// checkAndTriggerMemoryExtraction checks if memory extraction threshold is reached -// This is called asynchronously after each assistant message -func (s *ChatService) checkAndTriggerMemoryExtraction(userConn *models.UserConnection) { - // Recover from panics - defer func() { - if r := recover(); r != nil { - log.Printf("⚠️ [MEMORY] Recovered from panic during memory extraction check: %v", r) - } - }() - - // Get user preferences to check if memory is enabled and get threshold - ctx := context.Background() - user, err := s.userService.GetUserBySupabaseID(ctx, userConn.UserID) - if err != nil { - log.Printf("⚠️ [MEMORY] Failed to get user preferences: %v", err) - return - } - - // Check if memory system is enabled for this user - if !user.Preferences.MemoryEnabled { - return // Memory system disabled, skip extraction - } - - // Get user's configured threshold (default to 20 if not set) - threshold := user.Preferences.MemoryExtractionThreshold - if threshold <= 0 { - threshold = 20 // Default to 20 messages (conservative) - } - - // Get current messages from cache - messages := s.getConversationMessages(userConn.ConversationID) - messageCount := len(messages) - - // Check if threshold reached (message count is multiple of threshold) - if messageCount > 0 && messageCount%threshold == 0 { - log.Printf("🧠 [MEMORY] Threshold reached (%d messages), enqueuing extraction job for conversation %s", - messageCount, userConn.ConversationID) - - // Get recent messages (last 'threshold' messages for extraction) - startIndex := messageCount - threshold - if startIndex < 0 { - startIndex = 0 - } - recentMessages := messages[startIndex:] - - // Enqueue extraction job (non-blocking) - ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) - defer cancel() - - err := s.memoryExtractionService.EnqueueExtraction( - ctx, - userConn.UserID, - userConn.ConversationID, - recentMessages, - ) - if err != nil { - log.Printf("⚠️ [MEMORY] Failed to enqueue extraction job: %v", err) - } else { - log.Printf("✅ [MEMORY] Extraction job enqueued successfully") - } - } -} - -// SetConversationMessages stores messages in cache with TTL (public) -func (s *ChatService) SetConversationMessages(conversationID string, messages []map[string]interface{}) { - s.setConversationMessages(conversationID, messages) -} - -// clearConversation removes all messages for a conversation (internal) -func (s *ChatService) clearConversation(conversationID string) { - s.conversationCache.Delete(conversationID) - log.Printf("🗑️ [CACHE] Cleared conversation %s", conversationID) -} - -// ClearConversation removes all messages for a conversation (public) -func (s *ChatService) ClearConversation(conversationID string) { - s.clearConversation(conversationID) -} - -// CreateConversation creates a new conversation in the database with ownership tracking -func (s *ChatService) CreateConversation(conversationID, userID, title string) error { - _, err := s.db.Exec(` - INSERT INTO conversations (id, user_id, title, expires_at) - VALUES (?, ?, ?, DATE_ADD(NOW(), INTERVAL 30 MINUTE)) - ON DUPLICATE KEY UPDATE - last_activity_at = NOW(), - expires_at = DATE_ADD(NOW(), INTERVAL 30 MINUTE) - `, conversationID, userID, title) - - if err != nil { - return fmt.Errorf("failed to create conversation: %w", err) - } - - log.Printf("📝 [OWNERSHIP] Created conversation %s for user %s", conversationID, userID) - return nil -} - -// IsConversationOwner checks if a user owns a specific conversation -func (s *ChatService) IsConversationOwner(conversationID, userID string) bool { - var ownerID string - err := s.db.QueryRow("SELECT user_id FROM conversations WHERE id = ?", conversationID).Scan(&ownerID) - - if err != nil { - // Conversation doesn't exist in database - allow access (for backward compatibility with cache-only conversations) - log.Printf("⚠️ [OWNERSHIP] Conversation %s not in database, allowing access", conversationID) - return true - } - - isOwner := ownerID == userID - if !isOwner { - log.Printf("🚫 [OWNERSHIP] User %s denied access to conversation %s (owned by %s)", userID, conversationID, ownerID) - } - - return isOwner -} - -// UpdateConversationActivity updates the last activity timestamp and extends expiration -func (s *ChatService) UpdateConversationActivity(conversationID string) error { - _, err := s.db.Exec(` - UPDATE conversations - SET last_activity_at = NOW(), - expires_at = DATE_ADD(NOW(), INTERVAL 30 MINUTE), - updated_at = NOW() - WHERE id = ? - `, conversationID) - - if err != nil { - return fmt.Errorf("failed to update conversation activity: %w", err) - } - - return nil -} - -// DeleteAllConversationsByUser deletes all conversations for a specific user (for GDPR compliance) -func (s *ChatService) DeleteAllConversationsByUser(userID string) error { - result, err := s.db.Exec("DELETE FROM conversations WHERE user_id = ?", userID) - if err != nil { - return fmt.Errorf("failed to delete user conversations: %w", err) - } - - rowsAffected, _ := result.RowsAffected() - log.Printf("🗑️ [GDPR] Deleted %d conversations for user %s", rowsAffected, userID) - - return nil -} - -// GetAllConversationsByUser retrieves all conversations for a user (for GDPR data export) -func (s *ChatService) GetAllConversationsByUser(userID string) ([]map[string]interface{}, error) { - rows, err := s.db.Query(` - SELECT id, title, created_at, updated_at, last_activity_at, expires_at - FROM conversations - WHERE user_id = ? - ORDER BY created_at DESC - `, userID) - - if err != nil { - return nil, fmt.Errorf("failed to query conversations: %w", err) - } - defer rows.Close() - - conversations := make([]map[string]interface{}, 0) - - for rows.Next() { - var id, title, createdAt, updatedAt, lastActivityAt, expiresAt string - - if err := rows.Scan(&id, &title, &createdAt, &updatedAt, &lastActivityAt, &expiresAt); err != nil { - log.Printf("⚠️ Failed to scan conversation: %v", err) - continue - } - - // Get messages from cache if available - messages := s.getConversationMessages(id) - - conversation := map[string]interface{}{ - "id": id, - "title": title, - "created_at": createdAt, - "updated_at": updatedAt, - "last_activity_at": lastActivityAt, - "expires_at": expiresAt, - "message_count": len(messages), - "messages": messages, - } - - conversations = append(conversations, conversation) - } - - return conversations, nil -} - -// ConversationStatus holds status information about a conversation -type ConversationStatus struct { - Exists bool `json:"exists"` - HasFiles bool `json:"hasFiles"` - ExpiresIn int64 `json:"expiresIn"` // seconds until expiration, -1 if expired -} - -// GetConversationStatus checks if a conversation exists and when it expires -func (s *ChatService) GetConversationStatus(conversationID string) *ConversationStatus { - status := &ConversationStatus{ - Exists: false, - HasFiles: false, - ExpiresIn: -1, - } - - // Check if conversation exists in cache - if _, expiration, found := s.conversationCache.GetWithExpiration(conversationID); found { - status.Exists = true - - // Calculate time until expiration - if !expiration.IsZero() { - timeUntilExpiration := time.Until(expiration) - status.ExpiresIn = int64(timeUntilExpiration.Seconds()) - } - - // Check if conversation has files - fileCache := GetFileCacheService() - fileIDs := fileCache.GetConversationFiles(conversationID) - status.HasFiles = len(fileIDs) > 0 - - log.Printf("📊 [STATUS] Conversation %s: exists=%v, hasFiles=%v, expiresIn=%ds", - conversationID, status.Exists, status.HasFiles, status.ExpiresIn) - } else { - log.Printf("📊 [STATUS] Conversation %s: not found in cache", conversationID) - } - - return status -} - -// AddUserMessage adds a user message to the conversation cache -func (s *ChatService) AddUserMessage(conversationID string, content interface{}) { - messages := s.getConversationMessages(conversationID) - - // 🔍 DIAGNOSTIC: Log messages retrieved before adding new one - log.Printf("🔍 [ADD-USER] Retrieved %d messages from cache for conversation %s", - len(messages), conversationID) - - messages = append(messages, map[string]interface{}{ - "role": "user", - "content": content, - }) - - // 🔍 DIAGNOSTIC: Log messages after adding new user message - log.Printf("🔍 [ADD-USER] After append: %d messages (added 1 user message)", len(messages)) - - s.setConversationMessages(conversationID, messages) -} - -// hasImageAttachments checks if messages contain any image attachments -func (s *ChatService) hasImageAttachments(messages []map[string]interface{}) bool { - for _, msg := range messages { - content := msg["content"] - if content == nil { - continue - } - - // Try []interface{} first (generic slice) - if contentArr, ok := content.([]interface{}); ok { - for _, part := range contentArr { - if partMap, ok := part.(map[string]interface{}); ok { - if partType, ok := partMap["type"].(string); ok && partType == "image_url" { - log.Printf("🖼️ [VISION-CHECK] Found image_url in []interface{} content") - return true - } - } - } - } - - // Try []map[string]interface{} (typed slice - this is what websocket handler creates) - if contentArr, ok := content.([]map[string]interface{}); ok { - for _, part := range contentArr { - if partType, ok := part["type"].(string); ok && partType == "image_url" { - log.Printf("🖼️ [VISION-CHECK] Found image_url in []map[string]interface{} content") - return true - } - } - } - } - return false -} - -// modelSupportsVision checks if a model supports vision/image inputs -func (s *ChatService) modelSupportsVision(modelID string) bool { - // First check if it's an alias and get the actual model info - configService := GetConfigService() - var actualModelName string - - for providerID, aliases := range s.modelAliases { - for aliasKey, aliasValue := range aliases { - if aliasKey == modelID { - // Found as alias - check the alias config for vision support - aliasInfo := configService.GetModelAlias(providerID, aliasKey) - if aliasInfo != nil && aliasInfo.SupportsVision != nil { - log.Printf("📊 [VISION CHECK] Alias '%s' has explicit vision support: %v", modelID, *aliasInfo.SupportsVision) - return *aliasInfo.SupportsVision - } - // If not explicitly set in alias, use actual model name for DB lookup - actualModelName = aliasValue - log.Printf("📊 [VISION CHECK] Alias '%s' -> actual model '%s' (no explicit vision setting)", modelID, actualModelName) - break - } - } - if actualModelName != "" { - break - } - } - - // Use actual model name if found via alias, otherwise use the provided modelID - queryModelName := modelID - if actualModelName != "" { - queryModelName = actualModelName - } - - // Check database for model's vision support - var supportsVision int - err := s.db.QueryRow("SELECT supports_vision FROM models WHERE id = ? OR name = ?", queryModelName, queryModelName).Scan(&supportsVision) - if err != nil { - // Model not found - assume it doesn't support vision (safer approach) - log.Printf("📊 [VISION CHECK] Model '%s' not found in database - assuming no vision support", queryModelName) - return false - } - - result := supportsVision == 1 - log.Printf("📊 [VISION CHECK] Model '%s' supports_vision=%v", queryModelName, result) - return result -} - -// findVisionCapableModel finds a vision-capable model to use as fallback -// Returns (providerID, modelName, found) -func (s *ChatService) findVisionCapableModel() (int, string, bool) { - // First, check aliases for vision-capable models (preferred) - configService := GetConfigService() - allAliases := configService.GetAllModelAliases() - - for providerID, aliases := range allAliases { - for aliasKey, aliasInfo := range aliases { - if aliasInfo.SupportsVision != nil && *aliasInfo.SupportsVision { - log.Printf("🔍 [VISION FALLBACK] Found vision-capable alias: %s (provider %d)", aliasKey, providerID) - return providerID, aliasKey, true - } - } - } - - // Query database for any vision-capable model - var providerID int - var modelName string - err := s.db.QueryRow(` - SELECT m.provider_id, m.name - FROM models m - JOIN providers p ON m.provider_id = p.id - WHERE m.supports_vision = 1 AND m.is_visible = 1 AND p.enabled = 1 - ORDER BY m.provider_id ASC - LIMIT 1 - `).Scan(&providerID, &modelName) - - if err != nil { - log.Printf("⚠️ [VISION FALLBACK] No vision-capable model found in database") - return 0, "", false - } - - log.Printf("🔍 [VISION FALLBACK] Found vision-capable model: %s (provider %d)", modelName, providerID) - return providerID, modelName, true -} - -// modelSupportsTools checks if a model supports tools (returns true if unknown - optimistic approach) -func (s *ChatService) modelSupportsTools(modelID string) bool { - log.Printf("🔍 [REQUEST] Checking if model '%s' supports tools...", modelID) - log.Printf("🔍 [DB CHECK] Querying database for model: '%s'", modelID) - - var supportsTools int - err := s.db.QueryRow("SELECT supports_tools FROM model_capabilities WHERE model_id = ?", modelID).Scan(&supportsTools) - - if err != nil { - // Model not in database or error, assume it supports tools (optimistic) - log.Printf("📊 [DB CHECK] Model '%s' NOT FOUND in database - assuming tools supported (optimistic)", modelID) - return true - } - - result := supportsTools == 1 - log.Printf("📊 [DB CHECK] Model '%s' found in database: supports_tools=%d (returning %v)", modelID, supportsTools, result) - return result -} - -// markModelNoToolSupport marks a model as not supporting tools -func (s *ChatService) markModelNoToolSupport(modelID string) error { - log.Printf("💾 [DB WRITE] Attempting to mark model '%s' as NOT supporting tools", modelID) - - result, err := s.db.Exec( - "REPLACE INTO model_capabilities (model_id, supports_tools) VALUES (?, 0)", - modelID, - ) - - if err != nil { - log.Printf("❌ [DB WRITE] Failed to mark model as no tool support: %v", err) - return fmt.Errorf("failed to mark model as no tool support: %v", err) - } - - rowsAffected, _ := result.RowsAffected() - log.Printf("✅ [DB WRITE] Successfully marked model '%s' as NOT supporting tools (rows affected: %d)", modelID, rowsAffected) - return nil -} - -// getFreeTierConfig returns the configuration for the free tier model -// This is used when anonymous users try to access restricted models -func (s *ChatService) getFreeTierConfig(connID string) (*models.Config, error) { - // Query for a free tier model - var freeTierModelID string - var freeTierModelName string - var freeTierProviderID int - - err := s.db.QueryRow(` - SELECT id, name, provider_id - FROM models - WHERE free_tier = 1 AND is_visible = 1 - LIMIT 1 - `).Scan(&freeTierModelID, &freeTierModelName, &freeTierProviderID) - - if err != nil { - log.Printf("❌ [AUTH] No free tier model configured! Anonymous users cannot use the system.") - return nil, fmt.Errorf("no free tier model available for anonymous users") - } - - provider, err := s.providerService.GetByID(freeTierProviderID) - if err != nil || !provider.Enabled { - log.Printf("❌ [AUTH] Free tier model provider is disabled or not found") - return nil, fmt.Errorf("free tier provider unavailable") - } - - log.Printf("🔒 [AUTH] Restricting connection %s to free tier model: %s", connID, freeTierModelName) - return &models.Config{ - BaseURL: provider.BaseURL, - APIKey: provider.APIKey, - Model: freeTierModelName, - }, nil -} - -// GetEffectiveConfig returns the appropriate configuration based on user's selection -func (s *ChatService) GetEffectiveConfig(userConn *models.UserConnection, modelID string) (*models.Config, error) { - // Priority 1: User provided their own API key (BYOK - Bring Your Own Key) - if userConn.CustomConfig != nil { - if userConn.CustomConfig.BaseURL != "" && - userConn.CustomConfig.APIKey != "" && - userConn.CustomConfig.Model != "" { - log.Printf("🔑 [CONFIG] Using BYOK for user %s: model=%s", userConn.ConnID, userConn.CustomConfig.Model) - return &models.Config{ - BaseURL: userConn.CustomConfig.BaseURL, - APIKey: userConn.CustomConfig.APIKey, - Model: userConn.CustomConfig.Model, - }, nil - } - - // Partial custom config - fall through to use platform providers if incomplete - log.Printf("⚠️ [CONFIG] Incomplete custom config for user %s, falling back to platform providers", userConn.ConnID) - } - - // Priority 2: User selected a model from platform (uses platform API keys) - if modelID != "" { - var providerID int - var modelName string - var foundModel bool - - // First, check if modelID is a model alias (e.g., "haiku-4.5" -> "glm-4.5-air") - if aliasProviderID, actualModel, found := s.resolveModelAlias(modelID); found { - // It's an alias - get the provider directly - provider, err := s.providerService.GetByID(aliasProviderID) - if err == nil && provider.Enabled { - // Check if anonymous user is trying to use non-free-tier model - if userConn.UserID == "anonymous" { - // Check if this model is free tier - var isFreeTier int - err := s.db.QueryRow( - "SELECT COALESCE(free_tier, 0) FROM models WHERE id = ?", - modelID, - ).Scan(&isFreeTier) - - if err != nil || isFreeTier == 0 { - // Not free tier - redirect to free tier model - log.Printf("⚠️ [AUTH] Anonymous user %s attempted to use restricted model %s (alias: %s), forcing free tier", - userConn.ConnID, actualModel, modelID) - return s.getFreeTierConfig(userConn.ConnID) - } - } - - log.Printf("🏢 [CONFIG] Using aliased model for user %s: alias=%s, actual_model=%s, provider=%s", - userConn.ConnID, modelID, actualModel, provider.Name) - - return &models.Config{ - BaseURL: provider.BaseURL, - APIKey: provider.APIKey, - Model: actualModel, - }, nil - } - } - - // Not an alias, try to find in database by model ID - err := s.db.QueryRow( - "SELECT provider_id, name FROM models WHERE id = ? AND is_visible = 1", - modelID, - ).Scan(&providerID, &modelName) - - if err == nil { - foundModel = true - } - - if foundModel { - // Check if anonymous user is trying to use non-free-tier model - if userConn.UserID == "anonymous" { - var isFreeTier int - err := s.db.QueryRow( - "SELECT COALESCE(free_tier, 0) FROM models WHERE id = ? AND is_visible = 1", - modelID, - ).Scan(&isFreeTier) - - if err != nil || isFreeTier == 0 { - // Not free tier - redirect to free tier model - log.Printf("⚠️ [AUTH] Anonymous user %s attempted to use restricted model %s, forcing free tier", - userConn.ConnID, modelName) - return s.getFreeTierConfig(userConn.ConnID) - } - } - - provider, err := s.providerService.GetByID(providerID) - if err == nil && provider.Enabled { - // Resolve model name using aliases (if configured) - actualModelName := s.resolveModelName(providerID, modelName) - - if actualModelName != modelName { - log.Printf("🏢 [CONFIG] Using platform model for user %s: frontend_model=%s, actual_model=%s, provider=%s", - userConn.ConnID, modelName, actualModelName, provider.Name) - } else { - log.Printf("🏢 [CONFIG] Using platform model for user %s: model=%s, provider=%s", - userConn.ConnID, modelName, provider.Name) - } - - return &models.Config{ - BaseURL: provider.BaseURL, - APIKey: provider.APIKey, - Model: actualModelName, // Use resolved model name - }, nil - } - } - } - - // Priority 3: Fallback to first enabled provider with visible models - log.Printf("⚙️ [CONFIG] No model selected, using fallback for user %s", userConn.ConnID) - - // Get first enabled provider - var providerID int - var providerName, baseURL, apiKey string - err := s.db.QueryRow(` - SELECT id, name, base_url, api_key - FROM providers - WHERE enabled = 1 - ORDER BY id ASC - LIMIT 1 - `).Scan(&providerID, &providerName, &baseURL, &apiKey) - - if err != nil { - return nil, fmt.Errorf("no enabled providers found: %w", err) - } - - // Get first visible model from this provider - var modelName string - err = s.db.QueryRow(` - SELECT name - FROM models - WHERE provider_id = ? AND is_visible = 1 - ORDER BY id ASC - LIMIT 1 - `, providerID).Scan(&modelName) - - if err != nil { - return nil, fmt.Errorf("no visible models found for provider %s: %w", providerName, err) - } - - log.Printf("🔄 [CONFIG] Fallback using provider=%s, model=%s for user %s", providerName, modelName, userConn.ConnID) - - return &models.Config{ - BaseURL: baseURL, - APIKey: apiKey, - Model: modelName, - }, nil -} - -// StreamChatCompletion streams chat completion responses -func (s *ChatService) StreamChatCompletion(userConn *models.UserConnection) error { - config, err := s.GetEffectiveConfig(userConn, userConn.ModelID) - if err != nil { - return fmt.Errorf("failed to get config: %w", err) - } - - // Get messages from cache instead of userConn.Messages - messages := s.getConversationMessages(userConn.ConversationID) - - // 🖼️ Auto-switch to vision model if images are present but current model doesn't support vision - if s.hasImageAttachments(messages) && !s.modelSupportsVision(userConn.ModelID) { - log.Printf("🖼️ [VISION] Images detected but model '%s' doesn't support vision - finding fallback", userConn.ModelID) - - if fallbackProviderID, fallbackModel, found := s.findVisionCapableModel(); found { - // Get the provider config for the fallback model - provider, err := s.providerService.GetByID(fallbackProviderID) - if err == nil && provider.Enabled { - // Check if fallback is an alias - if aliasProviderID, actualModel, isAlias := s.resolveModelAlias(fallbackModel); isAlias { - aliasProvider, err := s.providerService.GetByID(aliasProviderID) - if err == nil && aliasProvider.Enabled { - config = &models.Config{ - BaseURL: aliasProvider.BaseURL, - APIKey: aliasProvider.APIKey, - Model: actualModel, - } - log.Printf("🖼️ [VISION] Silently switched to vision model: %s (alias for %s)", fallbackModel, actualModel) - } - } else { - config = &models.Config{ - BaseURL: provider.BaseURL, - APIKey: provider.APIKey, - Model: fallbackModel, - } - log.Printf("🖼️ [VISION] Silently switched to vision model: %s", fallbackModel) - } - } - } else { - log.Printf("⚠️ [VISION] No vision-capable model available - proceeding with current model (may fail)") - } - } - - // 🔍 DIAGNOSTIC: Log messages retrieved from cache for streaming - log.Printf("🔍 [STREAM] Retrieved %d messages from cache for conversation %s", - len(messages), userConn.ConversationID) - if len(messages) > 0 { - // Count message types - systemCount, userCount, assistantCount := 0, 0, 0 - for _, msg := range messages { - if role, ok := msg["role"].(string); ok { - switch role { - case "system": - systemCount++ - case "user": - userCount++ - case "assistant": - assistantCount++ - } - } - } - log.Printf("🔍 [STREAM] Message breakdown BEFORE system prompt: system=%d, user=%d, assistant=%d", - systemCount, userCount, assistantCount) - } - - // ═══════════════════════════════════════════════════════════════════════════ - // TOOL SELECTION - Must happen BEFORE system prompt to determine ask_user inclusion - // ═══════════════════════════════════════════════════════════════════════════ - var tools []map[string]interface{} - if userConn.DisableTools { - log.Printf("🔒 [REQUEST] TOOLS DISABLED by client (agent builder mode)") - } else if s.modelSupportsTools(config.Model) { - // Get credential-filtered tools for user (only tools they have credentials for) - credentialFilteredTools := []map[string]interface{}{} - if s.toolService != nil { - credentialFilteredTools = s.toolService.GetAvailableTools(context.Background(), userConn.UserID) - } else { - // Fallback: Get ALL user tools (built-in + MCP tools) without filtering - credentialFilteredTools = s.toolRegistry.GetUserTools(userConn.UserID) - } - - log.Printf("📦 [REQUEST] Credential-filtered tools: %d", len(credentialFilteredTools)) - - // Use tool predictor to select subset of tools if available - if s.toolPredictorService != nil && len(credentialFilteredTools) > 0 { - // Extract user message from messages array (last user message) - userMessage := extractLastUserMessage(messages) - - log.Printf("🤖 [TOOL-PREDICTOR] Starting tool prediction with conversation history (%d messages)...", len(messages)) - predictedTools, err := s.toolPredictorService.PredictTools( - context.Background(), - userConn.UserID, - userMessage, - credentialFilteredTools, - messages, // Pass full conversation history for context-aware tool selection - ) - - if err != nil { - log.Printf("⚠️ [TOOL-PREDICTOR] Prediction failed: %v, falling back to all tools", err) - tools = credentialFilteredTools // Graceful fallback - } else { - log.Printf("✅ [TOOL-PREDICTOR] Using predicted tools: %d selected", len(predictedTools)) - tools = predictedTools - } - } else { - if s.toolPredictorService == nil { - log.Printf("📦 [REQUEST] Tool predictor not initialized, using all filtered tools") - } - tools = credentialFilteredTools - } - - // Log MCP connection status - if s.mcpBridge != nil && s.mcpBridge.IsUserConnected(userConn.UserID) { - builtinCount := s.toolRegistry.Count() - mcpCount := s.toolRegistry.CountUserTools(userConn.UserID) - builtinCount - log.Printf("📦 [REQUEST] INCLUDING TOOLS for model: %s (built-in: %d, MCP: %d, selected: %d)", - config.Model, builtinCount, mcpCount, len(tools)) - } else { - log.Printf("📦 [REQUEST] INCLUDING TOOLS for model: %s (selected tools: %d)", config.Model, len(tools)) - } - } else { - log.Printf("🚫 [REQUEST] EXCLUDING TOOLS for model: %s (marked as incompatible)", config.Model) - } - - // Get system prompt - include ask_user instructions only if tools are available - // This prevents models like Gemini from failing with MALFORMED_FUNCTION_CALL - includeAskUser := len(tools) > 0 - systemPrompt := s.GetSystemPrompt(userConn, includeAskUser) - - // Inject available images context if there are images in this conversation - imageRegistry := GetImageRegistryService() - if imageRegistry.HasImages(userConn.ConversationID) { - imageContext := imageRegistry.BuildSystemContext(userConn.ConversationID) - if imageContext != "" { - systemPrompt = systemPrompt + "\n\n" + imageContext - log.Printf("📸 [SYSTEM] Injected image context for conversation %s", userConn.ConversationID) - } - } - - messages = s.buildMessagesWithSystemPrompt(systemPrompt, messages) - - // Note: Context optimization now happens AFTER streaming ends (in processStream) - // This prevents blocking the response while the user waits - - // Prepare chat request - chatReq := models.ChatRequest{ - Model: config.Model, - Messages: messages, - Stream: true, - Temperature: 0.7, - } - - // Only include tools if non-empty (some APIs reject empty tools array) - if len(tools) > 0 { - chatReq.Tools = tools - } - - reqBody, err := json.Marshal(chatReq) - if err != nil { - return fmt.Errorf("failed to marshal request: %w", err) - } - - // 🔍 DIAGNOSTIC: Log exactly what's being sent to LLM - log.Printf("🔍 [LLM-REQUEST] Sending to LLM - Model: %s, Messages: %d, Tools: %d", - chatReq.Model, len(chatReq.Messages), len(chatReq.Tools)) - log.Printf("🔍 [LLM-REQUEST] Request body size: %d bytes", len(reqBody)) - - // 📋 Print the FULL JSON payload being sent to LLM - prettyJSON, _ := json.MarshalIndent(chatReq, "", " ") - log.Printf("📋 [LLM-REQUEST] FULL JSON PAYLOAD:\n%s", string(prettyJSON)) - - // Log all messages with FULL content for the user message - if len(chatReq.Messages) > 0 { - log.Printf("🔍 [LLM-REQUEST] === ALL MESSAGES BEING SENT TO LLM ===") - for i, msg := range chatReq.Messages { - role, _ := msg["role"].(string) - contentStr := "" - - // Handle different content types - if content, ok := msg["content"].(string); ok { - contentStr = content - } else if contentArr, ok := msg["content"].([]interface{}); ok { - // Multi-part content (vision models) - for j, part := range contentArr { - if partMap, ok := part.(map[string]interface{}); ok { - partType, _ := partMap["type"].(string) - if partType == "text" { - if text, ok := partMap["text"].(string); ok { - contentStr += fmt.Sprintf("[Part %d - text]: %s\n", j, text) - } - } else if partType == "image_url" { - contentStr += fmt.Sprintf("[Part %d - image_url]: \n", j) - } - } - } - } - - toolCallID, _ := msg["tool_call_id"].(string) - toolName, _ := msg["name"].(string) - - log.Printf("━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━") - if role == "tool" { - log.Printf("📨 [MSG %d] role=%s, tool_call_id=%s, name=%s", i, role, toolCallID, toolName) - // Truncate tool responses for readability - if len(contentStr) > 500 { - log.Printf(" content (truncated): %s...", contentStr[:500]) - } else { - log.Printf(" content: %s", contentStr) - } - } else if role == "user" { - // Show FULL user message content (includes injected CSV context) - log.Printf("👤 [MSG %d] role=%s", i, role) - log.Printf(" FULL CONTENT:\n%s", contentStr) - } else if role == "system" { - log.Printf("⚙️ [MSG %d] role=%s", i, role) - if len(contentStr) > 200 { - log.Printf(" content (truncated): %s...", contentStr[:200]) - } else { - log.Printf(" content: %s", contentStr) - } - } else if role == "assistant" { - log.Printf("🤖 [MSG %d] role=%s", i, role) - if len(contentStr) > 300 { - log.Printf(" content (truncated): %s...", contentStr[:300]) - } else { - log.Printf(" content: %s", contentStr) - } - // Log tool calls if present - if toolCalls, ok := msg["tool_calls"].([]interface{}); ok && len(toolCalls) > 0 { - log.Printf(" tool_calls: %d calls", len(toolCalls)) - for _, tc := range toolCalls { - if tcMap, ok := tc.(map[string]interface{}); ok { - if fn, ok := tcMap["function"].(map[string]interface{}); ok { - fnName, _ := fn["name"].(string) - fnArgs, _ := fn["arguments"].(string) - log.Printf(" - %s(%s)", fnName, fnArgs) - } - } - } - } - } else { - log.Printf("❓ [MSG %d] role=%s, content=%s", i, role, contentStr) - } - } - log.Printf("━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━") - log.Printf("🔍 [LLM-REQUEST] === END OF MESSAGES ===") - } - - // Create HTTP request - req, err := http.NewRequest("POST", config.BaseURL+"/chat/completions", bytes.NewBuffer(reqBody)) - if err != nil { - return fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Authorization", "Bearer "+config.APIKey) - - // Send request - client := &http.Client{Timeout: 120 * time.Second} - resp, err := client.Do(req) - if err != nil { - return fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - if resp.StatusCode != http.StatusOK { - body, _ := io.ReadAll(resp.Body) - errorMsg := string(body) - - log.Printf("⚠️ [API ERROR] API Error for %s: %s", userConn.ConnID, errorMsg) - - // Check if error is due to tool incompatibility - if len(tools) > 0 && s.detectToolIncompatibility(errorMsg) { - log.Printf("🔍 [ERROR DETECTION] Tool incompatibility detected for model: %s", config.Model) - - // Mark model as not supporting tools - if err := s.markModelNoToolSupport(config.Model); err != nil { - log.Printf("⚠️ [ERROR DETECTION] Failed to mark model: %v", err) - } - - // Add assistant error message to maintain alternation - messages := s.getConversationMessages(userConn.ConversationID) - errorMsgText := "I encountered an error. This model doesn't support tool calling. Tools have been disabled for future requests." - messages = append(messages, map[string]interface{}{ - "role": "assistant", - "content": errorMsgText, - }) - s.setConversationMessages(userConn.ConversationID, messages) - log.Printf("✅ [ERROR DETECTION] Added assistant error message to cache to maintain alternation") - - // Inform user about the error - userConn.WriteChan <- models.ServerMessage{ - Type: "error", - ErrorCode: "model_tool_incompatible", - ErrorMessage: fmt.Sprintf("Model '%s' doesn't support tool calling. Tools will be automatically disabled for this model on the next message.", config.Model), - } - - // Retry WITHOUT tools - log.Printf("🔄 [ERROR DETECTION] Retrying request WITHOUT tools for model: %s", config.Model) - return s.StreamChatCompletion(userConn) - } - - return fmt.Errorf("API error (status %d): %s", resp.StatusCode, errorMsg) - } - - // Process SSE stream - return s.processStream(resp.Body, userConn) -} - -// detectToolIncompatibility checks if an error message indicates tool incompatibility -func (s *ChatService) detectToolIncompatibility(errorMsg string) bool { - errorLower := strings.ToLower(errorMsg) - - // Common error patterns for tool incompatibility - patterns := []string{ - "roles must alternate", - "tool", - "not supported", - "function calling", - "unsupported", - } - - // Check if error contains patterns related to tools - hasToolKeyword := false - hasErrorKeyword := false - - for _, pattern := range patterns { - if strings.Contains(errorLower, pattern) { - if pattern == "tool" || pattern == "function calling" { - hasToolKeyword = true - } else { - hasErrorKeyword = true - } - } - } - - // Must have both a tool-related keyword AND an error keyword - result := hasToolKeyword && hasErrorKeyword - - if result { - log.Printf("🔍 [ERROR DETECTION] Tool incompatibility pattern detected in error: %s", errorMsg) - } - - return result || strings.Contains(errorLower, "roles must alternate") -} - -// ToolCallAccumulator accumulates streaming tool call data -type ToolCallAccumulator struct { - ID string - Type string - Name string - Arguments strings.Builder -} - -// safeSendChunk sends a chunk to the client with graceful error handling -// This prevents panics if the channel is closed (client disconnected) -func (s *ChatService) safeSendChunk(userConn *models.UserConnection, content string) { - defer func() { - if r := recover(); r != nil { - log.Printf("⚠️ [STREAM] Recovered from WriteChan panic for %s: %v (chunk buffered)", userConn.ConnID, r) - // Chunk is already buffered in streamBuffer, so no data loss - } - }() - - select { - case userConn.WriteChan <- models.ServerMessage{ - Type: "stream_chunk", - Content: content, - }: - // Successfully sent - case <-time.After(100 * time.Millisecond): - // Channel backpressure detected - client rendering slower than generation - bufferLen := len(userConn.WriteChan) - log.Printf("⚠️ [STREAM] WriteChan backpressure for %s (buffer: %d/100), chunk buffered for resume", - userConn.ConnID, bufferLen) - - // Chunk is already buffered in streamBuffer via AppendChunk before this call - // If backpressure persists, client may need to reconnect and resume - } -} - -// processStream processes the SSE stream from the AI provider -func (s *ChatService) processStream(reader io.Reader, userConn *models.UserConnection) error { - scanner := bufio.NewScanner(reader) - - // Increase buffer to 1MB for large SSE chunks (default is 64KB) - // Prevents "bufio.Scanner: token too long" errors with large tool call arguments - const maxCapacity = 1024 * 1024 // 1MB - buf := make([]byte, maxCapacity) - scanner.Buffer(buf, maxCapacity) - - var fullContent strings.Builder - - // Create stream buffer for this conversation (for resume capability) - s.streamBuffer.CreateBuffer(userConn.ConversationID, userConn.UserID, userConn.ConnID) - log.Printf("📦 [STREAM] Buffer created for conversation %s", userConn.ConversationID) - - // Track tool calls by index to accumulate streaming arguments - toolCallsMap := make(map[int]*ToolCallAccumulator) - var finishReason string - - for scanner.Scan() { - select { - case <-userConn.StopChan: - log.Printf("⏹️ Generation stopped for %s", userConn.ConnID) - // Clear buffer on stop - user explicitly cancelled - s.streamBuffer.ClearBuffer(userConn.ConversationID) - return nil - default: - } - - line := scanner.Text() - if !strings.HasPrefix(line, "data: ") { - continue - } - - data := strings.TrimPrefix(line, "data: ") - if data == "[DONE]" { - break - } - - var chunk map[string]interface{} - if err := json.Unmarshal([]byte(data), &chunk); err != nil { - continue - } - - choices, ok := chunk["choices"].([]interface{}) - if !ok || len(choices) == 0 { - continue - } - - choice := choices[0].(map[string]interface{}) - delta, ok := choice["delta"].(map[string]interface{}) - if !ok { - continue - } - - // Check for finish reason - if reason, ok := choice["finish_reason"].(string); ok && reason != "" { - finishReason = reason - } - - // Handle reasoning/thinking content (o1/o3 models) - if reasoningContent, ok := delta["reasoning_content"].(string); ok { - userConn.WriteChan <- models.ServerMessage{ - Type: "reasoning_chunk", - Content: reasoningContent, - } - } - - // Handle content chunks - if content, ok := delta["content"].(string); ok { - fullContent.WriteString(content) - - // Buffer chunk for potential resume (always buffer, even if send succeeds) - s.streamBuffer.AppendChunk(userConn.ConversationID, content) - - // Send to client with graceful handling for closed channel - s.safeSendChunk(userConn, content) - } - - // Handle tool calls - ACCUMULATE, don't execute yet! - if toolCallsData, ok := delta["tool_calls"].([]interface{}); ok { - for _, tc := range toolCallsData { - toolCallChunk := tc.(map[string]interface{}) - - // Get tool call index - var index int - if idx, ok := toolCallChunk["index"].(float64); ok { - index = int(idx) - } - - // Initialize accumulator if needed - if _, exists := toolCallsMap[index]; !exists { - toolCallsMap[index] = &ToolCallAccumulator{} - } - - acc := toolCallsMap[index] - - // Accumulate fields - if id, ok := toolCallChunk["id"].(string); ok { - acc.ID = truncateToolCallID(id) // Truncate to 40 chars (OpenAI constraint) - } - if typ, ok := toolCallChunk["type"].(string); ok { - acc.Type = typ - } - if function, ok := toolCallChunk["function"].(map[string]interface{}); ok { - if name, ok := function["name"].(string); ok { - acc.Name = name - log.Printf("🔧 [TOOL] Starting to accumulate tool call: %s (index %d)", name, index) - } - // ✅ ACCUMULATE arguments, don't parse yet! - if args, ok := function["arguments"].(string); ok { - acc.Arguments.WriteString(args) - } - } - } - } - } - - // Execute tools ONLY after streaming completes with tool_calls finish reason - if finishReason == "tool_calls" { - log.Printf("🔧 [TOOL] Streaming complete, executing %d tool call(s)", len(toolCallsMap)) - - // Get messages from cache - messages := s.getConversationMessages(userConn.ConversationID) - - // Build tool call messages for conversation history - var toolCallMessages []map[string]interface{} - var toolResults []map[string]interface{} - - // Execute all tools and collect results - for index, acc := range toolCallsMap { - if acc.Name != "" && acc.Arguments.Len() > 0 { - argsStr := acc.Arguments.String() - log.Printf("🔧 [TOOL] Executing tool %s (index %d, args length: %d bytes)", acc.Name, index, len(argsStr)) - - // Add to tool call messages - toolCallMessages = append(toolCallMessages, map[string]interface{}{ - "id": acc.ID, - "type": acc.Type, - "function": map[string]interface{}{ - "name": acc.Name, - "arguments": argsStr, - }, - }) - - // Execute tool and get result - result := s.executeToolSyncWithResult(acc.ID, acc.Name, argsStr, userConn) - toolResults = append(toolResults, map[string]interface{}{ - "role": "tool", - "tool_call_id": acc.ID, - "name": acc.Name, - "content": result, - }) - } - } - - // Only add assistant message if we have actual tool calls - if len(toolCallMessages) > 0 { - assistantMsg := map[string]interface{}{ - "role": "assistant", - "tool_calls": toolCallMessages, - } - // Only include content if it's not empty - if fullContent.Len() > 0 { - assistantMsg["content"] = fullContent.String() - } - messages = append(messages, assistantMsg) - - // Add all tool results - for _, toolResult := range toolResults { - messages = append(messages, toolResult) - } - - // Save updated messages to cache - s.setConversationMessages(userConn.ConversationID, messages) - - // Clear buffer for tool calls - a new stream will start - s.streamBuffer.ClearBuffer(userConn.ConversationID) - - // After ALL tools complete, continue conversation ONCE - log.Printf("🔄 [TOOL] All tools executed, continuing conversation with %d tool result(s)", len(toolCallMessages)) - go s.StreamChatCompletion(userConn) - } else { - // No valid tool calls - treat as error - log.Printf("⚠️ [STREAM] Tool calls detected but none were valid") - userConn.WriteChan <- models.ServerMessage{ - Type: "error", - ErrorCode: "invalid_tool_calls", - ErrorMessage: "The model attempted to call tools but the calls were invalid. Please try again.", - } - } - } else { - // Regular message without tool calls - content := fullContent.String() - - // Only add assistant message if there's actual content - if content != "" { - // Get messages from cache and add assistant response - messages := s.getConversationMessages(userConn.ConversationID) - messages = append(messages, map[string]interface{}{ - "role": "assistant", - "content": content, - }) - s.setConversationMessages(userConn.ConversationID, messages) - - // Mark stream buffer as complete before sending stream_end - s.streamBuffer.MarkComplete(userConn.ConversationID, content) - log.Printf("📦 [STREAM] Buffer marked complete for conversation %s", userConn.ConversationID) - - // Increment message counter - userConn.Mutex.Lock() - userConn.MessageCount++ - currentCount := userConn.MessageCount - userConn.Mutex.Unlock() - - // Send completion message - userConn.WriteChan <- models.ServerMessage{ - Type: "stream_end", - ConversationID: userConn.ConversationID, - } - - // Generate title after first user-assistant exchange (2 messages: user + assistant) - log.Printf("🔍 [TITLE] MessageCount=%d for conversation %s", currentCount, userConn.ConversationID) - if currentCount == 1 { - log.Printf("🎯 [TITLE] Triggering title generation for %s", userConn.ConversationID) - go s.generateConversationTitle(userConn, content) - } else { - log.Printf("⏭️ [TITLE] Skipping title generation (MessageCount=%d, need 1)", currentCount) - } - - // 🗜️ Context optimization - runs AFTER streaming ends (non-blocking) - // This compacts conversation history for the NEXT message - go s.optimizeContextAfterStream(userConn) - - // 🧠 Memory extraction - check if threshold reached (non-blocking) - if s.memoryExtractionService != nil { - go s.checkAndTriggerMemoryExtraction(userConn) - } - } else { - // Empty response - log warning and send error to client - log.Printf("⚠️ [STREAM] Received empty response from API for %s", userConn.ConnID) - userConn.WriteChan <- models.ServerMessage{ - Type: "error", - ErrorCode: "empty_response", - ErrorMessage: "The model returned an empty response. Please try again.", - } - } - } - - // Check for scanner errors (e.g., buffer overflow, I/O errors) - if err := scanner.Err(); err != nil { - log.Printf("❌ [STREAM] Scanner error for %s: %v", userConn.ConnID, err) - userConn.WriteChan <- models.ServerMessage{ - Type: "error", - ErrorCode: "stream_error", - ErrorMessage: "An error occurred while processing the stream. Please try again.", - } - return fmt.Errorf("stream scanner error: %w", err) - } - - return nil -} - -// executeToolSyncWithResult executes a tool call synchronously and returns the result -func (s *ChatService) executeToolSyncWithResult(toolCallID, toolName, argsJSON string, userConn *models.UserConnection) string { - // Get tool metadata from registry - toolDisplayName := toolName - toolIcon := "" - toolDescription := "" - if tool, exists := s.toolRegistry.Get(toolName); exists { - toolDisplayName = tool.DisplayName - toolIcon = tool.Icon - toolDescription = tool.Description - } - - // Parse complete JSON arguments - var args map[string]interface{} - if err := json.Unmarshal([]byte(argsJSON), &args); err != nil { - log.Printf("❌ Failed to parse tool arguments for %s: %v (length: %d bytes)", toolName, err, len(argsJSON)) - - // Send error to client - errorMsg := fmt.Sprintf("Failed to parse arguments: %v", err) - userConn.WriteChan <- models.ServerMessage{ - Type: "tool_result", - ToolName: toolName, - ToolDisplayName: toolDisplayName, - ToolIcon: toolIcon, - ToolDescription: toolDescription, - Status: "failed", - Result: errorMsg, - } - - return fmt.Sprintf("Error: %v", err) - } - - log.Printf("✅ [TOOL] Successfully parsed arguments for %s: %+v", toolName, args) - - // Inject user context into args (internal use only, not exposed to AI) - // This allows tools to access authenticated user info without breaking the tool interface - args["__user_id__"] = userConn.UserID - args["__conversation_id__"] = userConn.ConversationID - - // Auto-inject credentials for tools that require them - if s.toolService != nil { - // Inject credential resolver for secure credential access - resolver := s.toolService.CreateCredentialResolver(userConn.UserID) - if resolver != nil { - args[tools.CredentialResolverKey] = resolver - } - - // Auto-inject credential_id for tools that need it - credentialID := s.toolService.GetCredentialForTool(context.Background(), userConn.UserID, toolName) - if credentialID != "" { - args["credential_id"] = credentialID - log.Printf("🔐 [CHAT] Auto-injected credential_id=%s for tool=%s", credentialID, toolName) - } - } - - // Inject user connection and waiter for ask_user tool (interactive prompts) - if toolName == "ask_user" { - args[tools.UserConnectionKey] = userConn - args[tools.PromptWaiterKey] = userConn.PromptWaiter - log.Printf("🔌 [CHAT] Injected user connection and prompt waiter for ask_user tool") - } - - // Inject image provider config and registry for generate_image tool - if toolName == "generate_image" { - imageProviderService := GetImageProviderService() - provider := imageProviderService.GetProvider() - if provider != nil { - args[tools.ImageProviderConfigKey] = &tools.ImageProviderConfig{ - Name: provider.Name, - BaseURL: provider.BaseURL, - APIKey: provider.APIKey, - DefaultModel: provider.DefaultModel, - } - log.Printf("🎨 [CHAT] Injected image provider config for generate_image tool (provider: %s)", provider.Name) - } - // Inject image registry for registering generated images - imageRegistry := GetImageRegistryService() - args[tools.ImageRegistryKey] = &ImageRegistryAdapter{registry: imageRegistry} - - // Inject usage limiter for tier-based image generation limits - if s.usageLimiter != nil { - args[tools.UsageLimiterKey] = s.usageLimiter - } - } - - // Inject image edit config and registry for edit_image tool - if toolName == "edit_image" { - // Inject image registry adapter for handle lookup (adapter implements tools.ImageRegistryInterface) - imageRegistry := GetImageRegistryService() - args[tools.ImageRegistryKey] = &ImageRegistryAdapter{registry: imageRegistry} - - // Inject image edit provider config from dedicated edit provider - imageEditProviderService := GetImageEditProviderService() - editProvider := imageEditProviderService.GetProvider() - if editProvider != nil { - args[tools.ImageEditConfigKey] = &tools.ImageEditConfig{ - BaseURL: editProvider.BaseURL, - APIKey: editProvider.APIKey, - } - log.Printf("🖌️ [CHAT] Injected image edit config for edit_image tool (provider: %s)", editProvider.Name) - } else { - log.Printf("⚠️ [CHAT] No image edit provider configured - edit_image tool will fail") - } - } - - // Inject image registry for describe_image tool (allows using image_id handles) - if toolName == "describe_image" { - imageRegistry := GetImageRegistryService() - args[tools.ImageRegistryKey] = &ImageRegistryAdapter{registry: imageRegistry} - log.Printf("🖼️ [CHAT] Injected image registry for describe_image tool") - } - - // Notify client that tool is executing (send original args without internal params) - displayArgs := make(map[string]interface{}) - for k, v := range args { - // Filter out internal/sensitive params - if k != "__user_id__" && k != "__conversation_id__" && k != tools.CredentialResolverKey && k != "credential_id" && k != tools.ImageProviderConfigKey && k != tools.ImageEditConfigKey && k != tools.ImageRegistryKey && k != tools.UsageLimiterKey && k != tools.UserConnectionKey && k != tools.PromptWaiterKey { - displayArgs[k] = v - } - } - - // Use SafeSend to prevent panic if connection was closed - if !userConn.SafeSend(models.ServerMessage{ - Type: "tool_call", - ToolName: toolName, - ToolDisplayName: toolDisplayName, - ToolIcon: toolIcon, - ToolDescription: toolDescription, - Status: "executing", - Arguments: displayArgs, - }) { - log.Printf("⚠️ [TOOL] Connection closed before tool execution for %s", toolName) - return "" - } - - // Execute tool (with injected user context) - // Check if this is a built-in tool or MCP tool - tool, exists := s.toolRegistry.GetUserTool(userConn.UserID, toolName) - var result string - var err error - - if exists && tool.Source == tools.ToolSourceMCPLocal { - // MCP tool - route to local client - log.Printf("🔌 [MCP] Routing tool %s to local MCP client for user %s", toolName, userConn.UserID) - - if s.mcpBridge == nil || !s.mcpBridge.IsUserConnected(userConn.UserID) { - errorMsg := "MCP client not connected. Please start your local MCP client." - log.Printf("❌ [MCP] No client connected for user %s", userConn.UserID) - userConn.SafeSend(models.ServerMessage{ - Type: "tool_result", - ToolName: toolName, - ToolDisplayName: toolDisplayName, - ToolIcon: toolIcon, - ToolDescription: toolDescription, - Status: "failed", - Result: errorMsg, - }) - return errorMsg - } - - // Execute on MCP client with 30 second timeout - startTime := time.Now() - result, err = s.mcpBridge.ExecuteToolOnClient(userConn.UserID, toolName, args, 30*time.Second) - executionTime := int(time.Since(startTime).Milliseconds()) - - // Log execution for audit - s.mcpBridge.LogToolExecution(userConn.UserID, toolName, userConn.ConversationID, executionTime, err == nil, "") - - if err != nil { - log.Printf("❌ [MCP] Tool execution failed for %s: %v", toolName, err) - errorMsg := fmt.Sprintf("Error: %v", err) - userConn.SafeSend(models.ServerMessage{ - Type: "tool_result", - ToolName: toolName, - ToolDisplayName: toolDisplayName, - ToolIcon: toolIcon, - ToolDescription: toolDescription, - Status: "failed", - Result: errorMsg, - }) - return errorMsg - } - } else { - // Built-in tool - execute locally - result, err = s.toolRegistry.Execute(toolName, args) - if err != nil { - log.Printf("❌ Tool execution failed for %s: %v", toolName, err) - errorMsg := fmt.Sprintf("Error: %v", err) - userConn.SafeSend(models.ServerMessage{ - Type: "tool_result", - ToolName: toolName, - ToolDisplayName: toolDisplayName, - ToolIcon: toolIcon, - ToolDescription: toolDescription, - Status: "failed", - Result: errorMsg, - }) - - return errorMsg - } - } - - log.Printf("✅ [TOOL] Tool %s executed successfully, result length: %d", toolName, len(result)) - - // Try to parse result as JSON to extract plots and files (for E2B tools) - // We strip base64 data from the LLM result to avoid sending huge payloads - var resultData map[string]interface{} - var plots []models.PlotData - llmResult := result // Default: send full result to LLM - needsLLMSummary := false - - if err := json.Unmarshal([]byte(result), &resultData); err == nil { - // Check for plots - extract for frontend, strip from LLM - if plotsRaw, hasPlots := resultData["plots"]; hasPlots { - if plotsArray, ok := plotsRaw.([]interface{}); ok && len(plotsArray) > 0 { - // Extract plots for frontend - for _, p := range plotsArray { - if plotMap, ok := p.(map[string]interface{}); ok { - format, _ := plotMap["format"].(string) - data, _ := plotMap["data"].(string) - if format != "" && data != "" { - plots = append(plots, models.PlotData{ - Format: format, - Data: data, - }) - } - } - } - needsLLMSummary = true - log.Printf("📊 [TOOL] Extracted %d plot(s) from %s result", len(plots), toolName) - } - } - - // Check for files - strip base64 data from LLM result - if filesRaw, hasFiles := resultData["files"]; hasFiles { - if filesArray, ok := filesRaw.([]interface{}); ok && len(filesArray) > 0 { - needsLLMSummary = true - log.Printf("📁 [TOOL] Detected %d file(s) in %s result, stripping base64 from LLM", len(filesArray), toolName) - } - } - - // Create LLM-friendly summary (without base64 image/file data) - if needsLLMSummary { - llmSummary := map[string]interface{}{ - "success": resultData["success"], - "stdout": resultData["stdout"], - "stderr": resultData["stderr"], - } - - // Add plot count if plots exist - if len(plots) > 0 { - llmSummary["plot_count"] = len(plots) - llmSummary["plots_generated"] = fmt.Sprintf("%d visualization(s) generated and shown to user", len(plots)) - } - - // Add file info without base64 data - if filesRaw, hasFiles := resultData["files"]; hasFiles { - if filesArray, ok := filesRaw.([]interface{}); ok && len(filesArray) > 0 { - var fileNames []string - for _, f := range filesArray { - if fileMap, ok := f.(map[string]interface{}); ok { - if filename, ok := fileMap["filename"].(string); ok { - fileNames = append(fileNames, filename) - } - } - } - llmSummary["file_count"] = len(filesArray) - llmSummary["files_generated"] = fileNames - llmSummary["files_message"] = fmt.Sprintf("%d file(s) generated and available for user download", len(filesArray)) - } - } - - // Preserve other useful fields - if analysis, ok := resultData["analysis"]; ok { - llmSummary["analysis"] = analysis - } - if filename, ok := resultData["filename"]; ok { - llmSummary["filename"] = filename - } - if execTime, ok := resultData["execution_time"]; ok { - llmSummary["execution_time"] = execTime - } - if installOutput, ok := resultData["install_output"]; ok { - llmSummary["install_output"] = installOutput - } - - llmResultBytes, _ := json.Marshal(llmSummary) - llmResult = string(llmResultBytes) - } - } - - // Send result to client (with plots for frontend visualization) - // Use SafeSend to prevent panic if connection was closed during long tool execution - toolResultMsg := models.ServerMessage{ - Type: "tool_result", - ToolName: toolName, - ToolDisplayName: toolDisplayName, - ToolIcon: toolIcon, - ToolDescription: toolDescription, - Status: "completed", - Result: result, // Full result for frontend - Plots: plots, // Extracted plots for rendering - } - - // Try to send the tool result - if !userConn.SafeSend(toolResultMsg) { - log.Printf("⚠️ [TOOL] Connection closed, could not send tool result for %s", toolName) - - // Buffer tool results with artifacts (images, etc.) for reconnection recovery - // Only buffer if send failed - this ensures users don't lose generated images - if len(plots) > 0 && userConn.ConversationID != "" { - s.streamBuffer.AppendMessage(userConn.ConversationID, BufferedMessage{ - Type: "tool_result", - ToolName: toolName, - ToolDisplayName: toolDisplayName, - ToolIcon: toolIcon, - ToolDescription: toolDescription, - Status: "completed", - Result: result, - Plots: plots, - }) - log.Printf("📦 [TOOL] Buffered tool result for %s for reconnection recovery", toolName) - } - return llmResult - } - - log.Printf("✅ [TOOL] Tool result for %s ready (plots: %d)", toolName, len(plots)) - // Return LLM-friendly result (without heavy image data) - return llmResult -} - -// getMarkdownFormattingGuidelines returns formatting rules appended to all system prompts -// getAskUserInstructions returns intelligent guidance for ask_user tool usage -// Balanced approach: use when it adds value, skip when it interrupts natural flow -func getAskUserInstructions() string { - return ` - -## 🎯 Interactive Tool - ask_user - -You have an **ask_user** tool that creates interactive modal dialogs. Use it intelligently for gathering structured input. - -**When to USE ask_user (high value scenarios):** - -1. **Planning complex tasks** - Gathering requirements before implementation - - Example: "Create a website" → ask: style, colors, features, pages - - Example: "Build a game" → ask: language, library, controls, difficulty - -2. **User explicitly requests questions** - - User: "Ask me questions to understand what I need" - - User: "Help me figure out what I want" - - User: "Guide me through this" - -3. **Important decisions with multiple valid options** - - Technical choices: "Which framework? React/Vue/Angular" - - Approach selection: "Approach A (fast) or B (thorough)?" - - Confirmation for destructive actions: "Delete all files?" - -4. **Missing critical information for task execution** - - Need specific values: project name, API key, configuration - - Need preferences that significantly impact output: code style, documentation level - -**When NOT to use ask_user (let conversation flow naturally):** - -1. **Casual conversation** - Just chat normally -2. **Emotional support** - Be empathetic in text, don't interrupt with modals -3. **Simple clarifications** - Ask in text: "Did you mean X or Y?" -4. **Follow-up questions in dialogue** - Natural back-and-forth -5. **Rhetorical questions** - Part of your explanation style - -**Smart Usage Examples:** - -✅ GOOD: -- User: "Create a landing page" → ask_user: Design style? Color scheme? Sections? -- User: "I need help planning my app" → ask_user: Features? Users? Platform? -- User: "Build me a calculator" → ask_user: Basic or scientific? UI style? - -❌ NOT NEEDED: -- User: "I'm feeling lost" → Just respond with empathy, don't open modal -- User: "Tell me about React" → Just explain, don't ask questions -- Natural conversation → Keep it flowing, don't interrupt - -**Guideline:** Use ask_user when it **helps you gather structured input for better results**. Skip it when it would **interrupt natural conversation flow**. -` -} - -func getMarkdownFormattingGuidelines() string { - return ` - -## Response Style (CRITICAL) -- **Answer first**: Lead with the direct answer or solution. Context and explanation come after. -- **No filler phrases**: Never start with "Great question!", "Certainly!", "Of course!", "I'd be happy to", "Absolutely!", or similar. Just answer. -- **Be concise**: Give complete answers without unnecessary padding. Every sentence should add value. -- **No excessive caveats**: Don't lead with disclaimers or hedging. If caveats are needed, put them at the end. -- **Use structure for complex answers**: Use headers and lists for multi-part responses, but avoid over-formatting simple answers. -- **Match response length to question complexity**: Simple questions get short answers. Complex questions get thorough answers. - -## Markdown Formatting -- **Tables**: Use standard syntax with ` + "`|`" + ` separators and ` + "`---`" + ` header dividers -- **Lists**: Use ` + "`-`" + ` for unordered lists, ` + "`1.`" + ` for ordered lists (not ` + "`1)`" + `) -- **Headers**: Always include a space after ` + "`#`" + ` symbols (` + "`## Title`" + ` not ` + "`##Title`" + `) -- **Code blocks**: Always specify language after ` + "` + \"```\" + `" + ` (e.g., ` + "` + \"```python\" + `" + `, ` + "` + \"```json\" + `" + `) -- **Links**: Use ` + "`[text](url)`" + ` with no space between ` + "`]`" + ` and ` + "`(`" + ` -- **Avoid**: Citation-style ` + "`[1]`" + ` references, decorative unicode lines, non-standard bullets, emojis (unless user requests them)` -} - -// buildTemporalContext builds context string with current date/time and user name -// This provides the model with temporal awareness and personalization -func (s *ChatService) buildTemporalContext(userID string) string { - now := time.Now() - - // Format date and time - currentDate := now.Format("Monday, January 2, 2006") - currentTime := now.Format("3:04 PM MST") - - // Try to get user's name from database (if MongoDB is available) - userName := "User" // Default fallback - - // Check if we have MongoDB access to get user name - // Note: ChatService doesn't have direct MongoDB access, but we can try via the database - // For now, use a simple approach - just use UserID as identifier - // TODO: Could enhance this with UserService integration if needed - if userID != "" { - userName = userID // Use user ID as fallback - } - - // Build temporal context - context := fmt.Sprintf(`# Current Context -- **User**: %s -- **Date**: %s -- **Time**: %s - -`, userName, currentDate, currentTime) - - return context -} - -// buildMemoryContext selects and formats relevant memories for injection -func (s *ChatService) buildMemoryContext(userConn *models.UserConnection) string { - // Check if memory selection service is available - if s.memorySelectionService == nil { - return "" - } - - // Get recent messages from cache for context - messages := s.getConversationMessages(userConn.ConversationID) - if len(messages) == 0 { - return "" // No conversation history yet - } - - // Limit to last 10 messages for context - recentMessages := messages - if len(messages) > 10 { - recentMessages = messages[len(messages)-10:] - } - - // TODO: Get max memories from user preferences (default: 5) - maxMemories := 5 - - // Select relevant memories - ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second) - defer cancel() - - selectedMemories, err := s.memorySelectionService.SelectRelevantMemories( - ctx, - userConn.UserID, - recentMessages, - maxMemories, - ) - - if err != nil { - log.Printf("⚠️ [MEMORY] Failed to select memories: %v", err) - return "" - } - - if len(selectedMemories) == 0 { - return "" // No relevant memories - } - - // Build memory context string - var builder strings.Builder - builder.WriteString("\n\n## Relevant Context from Previous Conversations\n\n") - builder.WriteString("The following information was extracted from your past interactions with this user:\n\n") - - for i, mem := range selectedMemories { - builder.WriteString(fmt.Sprintf("%d. %s\n", i+1, mem.DecryptedContent)) - } - - builder.WriteString("\nUse this context to personalize responses and avoid asking for information the user has already provided.\n") - - log.Printf("🧠 [MEMORY] Injected %d memories into system prompt for user %s", len(selectedMemories), userConn.UserID) - - return builder.String() -} - -// GetSystemPrompt returns the appropriate system prompt based on priority hierarchy -// includeAskUser: whether to include ask_user tool instructions (should be true if tools are available) -func (s *ChatService) GetSystemPrompt(userConn *models.UserConnection, includeAskUser bool) string { - formattingGuidelines := getMarkdownFormattingGuidelines() - - // Only include ask_user instructions if tools will be available in the request - // Otherwise models like Gemini will fail with MALFORMED_FUNCTION_CALL when trying to use a tool that doesn't exist - var appendix string - if includeAskUser { - appendix = getAskUserInstructions() + formattingGuidelines - } else { - appendix = formattingGuidelines - log.Printf("📝 [SYSTEM] Skipping ask_user instructions (no tools selected)") - } - - // Build temporal context (user name, date, time) - prepended to all prompts - temporalContext := s.buildTemporalContext(userConn.UserID) - - // 🧠 Build memory context (injected memories from user's memory bank) - memoryContext := s.buildMemoryContext(userConn) - - // Priority 1: User-provided system instructions (per-request override) - if userConn.SystemInstructions != "" { - log.Printf("🎯 [SYSTEM] Using user-provided system instructions for %s", userConn.ConnID) - log.Printf("✅ [SYSTEM] Appending MANDATORY ask_user instructions") - return temporalContext + userConn.SystemInstructions + memoryContext + appendix - } - - // Priority 2: Model-specific system prompt (from database) - if userConn.ModelID != "" { - var modelSystemPrompt string - err := s.db.QueryRow(` - SELECT system_prompt FROM models WHERE id = ? AND system_prompt IS NOT NULL AND system_prompt != '' - `, userConn.ModelID).Scan(&modelSystemPrompt) - - if err == nil && modelSystemPrompt != "" { - log.Printf("📋 [SYSTEM] Using model-specific system prompt for %s (model: %s)", userConn.ConnID, userConn.ModelID) - log.Printf("✅ [SYSTEM] Appending MANDATORY ask_user instructions to database prompt") - return temporalContext + modelSystemPrompt + memoryContext + appendix - } - } - - // Priority 3: Provider default system prompt (from providers table) - if userConn.ModelID != "" { - var providerSystemPrompt string - err := s.db.QueryRow(` - SELECT p.system_prompt - FROM providers p - JOIN models m ON m.provider_id = p.id - WHERE m.id = ? AND p.system_prompt IS NOT NULL AND p.system_prompt != '' - `, userConn.ModelID).Scan(&providerSystemPrompt) - - if err == nil && providerSystemPrompt != "" { - log.Printf("🏢 [SYSTEM] Using provider default system prompt for %s", userConn.ConnID) - log.Printf("✅ [SYSTEM] Appending MANDATORY ask_user instructions to provider prompt") - return temporalContext + providerSystemPrompt + memoryContext + appendix - } - } - - // Priority 4: Global fallback prompt (already has ask_user instructions built-in) - log.Printf("🌐 [SYSTEM] Using global fallback system prompt for %s", userConn.ConnID) - defaultPrompt := getDefaultSystemPrompt() - - // Verify ask_user instructions are present - if strings.Contains(defaultPrompt, "ask_user") { - log.Printf("✅ [SYSTEM] ask_user tool instructions included in system prompt") - } else { - log.Printf("⚠️ [SYSTEM] WARNING: ask_user instructions NOT found in system prompt!") - } - - return temporalContext + defaultPrompt + memoryContext -} - -// getDefaultSystemPrompt returns the ClaraVerse-specific system prompt -// Tailored to the platform's actual capabilities and tools -func getDefaultSystemPrompt() string { - return `You are ClaraVerse AI, an intelligent and helpful assistant with access to powerful tools. - -## Your Capabilities - -### Interactive Prompts -- **ask_user** - Create interactive modal dialogs to gather structured input when planning tasks or making important decisions. Use this intelligently for complex workflows, not casual conversation - -### Research & Information -- **search_web** - Search the internet for current information -- **search_images** - Find images on any topic -- **scrape_web** - Extract content from specific web pages -- **get_current_time** - Get current time in any timezone - -### File Processing (when user uploads files) -- **describe_image** - Analyze and describe images in detail -- **read_document** - Extract text from PDF, DOCX, PPTX files -- **read_data_file** - Parse CSV, JSON, Excel files -- **transcribe_audio** - Convert speech to text (MP3, WAV, M4A, etc.) - -### Data Analysis & Code -- **analyze_data** - Statistical analysis with automatic visualizations (charts, graphs) -- **run_python** - Execute Python code with package support -- **train_model** - Build ML models (classification, regression, clustering) - -### Content Generation -- **generate_image** - Create AI-generated images from descriptions -- **create_presentation** - Build Reveal.js slideshows - -### Integrations (when user has credentials configured) -- **GitHub** - Create issues, list repos, add comments -- **Notion** - Search, query databases, create/update pages -- **Discord/Slack/Telegram** - Send messages to channels -- **Custom webhooks** - Send data to any endpoint - -## How to Use Tools (In Priority Order) - -1. **ask_user FIRST - ALWAYS** - - ANY question you want to ask → Use ask_user (no exceptions) - - User says ANYTHING that could go multiple ways → Use ask_user to clarify - - Examples: - * User: "Create a website" → ask_user: "What style? Modern/Classic/Minimal?" + "What colors?" + "How many pages?" - * User: "I'm not figuring out myself" → ask_user: "What did you enjoy as a kid?" + "What makes you lose track of time?" - * User: "Help me code" → ask_user: "What language?" + "What are you building?" - * User: "I'm sad" → ask_user: "What's going on?" + "How can I help?" - - **Key point**: It doesn't matter if it's technical, emotional, casual, or formal → If you're asking a question → Use ask_user - -2. **Other tools SECOND** - - search_web - For current events, prices, facts - - analyze_data - When user uploads files - - generate_image - For illustrations - - etc. - -## Response Guidelines - -### Format Rules -- **Start with the answer** - No preambles like "Great question!" -- **Match complexity** - Simple question = brief answer, Complex = structured with headers -- **Use markdown** - Headers, lists, code blocks, tables where appropriate -- **Code inline** - Add comments in code, not paragraphs explaining it - -### Markdown Formatting (IMPORTANT) -- **Tables**: Use standard syntax with ` + "`|`" + ` separators and ` + "`---`" + ` header dividers -- **Lists**: Use ` + "`-`" + ` for unordered lists, ` + "`1.`" + ` for ordered lists (not ` + "`1)`" + `) -- **Headers**: Always include a space after ` + "`#`" + ` symbols (` + "`## Title`" + ` not ` + "`##Title`" + `) -- **Code blocks**: Always specify language after ` + "` + \"```\" + `" + ` (e.g., ` + "` + \"```python\" + `" + `, ` + "` + \"```json\" + `" + `) -- **Links**: Use ` + "`[text](url)`" + ` with no space between ` + "`]`" + ` and ` + "`(`" + ` -- **Avoid**: Citation-style ` + "`[1]`" + ` references, decorative unicode lines, non-standard bullets - -### Source Citations (CRITICAL) -When using search_web or search_images, ALWAYS cite sources: - -**Inline:** "The iPhone 16 starts at $799 ([Apple](https://apple.com/iphone))." - -**End of response:** ---- -**Sources:** -- [Source Title](url) - -### Tool Usage Rules - -**🚨 MANDATORY: ask_user** - -Use ask_user for EVERY SINGLE QUESTION. No thinking. No judgment. Just do it. - -- Typing "Would you..." → STOP → ask_user -- Typing "Should I..." → STOP → ask_user -- Typing "Do you want..." → STOP → ask_user -- Typing "What do you think about..." → STOP → ask_user -- Typing ANY sentence with a question mark → STOP → ask_user - -**Examples that MUST use ask_user:** -- "What language do you prefer?" → ask_user -- "How are you feeling?" → ask_user -- "Want me to continue?" → ask_user -- "Which approach?" → ask_user -- "Tell me more?" → ask_user -- "What makes you happy?" → ask_user -- "Should I proceed?" → ask_user - -**The ONLY time you don't use ask_user:** -- When you're making a statement (no question mark) -- When you're explaining a concept they asked about -- When you're showing results/code/answers - -**Other Tools (use when appropriate):** -- search_web - Current events, prices, facts -- analyze_data - Uploaded files -- generate_image - Create images - -## Artifacts - -You can create interactive artifacts that render in the UI: - -1. **HTML/CSS/JS** - Use html code blocks for interactive web content -2. **SVG** - Use svg code blocks for vector graphics -3. **Mermaid diagrams** - Use mermaid code blocks for flowcharts, sequence diagrams, etc. - -Example Mermaid diagram: -` + "```" + `mermaid -graph LR - A[Start] --> B[Process] - B --> C[End] -` + "```" + ` - -## Tone & Style - -- **Interactive and Conversational** - Your defining trait. Engage users in dialogue via ask_user -- Professional but approachable -- No emojis unless user uses them first -- Direct and efficient (but not at the expense of being thorough with ask_user) -- Technical when appropriate, simple when not - -## Your Interactive Character - -You are designed to be a **collaborative partner**, not just a command executor. Embrace dialogue: - -✅ **DO THIS:** -- Ask questions before assuming -- Offer choices instead of making decisions for users -- Confirm understanding before executing -- Gather requirements through conversation -- Use ask_user early and often -- Treat every request as a conversation starter - -❌ **AVOID THIS:** -- Guessing at user preferences -- Making assumptions about unstated requirements -- Jumping straight to implementation without clarifying -- Asking questions in your text response when ask_user exists -- Being passive - actively engage the user - -**Remember:** Users chose ClaraVerse because they want an interactive AI that collaborates with them, not one that makes assumptions. - -## Never Do - -- Hallucinate URLs - only use URLs from actual search results -- Skip citations when using search tools -- Add unnecessary disclaimers -- Over-explain simple things -- Repeat the user's question back to them - -**🚨 CRITICALLY IMPORTANT - NEVER DO THESE:** -- **NEVER ask questions in your text response** - Use ask_user instead -- **NEVER rationalize why a question "doesn't need ask_user"** - Just use it -- **NEVER think "this is too casual for a modal"** - Wrong thinking, use ask_user -- **NEVER think "this is emotional support so I shouldn't use tools"** - Wrong thinking, use ask_user -- **NEVER think "I'll just ask in text this time"** - Wrong, use ask_user -- **NEVER write "You're right to notice that. The ask_user tool I have is designed for..."** - This is you rationalizing. Stop. Use the tool. -- **NEVER explain why you're not using ask_user** - You should be using it` -} - -// buildMessagesWithSystemPrompt ensures system prompt is the first message -func (s *ChatService) buildMessagesWithSystemPrompt(systemPrompt string, messages []map[string]interface{}) []map[string]interface{} { - // Check if first message is already a system message - if len(messages) > 0 { - if role, ok := messages[0]["role"].(string); ok && role == "system" { - // Update existing system message - messages[0]["content"] = systemPrompt - return messages - } - } - - // Prepend system message - systemMessage := map[string]interface{}{ - "role": "system", - "content": systemPrompt, - } - - return append([]map[string]interface{}{systemMessage}, messages...) -} - -// generateConversationTitle generates a short title from the conversation -func (s *ChatService) generateConversationTitle(userConn *models.UserConnection, assistantResponse string) { - // Recover from panics (e.g., send on closed channel if user disconnects) - defer func() { - if r := recover(); r != nil { - log.Printf("⚠️ [TITLE] Recovered from panic (user likely disconnected): %v", r) - } - }() - - // Get the first user message from cache - messages := s.getConversationMessages(userConn.ConversationID) - var firstUserMessage string - for _, msg := range messages { - if role, ok := msg["role"].(string); ok && role == "user" { - if content, ok := msg["content"].(string); ok { - firstUserMessage = content - break - } - } - } - - if firstUserMessage == "" { - log.Printf("⚠️ [TITLE] No user message found for title generation") - return - } - - config, err := s.GetEffectiveConfig(userConn, userConn.ModelID) - if err != nil { - log.Printf("❌ [TITLE] Failed to get config: %v", err) - return - } - - // Create a simple prompt for title generation - titlePrompt := []map[string]interface{}{ - { - "role": "system", - "content": "Generate a short, descriptive title (4-5 words maximum) for this conversation. Respond with only the title, no quotes or punctuation.", - }, - { - "role": "user", - "content": fmt.Sprintf("First message: %s\n\nAssistant response: %s", firstUserMessage, assistantResponse), - }, - } - - // Make a non-streaming request for title - chatReq := models.ChatRequest{ - Model: config.Model, - Messages: titlePrompt, - Stream: false, - Temperature: 0.7, - } - - reqBody, err := json.Marshal(chatReq) - if err != nil { - log.Printf("❌ [TITLE] Failed to marshal request: %v", err) - return - } - - req, err := http.NewRequest("POST", config.BaseURL+"/chat/completions", bytes.NewBuffer(reqBody)) - if err != nil { - log.Printf("❌ [TITLE] Failed to create request: %v", err) - return - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Authorization", "Bearer "+config.APIKey) - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - log.Printf("❌ [TITLE] Request failed: %v", err) - return - } - defer resp.Body.Close() - - if resp.StatusCode != http.StatusOK { - body, _ := io.ReadAll(resp.Body) - log.Printf("❌ [TITLE] API error (status %d): %s", resp.StatusCode, string(body)) - return - } - - var result struct { - Choices []struct { - Message struct { - Content string `json:"content"` - } `json:"message"` - } `json:"choices"` - } - - if err := json.NewDecoder(resp.Body).Decode(&result); err != nil { - log.Printf("❌ [TITLE] Failed to decode response: %v", err) - return - } - - if len(result.Choices) == 0 { - log.Printf("⚠️ [TITLE] No choices in response") - return - } - - title := strings.TrimSpace(result.Choices[0].Message.Content) - title = strings.Trim(title, `"'`) // Remove quotes if present - - // Limit to 5 words - words := strings.Fields(title) - if len(words) > 5 { - words = words[:5] - title = strings.Join(words, " ") - } - - log.Printf("📝 [TITLE] Generated title for %s (length: %d chars)", userConn.ConversationID, len(title)) - - // Send title to client (safe send - channel may be closed if user disconnected) - select { - case userConn.WriteChan <- models.ServerMessage{ - Type: "conversation_title", - ConversationID: userConn.ConversationID, - Title: title, - }: - log.Printf("✅ [TITLE] Sent title to client for %s", userConn.ConversationID) - default: - log.Printf("⚠️ [TITLE] Channel closed or full, skipping title send for %s", userConn.ConversationID) - } -} - -// extractLastUserMessage extracts the last user message content from messages array -// Handles both string content and array content (for vision messages) -func extractLastUserMessage(messages []map[string]interface{}) string { - for i := len(messages) - 1; i >= 0; i-- { - msg := messages[i] - role, _ := msg["role"].(string) - if role == "user" { - // Handle string content - if content, ok := msg["content"].(string); ok { - return content - } - // Handle array content (vision messages) - if contentArr, ok := msg["content"].([]interface{}); ok { - for _, part := range contentArr { - if partMap, ok := part.(map[string]interface{}); ok { - if partType, _ := partMap["type"].(string); partType == "text" { - if text, ok := partMap["text"].(string); ok { - return text - } - } - } - } - } - } - } - return "" -} diff --git a/backend/internal/services/chat_sync_service.go b/backend/internal/services/chat_sync_service.go deleted file mode 100644 index 612ebcaa..00000000 --- a/backend/internal/services/chat_sync_service.go +++ /dev/null @@ -1,581 +0,0 @@ -package services - -import ( - "bytes" - "claraverse/internal/crypto" - "claraverse/internal/database" - "claraverse/internal/models" - "compress/gzip" - "context" - "encoding/base64" - "encoding/json" - "fmt" - "io" - "log" - "strings" - "time" - - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/mongo" - "go.mongodb.org/mongo-driver/mongo/options" -) - -// ChatSyncService handles cloud sync operations for chats with encryption -type ChatSyncService struct { - db *database.MongoDB - collection *mongo.Collection - encryptionService *crypto.EncryptionService -} - -// NewChatSyncService creates a new chat sync service -func NewChatSyncService(db *database.MongoDB, encryptionService *crypto.EncryptionService) *ChatSyncService { - return &ChatSyncService{ - db: db, - collection: db.Collection(database.CollectionChats), - encryptionService: encryptionService, - } -} - -// CreateOrUpdateChat creates a new chat or updates an existing one -// Uses atomic upsert to prevent race conditions when multiple syncs arrive simultaneously -func (s *ChatSyncService) CreateOrUpdateChat(ctx context.Context, userID string, req *models.CreateChatRequest) (*models.ChatResponse, error) { - if userID == "" { - return nil, fmt.Errorf("user ID is required") - } - if req.ID == "" { - return nil, fmt.Errorf("chat ID is required") - } - - // Encrypt messages - messagesJSON, err := json.Marshal(req.Messages) - if err != nil { - return nil, fmt.Errorf("failed to serialize messages: %w", err) - } - - encryptedMessages, err := s.encryptionService.Encrypt(userID, messagesJSON) - if err != nil { - return nil, fmt.Errorf("failed to encrypt messages: %w", err) - } - - // Compress encrypted messages to reduce storage size (helps avoid MongoDB 16MB limit) - compressedMessages, err := s.compressData(encryptedMessages) - if err != nil { - return nil, fmt.Errorf("failed to compress messages: %w", err) - } - - now := time.Now() - - filter := bson.M{ - "userId": userID, - "chatId": req.ID, - } - - // Use atomic upsert to handle race conditions - // $setOnInsert only applies when creating a new document - // $set applies to both insert and update - // Note: Cannot use $setOnInsert and $inc on the same field (version), - // so we set version to 1 on insert via $setOnInsert, and increment for updates via $inc - update := bson.M{ - "$set": bson.M{ - "title": req.Title, - "encryptedMessages": compressedMessages, - "isStarred": req.IsStarred, - "model": req.Model, - "updatedAt": now, - }, - "$setOnInsert": bson.M{ - "userId": userID, - "chatId": req.ID, - "createdAt": now, - }, - "$inc": bson.M{ - "version": 1, - }, - } - - opts := options.FindOneAndUpdate(). - SetUpsert(true). - SetReturnDocument(options.After) - - var resultChat models.EncryptedChat - err = s.collection.FindOneAndUpdate(ctx, filter, update, opts).Decode(&resultChat) - if err != nil { - return nil, fmt.Errorf("failed to upsert chat: %w", err) - } - - return &models.ChatResponse{ - ID: req.ID, - Title: resultChat.Title, - Messages: req.Messages, - IsStarred: resultChat.IsStarred, - Model: resultChat.Model, - Version: resultChat.Version, - CreatedAt: resultChat.CreatedAt, - UpdatedAt: resultChat.UpdatedAt, - }, nil -} - -// GetChat retrieves and decrypts a single chat -func (s *ChatSyncService) GetChat(ctx context.Context, userID, chatID string) (*models.ChatResponse, error) { - if userID == "" || chatID == "" { - return nil, fmt.Errorf("user ID and chat ID are required") - } - - filter := bson.M{ - "userId": userID, - "chatId": chatID, - } - - var chat models.EncryptedChat - err := s.collection.FindOne(ctx, filter).Decode(&chat) - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("chat not found") - } - if err != nil { - return nil, fmt.Errorf("failed to get chat: %w", err) - } - - // Decrypt messages - messages, err := s.decryptMessages(userID, chat.EncryptedMessages) - if err != nil { - return nil, fmt.Errorf("failed to decrypt messages: %w", err) - } - - return &models.ChatResponse{ - ID: chat.ChatID, - Title: chat.Title, - Messages: messages, - IsStarred: chat.IsStarred, - Model: chat.Model, - Version: chat.Version, - CreatedAt: chat.CreatedAt, - UpdatedAt: chat.UpdatedAt, - }, nil -} - -// ListChats returns a paginated list of chats (metadata only, no messages) -func (s *ChatSyncService) ListChats(ctx context.Context, userID string, page, pageSize int, starredOnly bool) (*models.ChatListResponse, error) { - if userID == "" { - return nil, fmt.Errorf("user ID is required") - } - - if page < 1 { - page = 1 - } - if pageSize < 1 || pageSize > 100 { - pageSize = 20 - } - - filter := bson.M{"userId": userID} - if starredOnly { - filter["isStarred"] = true - } - - // Get total count - totalCount, err := s.collection.CountDocuments(ctx, filter) - if err != nil { - return nil, fmt.Errorf("failed to count chats: %w", err) - } - - // Find chats with pagination - skip := int64((page - 1) * pageSize) - opts := options.Find(). - SetSort(bson.D{{Key: "updatedAt", Value: -1}}). - SetSkip(skip). - SetLimit(int64(pageSize)). - SetProjection(bson.M{ - "_id": 1, - "chatId": 1, - "title": 1, - "isStarred": 1, - "model": 1, - "version": 1, - "createdAt": 1, - "updatedAt": 1, - "encryptedMessages": 1, // Need this to count messages - }) - - cursor, err := s.collection.Find(ctx, filter, opts) - if err != nil { - return nil, fmt.Errorf("failed to list chats: %w", err) - } - defer cursor.Close(ctx) - - var chats []models.ChatListItem - for cursor.Next(ctx) { - var encChat models.EncryptedChat - if err := cursor.Decode(&encChat); err != nil { - log.Printf("⚠️ Failed to decode chat: %v", err) - continue - } - - // Count messages (decrypt to get count) - messageCount := 0 - if encChat.EncryptedMessages != "" { - messages, err := s.decryptMessages(userID, encChat.EncryptedMessages) - if err == nil { - messageCount = len(messages) - } - } - - chats = append(chats, models.ChatListItem{ - ID: encChat.ChatID, - Title: encChat.Title, - IsStarred: encChat.IsStarred, - Model: encChat.Model, - MessageCount: messageCount, - Version: encChat.Version, - CreatedAt: encChat.CreatedAt, - UpdatedAt: encChat.UpdatedAt, - }) - } - - return &models.ChatListResponse{ - Chats: chats, - TotalCount: totalCount, - Page: page, - PageSize: pageSize, - HasMore: int64(page*pageSize) < totalCount, - }, nil -} - -// UpdateChat performs a partial update on a chat -func (s *ChatSyncService) UpdateChat(ctx context.Context, userID, chatID string, req *models.UpdateChatRequest) (*models.ChatListItem, error) { - if userID == "" || chatID == "" { - return nil, fmt.Errorf("user ID and chat ID are required") - } - - filter := bson.M{ - "userId": userID, - "chatId": chatID, - "version": req.Version, // Optimistic locking - } - - updateFields := bson.M{ - "updatedAt": time.Now(), - } - - if req.Title != nil { - updateFields["title"] = *req.Title - } - if req.IsStarred != nil { - updateFields["isStarred"] = *req.IsStarred - } - if req.Model != nil { - updateFields["model"] = *req.Model - } - - update := bson.M{ - "$set": updateFields, - "$inc": bson.M{"version": 1}, - } - - opts := options.FindOneAndUpdate().SetReturnDocument(options.After) - var updatedChat models.EncryptedChat - err := s.collection.FindOneAndUpdate(ctx, filter, update, opts).Decode(&updatedChat) - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("chat not found or version conflict") - } - if err != nil { - return nil, fmt.Errorf("failed to update chat: %w", err) - } - - // Count messages - messageCount := 0 - if updatedChat.EncryptedMessages != "" { - messages, err := s.decryptMessages(userID, updatedChat.EncryptedMessages) - if err == nil { - messageCount = len(messages) - } - } - - return &models.ChatListItem{ - ID: updatedChat.ChatID, - Title: updatedChat.Title, - IsStarred: updatedChat.IsStarred, - Model: updatedChat.Model, - MessageCount: messageCount, - Version: updatedChat.Version, - CreatedAt: updatedChat.CreatedAt, - UpdatedAt: updatedChat.UpdatedAt, - }, nil -} - -// DeleteChat removes a chat -func (s *ChatSyncService) DeleteChat(ctx context.Context, userID, chatID string) error { - if userID == "" || chatID == "" { - return fmt.Errorf("user ID and chat ID are required") - } - - filter := bson.M{ - "userId": userID, - "chatId": chatID, - } - - result, err := s.collection.DeleteOne(ctx, filter) - if err != nil { - return fmt.Errorf("failed to delete chat: %w", err) - } - - if result.DeletedCount == 0 { - return fmt.Errorf("chat not found") - } - - return nil -} - -// BulkSync uploads multiple chats at once -func (s *ChatSyncService) BulkSync(ctx context.Context, userID string, req *models.BulkSyncRequest) (*models.BulkSyncResponse, error) { - if userID == "" { - return nil, fmt.Errorf("user ID is required") - } - - response := &models.BulkSyncResponse{ - ChatIDs: make([]string, 0), - } - - for _, chatReq := range req.Chats { - _, err := s.CreateOrUpdateChat(ctx, userID, &chatReq) - if err != nil { - response.Failed++ - response.Errors = append(response.Errors, fmt.Sprintf("chat %s: %v", chatReq.ID, err)) - log.Printf("⚠️ Failed to sync chat %s: %v", chatReq.ID, err) - } else { - response.Synced++ - response.ChatIDs = append(response.ChatIDs, chatReq.ID) - } - } - - return response, nil -} - -// GetAllChats returns all chats for initial sync (with decrypted messages) -func (s *ChatSyncService) GetAllChats(ctx context.Context, userID string) (*models.SyncAllResponse, error) { - if userID == "" { - return nil, fmt.Errorf("user ID is required") - } - - filter := bson.M{"userId": userID} - opts := options.Find().SetSort(bson.D{{Key: "updatedAt", Value: -1}}) - - cursor, err := s.collection.Find(ctx, filter, opts) - if err != nil { - return nil, fmt.Errorf("failed to get chats: %w", err) - } - defer cursor.Close(ctx) - - chats := make([]models.ChatResponse, 0) // Initialize empty slice to avoid null in JSON - for cursor.Next(ctx) { - var encChat models.EncryptedChat - if err := cursor.Decode(&encChat); err != nil { - log.Printf("⚠️ Failed to decode chat: %v", err) - continue - } - - // Decrypt messages - messages, err := s.decryptMessages(userID, encChat.EncryptedMessages) - if err != nil { - log.Printf("⚠️ Failed to decrypt messages for chat %s: %v", encChat.ChatID, err) - continue - } - - chats = append(chats, models.ChatResponse{ - ID: encChat.ChatID, - Title: encChat.Title, - Messages: messages, - IsStarred: encChat.IsStarred, - Model: encChat.Model, - Version: encChat.Version, - CreatedAt: encChat.CreatedAt, - UpdatedAt: encChat.UpdatedAt, - }) - } - - return &models.SyncAllResponse{ - Chats: chats, - TotalCount: len(chats), - SyncedAt: time.Now(), - }, nil -} - -// AddMessage adds a single message to a chat with optimistic locking -func (s *ChatSyncService) AddMessage(ctx context.Context, userID, chatID string, req *models.ChatAddMessageRequest) (*models.ChatResponse, error) { - if userID == "" || chatID == "" { - return nil, fmt.Errorf("user ID and chat ID are required") - } - - // Get current chat - filter := bson.M{ - "userId": userID, - "chatId": chatID, - "version": req.Version, // Optimistic locking - } - - var chat models.EncryptedChat - err := s.collection.FindOne(ctx, filter).Decode(&chat) - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("chat not found or version conflict") - } - if err != nil { - return nil, fmt.Errorf("failed to get chat: %w", err) - } - - // Decrypt existing messages - messages, err := s.decryptMessages(userID, chat.EncryptedMessages) - if err != nil { - return nil, fmt.Errorf("failed to decrypt messages: %w", err) - } - - // Add new message - messages = append(messages, req.Message) - - // Re-encrypt messages - messagesJSON, err := json.Marshal(messages) - if err != nil { - return nil, fmt.Errorf("failed to serialize messages: %w", err) - } - - encryptedMessages, err := s.encryptionService.Encrypt(userID, messagesJSON) - if err != nil { - return nil, fmt.Errorf("failed to encrypt messages: %w", err) - } - - // Compress encrypted messages to reduce storage size - compressedMessages, err := s.compressData(encryptedMessages) - if err != nil { - return nil, fmt.Errorf("failed to compress messages: %w", err) - } - - // Update chat - now := time.Now() - update := bson.M{ - "$set": bson.M{ - "encryptedMessages": compressedMessages, - "updatedAt": now, - }, - "$inc": bson.M{"version": 1}, - } - - opts := options.FindOneAndUpdate().SetReturnDocument(options.After) - var updatedChat models.EncryptedChat - err = s.collection.FindOneAndUpdate(ctx, filter, update, opts).Decode(&updatedChat) - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("version conflict during update") - } - if err != nil { - return nil, fmt.Errorf("failed to update chat: %w", err) - } - - return &models.ChatResponse{ - ID: chatID, - Title: updatedChat.Title, - Messages: messages, - IsStarred: updatedChat.IsStarred, - Model: updatedChat.Model, - Version: updatedChat.Version, - CreatedAt: updatedChat.CreatedAt, - UpdatedAt: updatedChat.UpdatedAt, - }, nil -} - -// DeleteAllUserChats removes all chats for a user (GDPR compliance) -func (s *ChatSyncService) DeleteAllUserChats(ctx context.Context, userID string) (int64, error) { - if userID == "" { - return 0, fmt.Errorf("user ID is required") - } - - filter := bson.M{"userId": userID} - result, err := s.collection.DeleteMany(ctx, filter) - if err != nil { - return 0, fmt.Errorf("failed to delete user chats: %w", err) - } - - return result.DeletedCount, nil -} - -// decryptMessages decrypts and decompresses the encrypted messages JSON -func (s *ChatSyncService) decryptMessages(userID, encryptedMessages string) ([]models.ChatMessage, error) { - if encryptedMessages == "" { - return []models.ChatMessage{}, nil - } - - // Decompress if compressed (backward compatible - old data won't have gzip: prefix) - dataToDecrypt := encryptedMessages - if strings.HasPrefix(encryptedMessages, "gzip:") { - compressed := strings.TrimPrefix(encryptedMessages, "gzip:") - decompressed, err := s.decompressData(compressed) - if err != nil { - return nil, fmt.Errorf("failed to decompress messages: %w", err) - } - dataToDecrypt = decompressed - } - - decrypted, err := s.encryptionService.Decrypt(userID, dataToDecrypt) - if err != nil { - return nil, err - } - - var messages []models.ChatMessage - if err := json.Unmarshal(decrypted, &messages); err != nil { - return nil, fmt.Errorf("failed to parse messages: %w", err) - } - - return messages, nil -} - -// compressData compresses a string using gzip and returns it with a prefix marker -func (s *ChatSyncService) compressData(data string) (string, error) { - var buf bytes.Buffer - writer := gzip.NewWriter(&buf) - - if _, err := writer.Write([]byte(data)); err != nil { - return "", err - } - - if err := writer.Close(); err != nil { - return "", err - } - - // Encode to base64 and add prefix to identify compressed data - compressed := base64.StdEncoding.EncodeToString(buf.Bytes()) - return "gzip:" + compressed, nil -} - -// decompressData decompresses a base64-encoded gzip string -func (s *ChatSyncService) decompressData(compressed string) (string, error) { - // Decode base64 - data, err := base64.StdEncoding.DecodeString(compressed) - if err != nil { - return "", fmt.Errorf("failed to decode base64: %w", err) - } - - // Decompress gzip - reader, err := gzip.NewReader(bytes.NewReader(data)) - if err != nil { - return "", fmt.Errorf("failed to create gzip reader: %w", err) - } - defer reader.Close() - - decompressed, err := io.ReadAll(reader) - if err != nil { - return "", fmt.Errorf("failed to read decompressed data: %w", err) - } - - return string(decompressed), nil -} - -// EnsureIndexes creates necessary indexes for the chats collection -func (s *ChatSyncService) EnsureIndexes(ctx context.Context) error { - indexes := []mongo.IndexModel{ - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "updatedAt", Value: -1}}}, - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "chatId", Value: 1}}, Options: options.Index().SetUnique(true)}, - {Keys: bson.D{{Key: "userId", Value: 1}, {Key: "isStarred", Value: 1}}}, - } - - _, err := s.collection.Indexes().CreateMany(ctx, indexes) - if err != nil { - return fmt.Errorf("failed to create chat indexes: %w", err) - } - - return nil -} diff --git a/backend/internal/services/chat_sync_service_test.go b/backend/internal/services/chat_sync_service_test.go deleted file mode 100644 index 87eabfa2..00000000 --- a/backend/internal/services/chat_sync_service_test.go +++ /dev/null @@ -1,976 +0,0 @@ -package services - -import ( - "claraverse/internal/crypto" - "claraverse/internal/models" - "encoding/json" - "testing" - "time" -) - -// Test encryption service creation and basic operations -func TestEncryptionService(t *testing.T) { - // Generate a test master key (32 bytes = 64 hex chars) - masterKey := "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef" - - encService, err := crypto.NewEncryptionService(masterKey) - if err != nil { - t.Fatalf("Failed to create encryption service: %v", err) - } - - if encService == nil { - t.Fatal("Encryption service should not be nil") - } -} - -func TestEncryptDecryptRoundtrip(t *testing.T) { - masterKey := "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef" - encService, err := crypto.NewEncryptionService(masterKey) - if err != nil { - t.Fatalf("Failed to create encryption service: %v", err) - } - - userID := "test-user-123" - testCases := []struct { - name string - plaintext string - }{ - {"simple text", "Hello, World!"}, - {"empty string", ""}, - {"json array", `[{"id":"1","content":"test"}]`}, - {"unicode", "Hello, \u4e16\u754c! \U0001F600"}, - {"long text", string(make([]byte, 10000))}, // 10KB of null bytes - } - - for _, tc := range testCases { - t.Run(tc.name, func(t *testing.T) { - // Encrypt - encrypted, err := encService.EncryptString(userID, tc.plaintext) - if err != nil { - if tc.plaintext == "" { - // Empty string returns empty, not error - return - } - t.Fatalf("Encryption failed: %v", err) - } - - // Encrypted should not equal plaintext (unless empty) - if tc.plaintext != "" && encrypted == tc.plaintext { - t.Error("Encrypted text should not equal plaintext") - } - - // Decrypt - decrypted, err := encService.DecryptString(userID, encrypted) - if err != nil { - t.Fatalf("Decryption failed: %v", err) - } - - // Decrypted should equal original - if decrypted != tc.plaintext { - t.Errorf("Decrypted text doesn't match original. Got %q, want %q", decrypted, tc.plaintext) - } - }) - } -} - -func TestDifferentUsersGetDifferentKeys(t *testing.T) { - masterKey := "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef" - encService, err := crypto.NewEncryptionService(masterKey) - if err != nil { - t.Fatalf("Failed to create encryption service: %v", err) - } - - plaintext := "Secret message" - user1 := "user-1" - user2 := "user-2" - - // Encrypt same message for two different users - encrypted1, _ := encService.EncryptString(user1, plaintext) - encrypted2, _ := encService.EncryptString(user2, plaintext) - - // Encrypted values should be different (different user keys + different nonces) - if encrypted1 == encrypted2 { - t.Error("Same plaintext encrypted for different users should produce different ciphertext") - } - - // User 2 should not be able to decrypt user 1's message - decrypted, err := encService.DecryptString(user2, encrypted1) - if err == nil && decrypted == plaintext { - t.Error("User 2 should not be able to decrypt User 1's message") - } -} - -func TestChatMessageSerialization(t *testing.T) { - messages := []models.ChatMessage{ - { - ID: "msg-1", - Role: "user", - Content: "Hello!", - Timestamp: time.Now().UnixMilli(), - }, - { - ID: "msg-2", - Role: "assistant", - Content: "Hi there! How can I help you?", - Timestamp: time.Now().UnixMilli(), - }, - } - - // Serialize - jsonData, err := json.Marshal(messages) - if err != nil { - t.Fatalf("Failed to serialize messages: %v", err) - } - - // Deserialize - var decoded []models.ChatMessage - err = json.Unmarshal(jsonData, &decoded) - if err != nil { - t.Fatalf("Failed to deserialize messages: %v", err) - } - - if len(decoded) != len(messages) { - t.Errorf("Expected %d messages, got %d", len(messages), len(decoded)) - } - - for i, msg := range decoded { - if msg.ID != messages[i].ID { - t.Errorf("Message %d ID mismatch: got %s, want %s", i, msg.ID, messages[i].ID) - } - if msg.Role != messages[i].Role { - t.Errorf("Message %d Role mismatch: got %s, want %s", i, msg.Role, messages[i].Role) - } - if msg.Content != messages[i].Content { - t.Errorf("Message %d Content mismatch: got %s, want %s", i, msg.Content, messages[i].Content) - } - } -} - -func TestChatMessageWithAttachments(t *testing.T) { - message := models.ChatMessage{ - ID: "msg-1", - Role: "user", - Content: "Check out this file", - Timestamp: time.Now().UnixMilli(), - Attachments: []models.ChatAttachment{ - { - FileID: "att-1", - Filename: "document.pdf", - Type: "document", - MimeType: "application/pdf", - Size: 1024, - }, - }, - } - - // Serialize - jsonData, err := json.Marshal(message) - if err != nil { - t.Fatalf("Failed to serialize message with attachment: %v", err) - } - - // Deserialize - var decoded models.ChatMessage - err = json.Unmarshal(jsonData, &decoded) - if err != nil { - t.Fatalf("Failed to deserialize message with attachment: %v", err) - } - - if len(decoded.Attachments) != 1 { - t.Fatalf("Expected 1 attachment, got %d", len(decoded.Attachments)) - } - - att := decoded.Attachments[0] - if att.FileID != "att-1" { - t.Errorf("Attachment FileID mismatch: got %s, want att-1", att.FileID) - } - if att.Filename != "document.pdf" { - t.Errorf("Attachment filename mismatch: got %s, want document.pdf", att.Filename) - } -} - -func TestCreateChatRequestValidation(t *testing.T) { - tests := []struct { - name string - request models.CreateChatRequest - wantErr bool - }{ - { - name: "valid request", - request: models.CreateChatRequest{ - ID: "chat-123", - Title: "Test Chat", - Messages: []models.ChatMessage{ - {ID: "msg-1", Role: "user", Content: "Hello"}, - }, - }, - wantErr: false, - }, - { - name: "empty chat ID", - request: models.CreateChatRequest{ - ID: "", - Title: "Test Chat", - }, - wantErr: true, - }, - { - name: "empty messages", - request: models.CreateChatRequest{ - ID: "chat-123", - Title: "Test Chat", - Messages: []models.ChatMessage{}, - }, - wantErr: false, // Empty messages should be allowed - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - hasErr := tt.request.ID == "" - if hasErr != tt.wantErr { - t.Errorf("Validation error = %v, wantErr %v", hasErr, tt.wantErr) - } - }) - } -} - -func TestBulkSyncRequest(t *testing.T) { - req := models.BulkSyncRequest{ - Chats: []models.CreateChatRequest{ - { - ID: "chat-1", - Title: "Chat 1", - Messages: []models.ChatMessage{ - {ID: "msg-1", Role: "user", Content: "Hello"}, - }, - }, - { - ID: "chat-2", - Title: "Chat 2", - Messages: []models.ChatMessage{ - {ID: "msg-2", Role: "user", Content: "Hi"}, - }, - }, - }, - } - - // Serialize - jsonData, err := json.Marshal(req) - if err != nil { - t.Fatalf("Failed to serialize bulk sync request: %v", err) - } - - // Deserialize - var decoded models.BulkSyncRequest - err = json.Unmarshal(jsonData, &decoded) - if err != nil { - t.Fatalf("Failed to deserialize bulk sync request: %v", err) - } - - if len(decoded.Chats) != 2 { - t.Errorf("Expected 2 chats, got %d", len(decoded.Chats)) - } -} - -func TestChatResponseConversion(t *testing.T) { - response := models.ChatResponse{ - ID: "chat-123", - Title: "Test Chat", - Messages: []models.ChatMessage{{ID: "msg-1", Role: "user", Content: "Hello"}}, - IsStarred: true, - Model: "gpt-4", - Version: 5, - CreatedAt: time.Now(), - UpdatedAt: time.Now(), - } - - // Serialize - jsonData, err := json.Marshal(response) - if err != nil { - t.Fatalf("Failed to serialize chat response: %v", err) - } - - // Deserialize - var decoded models.ChatResponse - err = json.Unmarshal(jsonData, &decoded) - if err != nil { - t.Fatalf("Failed to deserialize chat response: %v", err) - } - - if decoded.ID != response.ID { - t.Errorf("ID mismatch: got %s, want %s", decoded.ID, response.ID) - } - if decoded.Version != response.Version { - t.Errorf("Version mismatch: got %d, want %d", decoded.Version, response.Version) - } - if decoded.IsStarred != response.IsStarred { - t.Errorf("IsStarred mismatch: got %v, want %v", decoded.IsStarred, response.IsStarred) - } -} - -func TestChatListItem(t *testing.T) { - item := models.ChatListItem{ - ID: "chat-123", - Title: "Test Chat", - IsStarred: true, - Model: "gpt-4", - MessageCount: 10, - Version: 3, - CreatedAt: time.Now(), - UpdatedAt: time.Now(), - } - - // Serialize - jsonData, err := json.Marshal(item) - if err != nil { - t.Fatalf("Failed to serialize chat list item: %v", err) - } - - // Check JSON contains expected fields - var jsonMap map[string]interface{} - json.Unmarshal(jsonData, &jsonMap) - - if _, ok := jsonMap["message_count"]; !ok { - t.Error("Expected message_count in JSON output") - } - if _, ok := jsonMap["is_starred"]; !ok { - t.Error("Expected is_starred in JSON output") - } -} - -func TestUpdateChatRequest(t *testing.T) { - title := "New Title" - starred := true - - req := models.UpdateChatRequest{ - Title: &title, - IsStarred: &starred, - Version: 5, - } - - // Serialize - jsonData, err := json.Marshal(req) - if err != nil { - t.Fatalf("Failed to serialize update request: %v", err) - } - - // Deserialize - var decoded models.UpdateChatRequest - err = json.Unmarshal(jsonData, &decoded) - if err != nil { - t.Fatalf("Failed to deserialize update request: %v", err) - } - - if decoded.Title == nil || *decoded.Title != title { - t.Error("Title should be set") - } - if decoded.IsStarred == nil || *decoded.IsStarred != starred { - t.Error("IsStarred should be set") - } - if decoded.Version != 5 { - t.Errorf("Version mismatch: got %d, want 5", decoded.Version) - } -} - -func TestChatAddMessageRequest(t *testing.T) { - req := models.ChatAddMessageRequest{ - Message: models.ChatMessage{ - ID: "msg-new", - Role: "user", - Content: "New message", - Timestamp: time.Now().UnixMilli(), - }, - Version: 3, - } - - // Serialize - jsonData, err := json.Marshal(req) - if err != nil { - t.Fatalf("Failed to serialize add message request: %v", err) - } - - // Deserialize - var decoded models.ChatAddMessageRequest - err = json.Unmarshal(jsonData, &decoded) - if err != nil { - t.Fatalf("Failed to deserialize add message request: %v", err) - } - - if decoded.Message.ID != "msg-new" { - t.Errorf("Message ID mismatch: got %s, want msg-new", decoded.Message.ID) - } - if decoded.Version != 3 { - t.Errorf("Version mismatch: got %d, want 3", decoded.Version) - } -} - -func TestNewChatSyncService(t *testing.T) { - // This test requires MongoDB to be set up - // In a real test environment, you'd use a test MongoDB instance - // For now, we just verify the service constructor doesn't panic with valid inputs - t.Skip("Requires MongoDB connection - run with integration tests") -} - -// ==================== EDGE CASE TESTS ==================== - -func TestEmptyMessagesArray(t *testing.T) { - masterKey := "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef" - encService, _ := crypto.NewEncryptionService(masterKey) - - userID := "user-123" - emptyMessages := []models.ChatMessage{} - - // Serialize empty array - jsonData, err := json.Marshal(emptyMessages) - if err != nil { - t.Fatalf("Failed to serialize empty messages: %v", err) - } - - // Encrypt - encrypted, err := encService.Encrypt(userID, jsonData) - if err != nil { - t.Fatalf("Failed to encrypt empty messages: %v", err) - } - - // Decrypt - decrypted, err := encService.Decrypt(userID, encrypted) - if err != nil { - t.Fatalf("Failed to decrypt: %v", err) - } - - // Deserialize - var recovered []models.ChatMessage - err = json.Unmarshal(decrypted, &recovered) - if err != nil { - t.Fatalf("Failed to deserialize: %v", err) - } - - if len(recovered) != 0 { - t.Errorf("Expected empty array, got %d messages", len(recovered)) - } -} - -func TestNullFieldsInMessage(t *testing.T) { - // Test that null/empty optional fields don't break serialization - message := models.ChatMessage{ - ID: "msg-1", - Role: "user", - Content: "Hello", - Timestamp: 1700000000000, - IsStreaming: false, - Attachments: nil, // nil attachments - AgentId: "", // empty agent id - AgentName: "", - AgentAvatar: "", - } - - jsonData, err := json.Marshal(message) - if err != nil { - t.Fatalf("Failed to serialize message with null fields: %v", err) - } - - var recovered models.ChatMessage - err = json.Unmarshal(jsonData, &recovered) - if err != nil { - t.Fatalf("Failed to deserialize: %v", err) - } - - if recovered.ID != message.ID { - t.Error("ID mismatch after roundtrip") - } - if recovered.Attachments != nil { - t.Error("Expected nil attachments") - } -} - -func TestSpecialCharactersInContent(t *testing.T) { - masterKey := "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef" - encService, _ := crypto.NewEncryptionService(masterKey) - - testCases := []string{ - "Hello\nWorld", // Newlines - "Tab\there", // Tabs - "Quote: \"test\"", // Quotes - "Backslash: \\path\\to\\file", // Backslashes - "Unicode: \u4e2d\u6587", // Chinese characters - "Emoji: \U0001F600\U0001F389", // Emojis - "HTML: ", // HTML - "SQL: SELECT * FROM users; DROP TABLE--", // SQL injection attempt - "Null byte: \x00", // Null byte - "Control chars: \x01\x02\x03", // Control characters - string(make([]byte, 100000)), // Large content (100KB) - } - - userID := "user-123" - - for i, content := range testCases { - t.Run("case_"+string(rune('0'+i)), func(t *testing.T) { - message := models.ChatMessage{ - ID: "msg-1", - Role: "user", - Content: content, - Timestamp: 1700000000000, - } - - // Serialize - jsonData, err := json.Marshal(message) - if err != nil { - t.Fatalf("Failed to serialize: %v", err) - } - - // Encrypt - encrypted, err := encService.Encrypt(userID, jsonData) - if err != nil { - t.Fatalf("Failed to encrypt: %v", err) - } - - // Decrypt - decrypted, err := encService.Decrypt(userID, encrypted) - if err != nil { - t.Fatalf("Failed to decrypt: %v", err) - } - - // Deserialize - var recovered models.ChatMessage - err = json.Unmarshal(decrypted, &recovered) - if err != nil { - t.Fatalf("Failed to deserialize: %v", err) - } - - if recovered.Content != content { - t.Errorf("Content mismatch after roundtrip") - } - }) - } -} - -func TestMaxMessageCount(t *testing.T) { - // Test with a large number of messages (stress test) - messageCount := 1000 - messages := make([]models.ChatMessage, messageCount) - - for i := 0; i < messageCount; i++ { - role := "user" - if i%2 == 1 { - role = "assistant" - } - messages[i] = models.ChatMessage{ - ID: "msg-" + string(rune('0'+i%10)) + string(rune('0'+i/10)), - Role: role, - Content: "This is message number " + string(rune('0'+i)), - Timestamp: int64(1700000000000 + i*1000), - } - } - - masterKey := "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef" - encService, _ := crypto.NewEncryptionService(masterKey) - userID := "user-123" - - // Serialize - jsonData, err := json.Marshal(messages) - if err != nil { - t.Fatalf("Failed to serialize %d messages: %v", messageCount, err) - } - - // Encrypt - encrypted, err := encService.Encrypt(userID, jsonData) - if err != nil { - t.Fatalf("Failed to encrypt: %v", err) - } - - // Decrypt - decrypted, err := encService.Decrypt(userID, encrypted) - if err != nil { - t.Fatalf("Failed to decrypt: %v", err) - } - - // Deserialize - var recovered []models.ChatMessage - err = json.Unmarshal(decrypted, &recovered) - if err != nil { - t.Fatalf("Failed to deserialize: %v", err) - } - - if len(recovered) != messageCount { - t.Errorf("Expected %d messages, got %d", messageCount, len(recovered)) - } -} - -func TestAttachmentTypes(t *testing.T) { - // Test all attachment types - attachments := []models.ChatAttachment{ - { - FileID: "att-1", - Filename: "document.pdf", - Type: "document", - MimeType: "application/pdf", - Size: 1024, - URL: "https://example.com/file.pdf", - }, - { - FileID: "att-2", - Filename: "image.png", - Type: "image", - MimeType: "image/png", - Size: 2048, - Preview: "base64encodedcontent", - }, - { - FileID: "att-3", - Filename: "data.csv", - Type: "data", - MimeType: "text/csv", - Size: 512, - }, - } - - message := models.ChatMessage{ - ID: "msg-1", - Role: "user", - Content: "Check these files", - Timestamp: 1700000000000, - Attachments: attachments, - } - - jsonData, err := json.Marshal(message) - if err != nil { - t.Fatalf("Failed to serialize: %v", err) - } - - var recovered models.ChatMessage - err = json.Unmarshal(jsonData, &recovered) - if err != nil { - t.Fatalf("Failed to deserialize: %v", err) - } - - if len(recovered.Attachments) != len(attachments) { - t.Fatalf("Expected %d attachments, got %d", len(attachments), len(recovered.Attachments)) - } - - for i, att := range recovered.Attachments { - if att.FileID != attachments[i].FileID { - t.Errorf("Attachment %d: FileID mismatch", i) - } - if att.Type != attachments[i].Type { - t.Errorf("Attachment %d: Type mismatch", i) - } - if att.Size != attachments[i].Size { - t.Errorf("Attachment %d: Size mismatch", i) - } - } -} - -// ==================== BACKWARD COMPATIBILITY TESTS ==================== - -func TestBackwardCompatibility_OldMessageFormat(t *testing.T) { - // Simulate old message format that might exist in localStorage - // This tests that the backend can handle messages without all new fields - oldFormatJSON := `{ - "id": "msg-1", - "role": "user", - "content": "Hello", - "timestamp": 1700000000000 - }` - - var message models.ChatMessage - err := json.Unmarshal([]byte(oldFormatJSON), &message) - if err != nil { - t.Fatalf("Failed to parse old format: %v", err) - } - - if message.ID != "msg-1" { - t.Error("ID mismatch") - } - if message.Role != "user" { - t.Error("Role mismatch") - } - if message.Attachments != nil { - t.Error("Attachments should be nil for old format") - } -} - -func TestBackwardCompatibility_OldChatFormat(t *testing.T) { - // Test parsing of chat without newer optional fields - oldFormatJSON := `{ - "id": "chat-123", - "title": "Test Chat", - "messages": [ - {"id": "msg-1", "role": "user", "content": "Hello", "timestamp": 1700000000000} - ] - }` - - var req models.CreateChatRequest - err := json.Unmarshal([]byte(oldFormatJSON), &req) - if err != nil { - t.Fatalf("Failed to parse old format: %v", err) - } - - if req.ID != "chat-123" { - t.Error("ID mismatch") - } - if req.IsStarred != false { - t.Error("IsStarred should default to false") - } - if req.Model != "" { - t.Error("Model should be empty for old format") - } -} - -func TestBackwardCompatibility_VersionZero(t *testing.T) { - // Test that version 0 (unset) is handled correctly - req := models.CreateChatRequest{ - ID: "chat-123", - Title: "Test", - Messages: []models.ChatMessage{}, - Version: 0, // Explicitly zero (or unset) - } - - jsonData, err := json.Marshal(req) - if err != nil { - t.Fatalf("Failed to marshal: %v", err) - } - - var parsed models.CreateChatRequest - err = json.Unmarshal(jsonData, &parsed) - if err != nil { - t.Fatalf("Failed to unmarshal: %v", err) - } - - if parsed.Version != 0 { - t.Errorf("Version should be 0, got %d", parsed.Version) - } -} - -func TestBackwardCompatibility_MixedTimestampFormats(t *testing.T) { - // Test both Unix milliseconds (new) and potential variations - testCases := []struct { - name string - timestamp int64 - }{ - {"current timestamp", 1700000000000}, - {"zero timestamp", 0}, - {"far future", 9999999999999}, - {"year 2000", 946684800000}, - } - - for _, tc := range testCases { - t.Run(tc.name, func(t *testing.T) { - message := models.ChatMessage{ - ID: "msg-1", - Role: "user", - Content: "Test", - Timestamp: tc.timestamp, - } - - jsonData, _ := json.Marshal(message) - var recovered models.ChatMessage - json.Unmarshal(jsonData, &recovered) - - if recovered.Timestamp != tc.timestamp { - t.Errorf("Timestamp mismatch: expected %d, got %d", tc.timestamp, recovered.Timestamp) - } - }) - } -} - -// ==================== VERSION CONFLICT TESTS ==================== - -func TestVersionConflictDetection(t *testing.T) { - // Test that version numbers work correctly for conflict detection - req1 := models.CreateChatRequest{ - ID: "chat-123", - Title: "Original", - Messages: []models.ChatMessage{}, - Version: 1, - } - - req2 := models.CreateChatRequest{ - ID: "chat-123", - Title: "Updated", - Messages: []models.ChatMessage{}, - Version: 2, - } - - // Simulate version check - if req2.Version <= req1.Version { - t.Error("req2 should have higher version than req1") - } -} - -func TestUpdateRequestPartialFields(t *testing.T) { - // Test that partial updates work correctly - title := "New Title" - req := models.UpdateChatRequest{ - Title: &title, - Version: 5, - } - - if req.IsStarred != nil { - t.Error("IsStarred should be nil (not set)") - } - if req.Model != nil { - t.Error("Model should be nil (not set)") - } - if *req.Title != title { - t.Error("Title should be set") - } -} - -// ==================== ENCRYPTION SECURITY TESTS ==================== - -func TestEncryptionDeterminism(t *testing.T) { - // Verify that same plaintext produces different ciphertext each time (due to random nonce) - masterKey := "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef" - encService, _ := crypto.NewEncryptionService(masterKey) - - plaintext := "Same message" - userID := "user-123" - - encrypted1, _ := encService.EncryptString(userID, plaintext) - encrypted2, _ := encService.EncryptString(userID, plaintext) - - if encrypted1 == encrypted2 { - t.Error("Same plaintext should produce different ciphertext (random nonce)") - } - - // But both should decrypt to the same value - decrypted1, _ := encService.DecryptString(userID, encrypted1) - decrypted2, _ := encService.DecryptString(userID, encrypted2) - - if decrypted1 != decrypted2 || decrypted1 != plaintext { - t.Error("Both encryptions should decrypt to the same plaintext") - } -} - -func TestDecryptionWithWrongKey(t *testing.T) { - masterKey1 := "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef" - masterKey2 := "fedcba9876543210fedcba9876543210fedcba9876543210fedcba9876543210" - - encService1, _ := crypto.NewEncryptionService(masterKey1) - encService2, _ := crypto.NewEncryptionService(masterKey2) - - plaintext := "Secret message" - userID := "user-123" - - encrypted, _ := encService1.EncryptString(userID, plaintext) - - // Try to decrypt with different master key - should fail - _, err := encService2.DecryptString(userID, encrypted) - if err == nil { - t.Error("Decryption with wrong key should fail") - } -} - -func TestTamperedCiphertext(t *testing.T) { - masterKey := "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef" - encService, err := crypto.NewEncryptionService(masterKey) - if err != nil { - t.Fatalf("Failed to create encryption service: %v", err) - } - - plaintext := "Secret message" - userID := "user-123" - - encrypted, err := encService.EncryptString(userID, plaintext) - if err != nil { - t.Fatalf("Failed to encrypt: %v", err) - } - - if len(encrypted) < 20 { - t.Skip("Encrypted string too short for tampering test") - } - - // Tamper with the ciphertext - flip multiple characters to ensure GCM detects it - tampered := encrypted[:len(encrypted)-5] + "XXXXX" - - _, err = encService.DecryptString(userID, tampered) - if err == nil { - t.Error("Tampered ciphertext should fail to decrypt") - } -} - -func TestFullEncryptionDecryptionPipeline(t *testing.T) { - // Simulate the full pipeline: messages -> JSON -> encrypt -> decrypt -> JSON -> messages - masterKey := "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef" - encService, err := crypto.NewEncryptionService(masterKey) - if err != nil { - t.Fatalf("Failed to create encryption service: %v", err) - } - - userID := "user-123" - - // Create test messages - originalMessages := []models.ChatMessage{ - { - ID: "msg-1", - Role: "user", - Content: "What's the weather like?", - Timestamp: 1700000000000, - }, - { - ID: "msg-2", - Role: "assistant", - Content: "I'd be happy to help with weather information! However, I don't have access to real-time weather data.", - Timestamp: 1700000001000, - }, - { - ID: "msg-3", - Role: "user", - Content: "Thanks!", - Timestamp: 1700000002000, - Attachments: []models.ChatAttachment{ - {FileID: "att-1", Filename: "image.png", Type: "image", MimeType: "image/png", Size: 5000}, - }, - }, - } - - // Step 1: Serialize to JSON - jsonData, err := json.Marshal(originalMessages) - if err != nil { - t.Fatalf("Failed to serialize messages: %v", err) - } - - // Step 2: Encrypt - encrypted, err := encService.Encrypt(userID, jsonData) - if err != nil { - t.Fatalf("Failed to encrypt: %v", err) - } - - // Step 3: Decrypt - decrypted, err := encService.Decrypt(userID, encrypted) - if err != nil { - t.Fatalf("Failed to decrypt: %v", err) - } - - // Step 4: Deserialize - var recoveredMessages []models.ChatMessage - err = json.Unmarshal(decrypted, &recoveredMessages) - if err != nil { - t.Fatalf("Failed to deserialize messages: %v", err) - } - - // Verify all messages match - if len(recoveredMessages) != len(originalMessages) { - t.Fatalf("Message count mismatch: got %d, want %d", len(recoveredMessages), len(originalMessages)) - } - - for i, original := range originalMessages { - recovered := recoveredMessages[i] - if recovered.ID != original.ID { - t.Errorf("Message %d: ID mismatch", i) - } - if recovered.Role != original.Role { - t.Errorf("Message %d: Role mismatch", i) - } - if recovered.Content != original.Content { - t.Errorf("Message %d: Content mismatch", i) - } - if recovered.Timestamp != original.Timestamp { - t.Errorf("Message %d: Timestamp mismatch", i) - } - if len(recovered.Attachments) != len(original.Attachments) { - t.Errorf("Message %d: Attachment count mismatch", i) - } - } -} diff --git a/backend/internal/services/config_service.go b/backend/internal/services/config_service.go deleted file mode 100644 index 1e3bead3..00000000 --- a/backend/internal/services/config_service.go +++ /dev/null @@ -1,150 +0,0 @@ -package services - -import ( - "claraverse/internal/models" - "log" - "sync" -) - -// ConfigService handles configuration management -type ConfigService struct { - mu sync.RWMutex - recommendedModels map[int]*models.RecommendedModels // Provider ID -> Recommended Models - modelAliases map[int]map[string]models.ModelAlias // Provider ID -> (Model Name -> Alias Info) - providerSecurity map[int]bool // Provider ID -> Secure flag -} - -var ( - configServiceInstance *ConfigService - configServiceOnce sync.Once -) - -// GetConfigService returns the singleton config service instance -func GetConfigService() *ConfigService { - configServiceOnce.Do(func() { - configServiceInstance = &ConfigService{ - recommendedModels: make(map[int]*models.RecommendedModels), - modelAliases: make(map[int]map[string]models.ModelAlias), - providerSecurity: make(map[int]bool), - } - }) - return configServiceInstance -} - -// SetRecommendedModels stores recommended models for a provider -func (s *ConfigService) SetRecommendedModels(providerID int, recommended *models.RecommendedModels) { - if recommended == nil { - return - } - - s.mu.Lock() - defer s.mu.Unlock() - - s.recommendedModels[providerID] = recommended - log.Printf("📌 [CONFIG] Set recommended models for provider %d: top=%s, medium=%s, fastest=%s", - providerID, recommended.Top, recommended.Medium, recommended.Fastest) -} - -// GetRecommendedModels retrieves recommended models for a provider -func (s *ConfigService) GetRecommendedModels(providerID int) *models.RecommendedModels { - s.mu.RLock() - defer s.mu.RUnlock() - - return s.recommendedModels[providerID] -} - -// GetAllRecommendedModels retrieves all recommended models across providers -func (s *ConfigService) GetAllRecommendedModels() map[int]*models.RecommendedModels { - s.mu.RLock() - defer s.mu.RUnlock() - - // Create a copy to avoid race conditions - result := make(map[int]*models.RecommendedModels) - for k, v := range s.recommendedModels { - result[k] = v - } - - return result -} - -// SetModelAliases stores model aliases for a provider -func (s *ConfigService) SetModelAliases(providerID int, aliases map[string]models.ModelAlias) { - if aliases == nil || len(aliases) == 0 { - return - } - - s.mu.Lock() - defer s.mu.Unlock() - - s.modelAliases[providerID] = aliases - log.Printf("📌 [CONFIG] Set %d model aliases for provider %d", len(aliases), providerID) -} - -// GetModelAliases retrieves model aliases for a provider -func (s *ConfigService) GetModelAliases(providerID int) map[string]models.ModelAlias { - s.mu.RLock() - defer s.mu.RUnlock() - - return s.modelAliases[providerID] -} - -// GetAllModelAliases retrieves all model aliases across providers -func (s *ConfigService) GetAllModelAliases() map[int]map[string]models.ModelAlias { - s.mu.RLock() - defer s.mu.RUnlock() - - // Create a copy to avoid race conditions - result := make(map[int]map[string]models.ModelAlias) - for k, v := range s.modelAliases { - result[k] = v - } - - return result -} - -// GetAliasForModel checks if a model has an alias configured -func (s *ConfigService) GetAliasForModel(providerID int, modelName string) *models.ModelAlias { - s.mu.RLock() - defer s.mu.RUnlock() - - if aliases, exists := s.modelAliases[providerID]; exists { - if alias, found := aliases[modelName]; found { - return &alias - } - } - - return nil -} - -// GetModelAlias retrieves a specific alias by its key for a provider -func (s *ConfigService) GetModelAlias(providerID int, aliasKey string) *models.ModelAlias { - s.mu.RLock() - defer s.mu.RUnlock() - - if aliases, exists := s.modelAliases[providerID]; exists { - if alias, found := aliases[aliasKey]; found { - return &alias - } - } - - return nil -} - -// SetProviderSecure stores the secure flag for a provider -func (s *ConfigService) SetProviderSecure(providerID int, secure bool) { - s.mu.Lock() - defer s.mu.Unlock() - - s.providerSecurity[providerID] = secure - if secure { - log.Printf("🔒 [CONFIG] Provider %d marked as secure (doesn't store user data)", providerID) - } -} - -// IsProviderSecure checks if a provider is marked as secure -func (s *ConfigService) IsProviderSecure(providerID int) bool { - s.mu.RLock() - defer s.mu.RUnlock() - - return s.providerSecurity[providerID] -} diff --git a/backend/internal/services/connection_manager.go b/backend/internal/services/connection_manager.go deleted file mode 100644 index a3a32a84..00000000 --- a/backend/internal/services/connection_manager.go +++ /dev/null @@ -1,67 +0,0 @@ -package services - -import ( - "claraverse/internal/models" - "log" - "sync" -) - -// ConnectionManager manages all active WebSocket connections -type ConnectionManager struct { - connections map[string]*models.UserConnection - mutex sync.RWMutex -} - -// NewConnectionManager creates a new connection manager -func NewConnectionManager() *ConnectionManager { - return &ConnectionManager{ - connections: make(map[string]*models.UserConnection), - } -} - -// Add adds a new connection -func (cm *ConnectionManager) Add(conn *models.UserConnection) { - cm.mutex.Lock() - defer cm.mutex.Unlock() - cm.connections[conn.ConnID] = conn - log.Printf("✅ Connection added: %s (Total: %d)", conn.ConnID, len(cm.connections)) -} - -// Remove removes a connection -func (cm *ConnectionManager) Remove(connID string) { - cm.mutex.Lock() - defer cm.mutex.Unlock() - if conn, exists := cm.connections[connID]; exists { - close(conn.WriteChan) - close(conn.StopChan) - delete(cm.connections, connID) - log.Printf("❌ Connection removed: %s (Total: %d)", connID, len(cm.connections)) - } -} - -// Get retrieves a connection by ID -func (cm *ConnectionManager) Get(connID string) (*models.UserConnection, bool) { - cm.mutex.RLock() - defer cm.mutex.RUnlock() - conn, exists := cm.connections[connID] - return conn, exists -} - -// Count returns the number of active connections -func (cm *ConnectionManager) Count() int { - cm.mutex.RLock() - defer cm.mutex.RUnlock() - return len(cm.connections) -} - -// GetAll returns all active connections -func (cm *ConnectionManager) GetAll() []*models.UserConnection { - cm.mutex.RLock() - defer cm.mutex.RUnlock() - - conns := make([]*models.UserConnection, 0, len(cm.connections)) - for _, conn := range cm.connections { - conns = append(conns, conn) - } - return conns -} diff --git a/backend/internal/services/credential_service.go b/backend/internal/services/credential_service.go deleted file mode 100644 index 2d473aae..00000000 --- a/backend/internal/services/credential_service.go +++ /dev/null @@ -1,622 +0,0 @@ -package services - -import ( - "claraverse/internal/crypto" - "claraverse/internal/database" - "claraverse/internal/models" - "context" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "os" - "time" - - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/bson/primitive" - "go.mongodb.org/mongo-driver/mongo" - "go.mongodb.org/mongo-driver/mongo/options" -) - -const ( - // CollectionCredentials is the MongoDB collection name - CollectionCredentials = "credentials" -) - -// CredentialService manages encrypted credentials for integrations -type CredentialService struct { - mongoDB *database.MongoDB - encryption *crypto.EncryptionService -} - -// NewCredentialService creates a new credential service -func NewCredentialService(mongoDB *database.MongoDB, encryption *crypto.EncryptionService) *CredentialService { - return &CredentialService{ - mongoDB: mongoDB, - encryption: encryption, - } -} - -// collection returns the credentials collection -func (s *CredentialService) collection() *mongo.Collection { - return s.mongoDB.Database().Collection(CollectionCredentials) -} - -// Create creates a new credential with encrypted data -func (s *CredentialService) Create(ctx context.Context, userID string, req *models.CreateCredentialRequest) (*models.CredentialListItem, error) { - // Validate integration type exists - integration, exists := models.GetIntegration(req.IntegrationType) - if !exists { - return nil, fmt.Errorf("unknown integration type: %s", req.IntegrationType) - } - - // Validate required fields - if err := models.ValidateCredentialData(req.IntegrationType, req.Data); err != nil { - return nil, err - } - - // Serialize data to JSON - dataJSON, err := json.Marshal(req.Data) - if err != nil { - return nil, fmt.Errorf("failed to serialize credential data: %w", err) - } - - // Encrypt the data - encryptedData, err := s.encryption.Encrypt(userID, dataJSON) - if err != nil { - return nil, fmt.Errorf("failed to encrypt credential data: %w", err) - } - - // Generate masked preview - maskedPreview := models.GenerateMaskedPreview(req.IntegrationType, req.Data) - - now := time.Now() - credential := &models.Credential{ - UserID: userID, - Name: req.Name, - IntegrationType: req.IntegrationType, - EncryptedData: encryptedData, - Metadata: models.CredentialMetadata{ - MaskedPreview: maskedPreview, - Icon: integration.Icon, - UsageCount: 0, - TestStatus: "pending", - }, - CreatedAt: now, - UpdatedAt: now, - } - - result, err := s.collection().InsertOne(ctx, credential) - if err != nil { - return nil, fmt.Errorf("failed to create credential: %w", err) - } - - credential.ID = result.InsertedID.(primitive.ObjectID) - - log.Printf("🔐 [CREDENTIAL] Created credential %s (%s) for user %s", - credential.ID.Hex(), req.IntegrationType, userID) - - return credential.ToListItem(), nil -} - -// GetByID retrieves a credential by ID (metadata only, no decryption) -func (s *CredentialService) GetByID(ctx context.Context, credentialID primitive.ObjectID) (*models.Credential, error) { - var credential models.Credential - err := s.collection().FindOne(ctx, bson.M{"_id": credentialID}).Decode(&credential) - if err != nil { - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("credential not found") - } - return nil, fmt.Errorf("failed to get credential: %w", err) - } - return &credential, nil -} - -// GetByIDAndUser retrieves a credential ensuring user ownership -func (s *CredentialService) GetByIDAndUser(ctx context.Context, credentialID primitive.ObjectID, userID string) (*models.Credential, error) { - var credential models.Credential - err := s.collection().FindOne(ctx, bson.M{ - "_id": credentialID, - "userId": userID, - }).Decode(&credential) - if err != nil { - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("credential not found") - } - return nil, fmt.Errorf("failed to get credential: %w", err) - } - return &credential, nil -} - -// GetDecrypted retrieves and decrypts a credential for tool use -// SECURITY: This should ONLY be called by tools, never exposed to API/LLM -func (s *CredentialService) GetDecrypted(ctx context.Context, userID string, credentialID primitive.ObjectID) (*models.DecryptedCredential, error) { - // Get the credential with ownership verification - credential, err := s.GetByIDAndUser(ctx, credentialID, userID) - if err != nil { - return nil, err - } - - // Decrypt the data - decryptedJSON, err := s.encryption.Decrypt(userID, credential.EncryptedData) - if err != nil { - log.Printf("⚠️ [CREDENTIAL] Decryption failed for credential %s: %v", credentialID.Hex(), err) - return nil, fmt.Errorf("failed to decrypt credential") - } - - // Parse JSON - var data map[string]interface{} - if err := json.Unmarshal(decryptedJSON, &data); err != nil { - return nil, fmt.Errorf("failed to parse credential data: %w", err) - } - - // Update usage stats asynchronously - go s.updateUsageStats(context.Background(), credentialID) - - return &models.DecryptedCredential{ - ID: credential.ID.Hex(), - Name: credential.Name, - IntegrationType: credential.IntegrationType, - Data: data, - }, nil -} - -// GetDecryptedByName retrieves and decrypts a credential by name for tool use -// SECURITY: This should ONLY be called by tools, never exposed to API/LLM -func (s *CredentialService) GetDecryptedByName(ctx context.Context, userID string, integrationType string, name string) (*models.DecryptedCredential, error) { - var credential models.Credential - err := s.collection().FindOne(ctx, bson.M{ - "userId": userID, - "integrationType": integrationType, - "name": name, - }).Decode(&credential) - if err != nil { - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("credential not found") - } - return nil, fmt.Errorf("failed to get credential: %w", err) - } - - // Decrypt the data - decryptedJSON, err := s.encryption.Decrypt(userID, credential.EncryptedData) - if err != nil { - log.Printf("⚠️ [CREDENTIAL] Decryption failed for credential %s: %v", credential.ID.Hex(), err) - return nil, fmt.Errorf("failed to decrypt credential") - } - - // Parse JSON - var data map[string]interface{} - if err := json.Unmarshal(decryptedJSON, &data); err != nil { - return nil, fmt.Errorf("failed to parse credential data: %w", err) - } - - // Update usage stats asynchronously - go s.updateUsageStats(context.Background(), credential.ID) - - return &models.DecryptedCredential{ - ID: credential.ID.Hex(), - Name: credential.Name, - IntegrationType: credential.IntegrationType, - Data: data, - }, nil -} - -// ListByUser returns all credentials for a user (metadata only) -func (s *CredentialService) ListByUser(ctx context.Context, userID string) ([]*models.CredentialListItem, error) { - cursor, err := s.collection().Find(ctx, bson.M{ - "userId": userID, - }, options.Find().SetSort(bson.D{ - {Key: "integrationType", Value: 1}, - {Key: "name", Value: 1}, - })) - if err != nil { - return nil, fmt.Errorf("failed to list credentials: %w", err) - } - defer cursor.Close(ctx) - - var credentials []*models.CredentialListItem - for cursor.Next(ctx) { - var cred models.Credential - if err := cursor.Decode(&cred); err != nil { - continue - } - credentials = append(credentials, cred.ToListItem()) - } - - if credentials == nil { - credentials = []*models.CredentialListItem{} - } - - return credentials, nil -} - -// ListByUserAndType returns credentials for a specific integration type -func (s *CredentialService) ListByUserAndType(ctx context.Context, userID string, integrationType string) ([]*models.CredentialListItem, error) { - cursor, err := s.collection().Find(ctx, bson.M{ - "userId": userID, - "integrationType": integrationType, - }, options.Find().SetSort(bson.D{{Key: "name", Value: 1}})) - if err != nil { - return nil, fmt.Errorf("failed to list credentials: %w", err) - } - defer cursor.Close(ctx) - - var credentials []*models.CredentialListItem - for cursor.Next(ctx) { - var cred models.Credential - if err := cursor.Decode(&cred); err != nil { - continue - } - credentials = append(credentials, cred.ToListItem()) - } - - if credentials == nil { - credentials = []*models.CredentialListItem{} - } - - return credentials, nil -} - -// Update updates a credential's name and/or data -func (s *CredentialService) Update(ctx context.Context, credentialID primitive.ObjectID, userID string, req *models.UpdateCredentialRequest) (*models.CredentialListItem, error) { - // Get existing credential - credential, err := s.GetByIDAndUser(ctx, credentialID, userID) - if err != nil { - return nil, err - } - - updateFields := bson.M{ - "updatedAt": time.Now(), - } - - // Update name if provided - if req.Name != "" { - updateFields["name"] = req.Name - } - - // Update data if provided (requires re-encryption) - if req.Data != nil { - // Validate the new data - if err := models.ValidateCredentialData(credential.IntegrationType, req.Data); err != nil { - return nil, err - } - - // Serialize and encrypt new data - dataJSON, err := json.Marshal(req.Data) - if err != nil { - return nil, fmt.Errorf("failed to serialize credential data: %w", err) - } - - encryptedData, err := s.encryption.Encrypt(userID, dataJSON) - if err != nil { - return nil, fmt.Errorf("failed to encrypt credential data: %w", err) - } - - updateFields["encryptedData"] = encryptedData - updateFields["metadata.maskedPreview"] = models.GenerateMaskedPreview(credential.IntegrationType, req.Data) - updateFields["metadata.testStatus"] = "pending" // Reset test status - } - - _, err = s.collection().UpdateByID(ctx, credentialID, bson.M{"$set": updateFields}) - if err != nil { - return nil, fmt.Errorf("failed to update credential: %w", err) - } - - // Get updated credential - updated, err := s.GetByIDAndUser(ctx, credentialID, userID) - if err != nil { - return nil, err - } - - log.Printf("📝 [CREDENTIAL] Updated credential %s for user %s", credentialID.Hex(), userID) - - return updated.ToListItem(), nil -} - -// Delete permanently deletes a credential -func (s *CredentialService) Delete(ctx context.Context, credentialID primitive.ObjectID, userID string) error { - // CRITICAL: Get credential data BEFORE deletion to revoke Composio connections - credential, err := s.GetByIDAndUser(ctx, credentialID, userID) - if err != nil { - return err - } - - // Revoke Composio OAuth if this is a Composio integration - if err := s.revokeComposioIfNeeded(ctx, credential); err != nil { - log.Printf("⚠️ [CREDENTIAL] Failed to revoke Composio connection for %s: %v", credentialID.Hex(), err) - // Continue with deletion even if revocation fails (connection might already be invalid) - } - - result, err := s.collection().DeleteOne(ctx, bson.M{ - "_id": credentialID, - "userId": userID, - }) - if err != nil { - return fmt.Errorf("failed to delete credential: %w", err) - } - - if result.DeletedCount == 0 { - return fmt.Errorf("credential not found") - } - - log.Printf("🗑️ [CREDENTIAL] Deleted credential %s for user %s", credentialID.Hex(), userID) - return nil -} - -// DeleteAllByUser deletes all credentials for a user (for account deletion) -func (s *CredentialService) DeleteAllByUser(ctx context.Context, userID string) (int64, error) { - result, err := s.collection().DeleteMany(ctx, bson.M{ - "userId": userID, - }) - if err != nil { - return 0, fmt.Errorf("failed to delete credentials: %w", err) - } - - log.Printf("🗑️ [CREDENTIAL] Deleted %d credentials for user %s", result.DeletedCount, userID) - return result.DeletedCount, nil -} - -// UpdateTestStatus updates the test status of a credential -func (s *CredentialService) UpdateTestStatus(ctx context.Context, credentialID primitive.ObjectID, userID string, status string, err error) error { - updateFields := bson.M{ - "metadata.testStatus": status, - "metadata.lastTestAt": time.Now(), - "updatedAt": time.Now(), - } - - _, updateErr := s.collection().UpdateOne(ctx, bson.M{ - "_id": credentialID, - "userId": userID, - }, bson.M{"$set": updateFields}) - if updateErr != nil { - return fmt.Errorf("failed to update test status: %w", updateErr) - } - - return nil -} - -// updateUsageStats updates the usage statistics for a credential -func (s *CredentialService) updateUsageStats(ctx context.Context, credentialID primitive.ObjectID) { - _, err := s.collection().UpdateByID(ctx, credentialID, bson.M{ - "$set": bson.M{ - "metadata.lastUsedAt": time.Now(), - }, - "$inc": bson.M{ - "metadata.usageCount": 1, - }, - }) - if err != nil { - log.Printf("⚠️ [CREDENTIAL] Failed to update usage stats: %v", err) - } -} - -// CountByUser counts credentials for a user -func (s *CredentialService) CountByUser(ctx context.Context, userID string) (int64, error) { - count, err := s.collection().CountDocuments(ctx, bson.M{ - "userId": userID, - }) - if err != nil { - return 0, fmt.Errorf("failed to count credentials: %w", err) - } - return count, nil -} - -// CountByUserAndType counts credentials for a user by type -func (s *CredentialService) CountByUserAndType(ctx context.Context, userID string, integrationType string) (int64, error) { - count, err := s.collection().CountDocuments(ctx, bson.M{ - "userId": userID, - "integrationType": integrationType, - }) - if err != nil { - return 0, fmt.Errorf("failed to count credentials: %w", err) - } - return count, nil -} - -// GetCredentialReferences returns credential references for use in LLM context -// This returns only names and IDs, safe to show to LLM -func (s *CredentialService) GetCredentialReferences(ctx context.Context, userID string, integrationTypes []string) ([]models.CredentialReference, error) { - filter := bson.M{"userId": userID} - if len(integrationTypes) > 0 { - filter["integrationType"] = bson.M{"$in": integrationTypes} - } - - cursor, err := s.collection().Find(ctx, filter, options.Find(). - SetProjection(bson.M{ - "_id": 1, - "name": 1, - "integrationType": 1, - }). - SetSort(bson.D{ - {Key: "integrationType", Value: 1}, - {Key: "name", Value: 1}, - })) - if err != nil { - return nil, fmt.Errorf("failed to get credential references: %w", err) - } - defer cursor.Close(ctx) - - var refs []models.CredentialReference - for cursor.Next(ctx) { - var cred struct { - ID primitive.ObjectID `bson:"_id"` - Name string `bson:"name"` - IntegrationType string `bson:"integrationType"` - } - if err := cursor.Decode(&cred); err != nil { - continue - } - refs = append(refs, models.CredentialReference{ - ID: cred.ID.Hex(), - Name: cred.Name, - IntegrationType: cred.IntegrationType, - }) - } - - if refs == nil { - refs = []models.CredentialReference{} - } - - return refs, nil -} - -// revokeComposioIfNeeded revokes Composio OAuth connection when deleting a Composio credential -func (s *CredentialService) revokeComposioIfNeeded(ctx context.Context, credential *models.Credential) error { - // Only revoke if this is a Composio integration - if len(credential.IntegrationType) < 9 || credential.IntegrationType[:9] != "composio_" { - return nil // Not a Composio integration - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return fmt.Errorf("COMPOSIO_API_KEY not set") - } - - // Decrypt credential data to get entity_id - decryptedJSON, err := s.encryption.Decrypt(credential.UserID, credential.EncryptedData) - if err != nil { - return fmt.Errorf("failed to decrypt credential: %w", err) - } - - var data map[string]interface{} - if err := json.Unmarshal(decryptedJSON, &data); err != nil { - return fmt.Errorf("failed to parse credential data: %w", err) - } - - entityID, ok := data["composio_entity_id"].(string) - if !ok || entityID == "" { - return fmt.Errorf("no composio_entity_id found") - } - - // Extract app name from integration type (e.g., "composio_gmail" -> "gmail") - appName := credential.IntegrationType[9:] // Remove "composio_" prefix - - // Get connected account ID from Composio v3 API - connectedAccountID, err := s.getComposioConnectedAccountID(ctx, composioAPIKey, entityID, appName) - if err != nil { - return fmt.Errorf("failed to get connected account: %w", err) - } - - // Delete the connected account (revokes OAuth) - deleteURL := fmt.Sprintf("https://backend.composio.dev/api/v3/connected_accounts/%s", connectedAccountID) - req, err := http.NewRequestWithContext(ctx, "DELETE", deleteURL, nil) - if err != nil { - return fmt.Errorf("failed to create delete request: %w", err) - } - - req.Header.Set("x-api-key", composioAPIKey) - - client := &http.Client{Timeout: 10 * time.Second} - resp, err := client.Do(req) - if err != nil { - return fmt.Errorf("failed to delete connection: %w", err) - } - defer resp.Body.Close() - - if resp.StatusCode >= 400 && resp.StatusCode != 404 { - bodyBytes, _ := io.ReadAll(resp.Body) - return fmt.Errorf("Composio API error (status %d): %s", resp.StatusCode, string(bodyBytes)) - } - - log.Printf("✅ [COMPOSIO] Revoked %s connection for entity %s", appName, entityID) - return nil -} - -// getComposioConnectedAccountID retrieves the connected account ID from Composio v3 API -func (s *CredentialService) getComposioConnectedAccountID(ctx context.Context, apiKey string, entityID string, appName string) (string, error) { - url := fmt.Sprintf("https://backend.composio.dev/api/v3/connected_accounts?user_ids=%s", entityID) - req, err := http.NewRequestWithContext(ctx, "GET", url, nil) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("x-api-key", apiKey) - - client := &http.Client{Timeout: 10 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to fetch connected accounts: %w", err) - } - defer resp.Body.Close() - - respBody, _ := io.ReadAll(resp.Body) - - if resp.StatusCode >= 400 { - return "", fmt.Errorf("Composio API error (status %d): %s", resp.StatusCode, string(respBody)) - } - - // Parse v3 response - var response struct { - Items []struct { - ID string `json:"id"` - Toolkit struct { - Slug string `json:"slug"` - } `json:"toolkit"` - } `json:"items"` - } - if err := json.Unmarshal(respBody, &response); err != nil { - return "", fmt.Errorf("failed to parse response: %w", err) - } - - // Find the connected account for this app - for _, account := range response.Items { - if account.Toolkit.Slug == appName { - return account.ID, nil - } - } - - return "", fmt.Errorf("no %s connection found for entity %s", appName, entityID) -} - -// EnsureIndexes creates the necessary indexes for the credentials collection -func (s *CredentialService) EnsureIndexes(ctx context.Context) error { - indexes := []mongo.IndexModel{ - // User ID for listing - { - Keys: bson.D{{Key: "userId", Value: 1}}, - }, - // User + integration type for filtering - { - Keys: bson.D{ - {Key: "userId", Value: 1}, - {Key: "integrationType", Value: 1}, - }, - }, - // User + name + type for uniqueness (optional, could enforce unique names per type) - { - Keys: bson.D{ - {Key: "userId", Value: 1}, - {Key: "integrationType", Value: 1}, - {Key: "name", Value: 1}, - }, - }, - } - - _, err := s.collection().Indexes().CreateMany(ctx, indexes) - if err != nil { - return fmt.Errorf("failed to create credential indexes: %w", err) - } - - log.Println("✅ [CREDENTIAL] Ensured indexes for credentials collection") - return nil -} - -// CreateCredentialResolver creates a credential resolver function that can be -// injected into tool args for runtime credential access. -// This function is here to avoid import cycles (tools cannot import services). -func (s *CredentialService) CreateCredentialResolver(userID string) func(credentialID string) (*models.DecryptedCredential, error) { - return func(credentialID string) (*models.DecryptedCredential, error) { - objID, err := primitive.ObjectIDFromHex(credentialID) - if err != nil { - return nil, fmt.Errorf("invalid credential ID: %w", err) - } - - cred, err := s.GetDecrypted(context.Background(), userID, objID) - if err != nil { - return nil, fmt.Errorf("failed to retrieve credential: %w", err) - } - - return cred, nil - } -} diff --git a/backend/internal/services/execution_service.go b/backend/internal/services/execution_service.go deleted file mode 100644 index 4d411d9b..00000000 --- a/backend/internal/services/execution_service.go +++ /dev/null @@ -1,861 +0,0 @@ -package services - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "context" - "encoding/json" - "fmt" - "log" - "regexp" - "time" - - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/bson/primitive" - "go.mongodb.org/mongo-driver/mongo" - "go.mongodb.org/mongo-driver/mongo/options" -) - -// ExecutionRecord is the MongoDB representation of an execution -type ExecutionRecord struct { - ID primitive.ObjectID `bson:"_id,omitempty" json:"id"` - AgentID string `bson:"agentId" json:"agentId"` - UserID string `bson:"userId" json:"userId"` - WorkflowVersion int `bson:"workflowVersion" json:"workflowVersion"` - - // Trigger info - TriggerType string `bson:"triggerType" json:"triggerType"` // manual, scheduled, webhook, api - ScheduleID primitive.ObjectID `bson:"scheduleId,omitempty" json:"scheduleId,omitempty"` - APIKeyID primitive.ObjectID `bson:"apiKeyId,omitempty" json:"apiKeyId,omitempty"` - - // Execution state - Status string `bson:"status" json:"status"` // pending, running, completed, failed, partial - Input map[string]interface{} `bson:"input,omitempty" json:"input,omitempty"` - Output map[string]interface{} `bson:"output,omitempty" json:"output,omitempty"` - BlockStates map[string]*models.BlockState `bson:"blockStates,omitempty" json:"blockStates,omitempty"` - Error string `bson:"error,omitempty" json:"error,omitempty"` - - // Standardized API response (clean, well-structured output) - Result string `bson:"result,omitempty" json:"result,omitempty"` // Primary text result - Artifacts []models.APIArtifact `bson:"artifacts,omitempty" json:"artifacts,omitempty"` // Generated charts/images - Files []models.APIFile `bson:"files,omitempty" json:"files,omitempty"` // Generated files - - // Timing - StartedAt time.Time `bson:"startedAt" json:"startedAt"` - CompletedAt *time.Time `bson:"completedAt,omitempty" json:"completedAt,omitempty"` - DurationMs int64 `bson:"durationMs,omitempty" json:"durationMs,omitempty"` - - // TTL (tier-based retention) - ExpiresAt time.Time `bson:"expiresAt" json:"expiresAt"` - - CreatedAt time.Time `bson:"createdAt" json:"createdAt"` -} - -// ExecutionService manages execution history in MongoDB -type ExecutionService struct { - mongoDB *database.MongoDB - tierService *TierService -} - -// NewExecutionService creates a new execution service -func NewExecutionService(mongoDB *database.MongoDB, tierService *TierService) *ExecutionService { - return &ExecutionService{ - mongoDB: mongoDB, - tierService: tierService, - } -} - -// collection returns the executions collection -func (s *ExecutionService) collection() *mongo.Collection { - return s.mongoDB.Database().Collection("executions") -} - -// Create creates a new execution record -func (s *ExecutionService) Create(ctx context.Context, req *CreateExecutionRequest) (*ExecutionRecord, error) { - // Calculate retention based on user tier - retentionDays := 30 // default free tier - if s.tierService != nil { - retentionDays = s.tierService.GetExecutionRetentionDays(ctx, req.UserID) - } - - now := time.Now() - record := &ExecutionRecord{ - AgentID: req.AgentID, - UserID: req.UserID, - WorkflowVersion: req.WorkflowVersion, - TriggerType: req.TriggerType, - ScheduleID: req.ScheduleID, - APIKeyID: req.APIKeyID, - Status: "pending", - Input: req.Input, - StartedAt: now, - ExpiresAt: now.Add(time.Duration(retentionDays) * 24 * time.Hour), - CreatedAt: now, - } - - result, err := s.collection().InsertOne(ctx, record) - if err != nil { - return nil, fmt.Errorf("failed to create execution: %w", err) - } - - record.ID = result.InsertedID.(primitive.ObjectID) - log.Printf("📝 [EXECUTION] Created execution %s for agent %s (trigger: %s)", - record.ID.Hex(), req.AgentID, req.TriggerType) - - return record, nil -} - -// CreateExecutionRequest contains the data needed to create an execution -type CreateExecutionRequest struct { - AgentID string - UserID string - WorkflowVersion int - TriggerType string // manual, scheduled, webhook, api - ScheduleID primitive.ObjectID - APIKeyID primitive.ObjectID - Input map[string]interface{} -} - -// UpdateStatus updates the execution status -func (s *ExecutionService) UpdateStatus(ctx context.Context, executionID primitive.ObjectID, status string) error { - update := bson.M{ - "$set": bson.M{ - "status": status, - }, - } - - _, err := s.collection().UpdateByID(ctx, executionID, update) - if err != nil { - return fmt.Errorf("failed to update execution status: %w", err) - } - - log.Printf("📊 [EXECUTION] Updated %s status to %s", executionID.Hex(), status) - return nil -} - -// Complete marks an execution as complete with output -func (s *ExecutionService) Complete(ctx context.Context, executionID primitive.ObjectID, result *ExecutionCompleteRequest) error { - now := time.Now() - - // Get the execution to calculate duration - exec, err := s.GetByID(ctx, executionID) - if err != nil { - return err - } - - durationMs := now.Sub(exec.StartedAt).Milliseconds() - - // Sanitize output and blockStates to remove large base64 data - // This prevents MongoDB document size limit (16MB) issues - sanitizedOutput := sanitizeOutputForStorage(result.Output) - sanitizedBlockStates := sanitizeBlockStatesForStorageV2(result.BlockStates) - - // Log sanitization to help debug - log.Printf("🧹 [EXECUTION] Sanitizing execution %s for storage", executionID.Hex()) - - update := bson.M{ - "$set": bson.M{ - "status": result.Status, - "output": sanitizedOutput, - "blockStates": sanitizedBlockStates, - "error": result.Error, - "completedAt": now, - "durationMs": durationMs, - // Store clean API response fields - "result": result.Result, - "artifacts": result.Artifacts, - "files": result.Files, - }, - } - - _, err = s.collection().UpdateByID(ctx, executionID, update) - if err != nil { - return fmt.Errorf("failed to complete execution: %w", err) - } - - log.Printf("✅ [EXECUTION] Completed %s with status %s (duration: %dms)", - executionID.Hex(), result.Status, durationMs) - - return nil -} - -// ExecutionCompleteRequest contains the completion data -type ExecutionCompleteRequest struct { - Status string - Output map[string]interface{} - BlockStates map[string]*models.BlockState - Error string - - // Clean API response fields - Result string // Primary text result - Artifacts []models.APIArtifact // Generated charts/images - Files []models.APIFile // Generated files -} - -// GetByID retrieves an execution by ID -func (s *ExecutionService) GetByID(ctx context.Context, executionID primitive.ObjectID) (*ExecutionRecord, error) { - var record ExecutionRecord - err := s.collection().FindOne(ctx, bson.M{"_id": executionID}).Decode(&record) - if err != nil { - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("execution not found") - } - return nil, fmt.Errorf("failed to get execution: %w", err) - } - return &record, nil -} - -// GetByIDAndUser retrieves an execution by ID ensuring user ownership -func (s *ExecutionService) GetByIDAndUser(ctx context.Context, executionID primitive.ObjectID, userID string) (*ExecutionRecord, error) { - var record ExecutionRecord - err := s.collection().FindOne(ctx, bson.M{ - "_id": executionID, - "userId": userID, - }).Decode(&record) - if err != nil { - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("execution not found") - } - return nil, fmt.Errorf("failed to get execution: %w", err) - } - return &record, nil -} - -// ListByAgent returns paginated executions for an agent -func (s *ExecutionService) ListByAgent(ctx context.Context, agentID, userID string, opts *ListExecutionsOptions) (*PaginatedExecutions, error) { - filter := bson.M{ - "agentId": agentID, - "userId": userID, - } - - if opts != nil && opts.Status != "" { - filter["status"] = opts.Status - } - - if opts != nil && opts.TriggerType != "" { - filter["triggerType"] = opts.TriggerType - } - - return s.listWithFilter(ctx, filter, opts) -} - -// ListByUser returns paginated executions for a user -func (s *ExecutionService) ListByUser(ctx context.Context, userID string, opts *ListExecutionsOptions) (*PaginatedExecutions, error) { - filter := bson.M{ - "userId": userID, - } - - if opts != nil && opts.Status != "" { - filter["status"] = opts.Status - } - - if opts != nil && opts.TriggerType != "" { - filter["triggerType"] = opts.TriggerType - } - - if opts != nil && opts.AgentID != "" { - filter["agentId"] = opts.AgentID - } - - return s.listWithFilter(ctx, filter, opts) -} - -// listWithFilter performs the actual paginated list query -func (s *ExecutionService) listWithFilter(ctx context.Context, filter bson.M, opts *ListExecutionsOptions) (*PaginatedExecutions, error) { - // Default pagination - limit := int64(20) - page := int64(1) - - if opts != nil { - if opts.Limit > 0 && opts.Limit <= 100 { - limit = int64(opts.Limit) - } - if opts.Page > 0 { - page = int64(opts.Page) - } - } - - skip := (page - 1) * limit - - // Count total - total, err := s.collection().CountDocuments(ctx, filter) - if err != nil { - return nil, fmt.Errorf("failed to count executions: %w", err) - } - - // Find with pagination (newest first) - findOpts := options.Find(). - SetSort(bson.D{{Key: "startedAt", Value: -1}}). - SetSkip(skip). - SetLimit(limit) - - cursor, err := s.collection().Find(ctx, filter, findOpts) - if err != nil { - return nil, fmt.Errorf("failed to find executions: %w", err) - } - defer cursor.Close(ctx) - - var executions []ExecutionRecord - if err := cursor.All(ctx, &executions); err != nil { - return nil, fmt.Errorf("failed to decode executions: %w", err) - } - - return &PaginatedExecutions{ - Executions: executions, - Total: total, - Page: page, - Limit: limit, - HasMore: skip+int64(len(executions)) < total, - }, nil -} - -// ListExecutionsOptions contains query options for listing executions -type ListExecutionsOptions struct { - Page int - Limit int - Status string // filter by status - TriggerType string // filter by trigger type - AgentID string // filter by agent (for user-wide queries) -} - -// PaginatedExecutions is the response for paginated execution lists -type PaginatedExecutions struct { - Executions []ExecutionRecord `json:"executions"` - Total int64 `json:"total"` - Page int64 `json:"page"` - Limit int64 `json:"limit"` - HasMore bool `json:"hasMore"` -} - -// GetStats returns execution statistics for an agent -func (s *ExecutionService) GetStats(ctx context.Context, agentID, userID string) (*ExecutionStats, error) { - filter := bson.M{ - "agentId": agentID, - "userId": userID, - } - - // Get counts by status - pipeline := mongo.Pipeline{ - {{Key: "$match", Value: filter}}, - {{Key: "$group", Value: bson.M{ - "_id": "$status", - "count": bson.M{"$sum": 1}, - "avgDuration": bson.M{"$avg": "$durationMs"}, - }}}, - } - - cursor, err := s.collection().Aggregate(ctx, pipeline) - if err != nil { - return nil, fmt.Errorf("failed to aggregate stats: %w", err) - } - defer cursor.Close(ctx) - - stats := &ExecutionStats{ - ByStatus: make(map[string]StatusStats), - } - - var results []struct { - ID string `bson:"_id"` - Count int64 `bson:"count"` - AvgDuration float64 `bson:"avgDuration"` - } - - if err := cursor.All(ctx, &results); err != nil { - return nil, fmt.Errorf("failed to decode stats: %w", err) - } - - for _, r := range results { - stats.Total += r.Count - stats.ByStatus[r.ID] = StatusStats{ - Count: r.Count, - AvgDuration: int64(r.AvgDuration), - } - if r.ID == "completed" { - stats.SuccessCount = r.Count - } else if r.ID == "failed" { - stats.FailedCount = r.Count - } - } - - if stats.Total > 0 { - stats.SuccessRate = float64(stats.SuccessCount) / float64(stats.Total) * 100 - } - - return stats, nil -} - -// ExecutionStats contains aggregated execution statistics -type ExecutionStats struct { - Total int64 `json:"total"` - SuccessCount int64 `json:"successCount"` - FailedCount int64 `json:"failedCount"` - SuccessRate float64 `json:"successRate"` - ByStatus map[string]StatusStats `json:"byStatus"` -} - -// StatusStats contains stats for a single status -type StatusStats struct { - Count int64 `json:"count"` - AvgDuration int64 `json:"avgDurationMs"` -} - -// DeleteExpired removes executions past their TTL (called by cleanup job) -func (s *ExecutionService) DeleteExpired(ctx context.Context) (int64, error) { - result, err := s.collection().DeleteMany(ctx, bson.M{ - "expiresAt": bson.M{"$lt": time.Now()}, - }) - if err != nil { - return 0, fmt.Errorf("failed to delete expired executions: %w", err) - } - - if result.DeletedCount > 0 { - log.Printf("🗑️ [EXECUTION] Deleted %d expired executions", result.DeletedCount) - } - - return result.DeletedCount, nil -} - -// DeleteAllByUser deletes all executions for a user (GDPR compliance) -func (s *ExecutionService) DeleteAllByUser(ctx context.Context, userID string) (int64, error) { - if userID == "" { - return 0, fmt.Errorf("user ID is required") - } - - result, err := s.collection().DeleteMany(ctx, bson.M{"userId": userID}) - if err != nil { - return 0, fmt.Errorf("failed to delete user executions: %w", err) - } - - log.Printf("🗑️ [GDPR] Deleted %d executions for user %s", result.DeletedCount, userID) - return result.DeletedCount, nil -} - -// EnsureIndexes creates the necessary indexes for the executions collection -func (s *ExecutionService) EnsureIndexes(ctx context.Context) error { - indexes := []mongo.IndexModel{ - // User + startedAt for listing user's executions - { - Keys: bson.D{ - {Key: "userId", Value: 1}, - {Key: "startedAt", Value: -1}, - }, - }, - // Agent + startedAt for listing agent's executions - { - Keys: bson.D{ - {Key: "agentId", Value: 1}, - {Key: "startedAt", Value: -1}, - }, - }, - // TTL index for automatic deletion - { - Keys: bson.D{{Key: "expiresAt", Value: 1}}, - Options: options.Index().SetExpireAfterSeconds(0), - }, - // Status index for filtering - { - Keys: bson.D{{Key: "status", Value: 1}}, - }, - // Schedule ID for scheduled execution lookups - { - Keys: bson.D{{Key: "scheduleId", Value: 1}}, - Options: options.Index().SetSparse(true), - }, - } - - _, err := s.collection().Indexes().CreateMany(ctx, indexes) - if err != nil { - return fmt.Errorf("failed to create execution indexes: %w", err) - } - - log.Println("✅ [EXECUTION] Ensured indexes for executions collection") - return nil -} - -// sanitizeOutputForStorage sanitizes execution output by converting to JSON, stripping base64 data, and converting back -// This approach handles any nested structure including typed structs -func sanitizeOutputForStorage(data map[string]interface{}) map[string]interface{} { - if data == nil { - return nil - } - - // Marshal to JSON - jsonBytes, err := json.Marshal(data) - if err != nil { - log.Printf("⚠️ [EXECUTION] Failed to marshal output for sanitization: %v", err) - return data - } - - originalSize := len(jsonBytes) - - // Apply regex patterns to strip base64 data - sanitized := stripBase64FromJSON(string(jsonBytes)) - - // Unmarshal back - var result map[string]interface{} - if err := json.Unmarshal([]byte(sanitized), &result); err != nil { - log.Printf("⚠️ [EXECUTION] Failed to unmarshal sanitized output: %v", err) - return data - } - - // Apply internal field filtering to remove noise fields (model, tokens, etc.) - result = filterInternalFields(result) - - newSize := len(sanitized) - if originalSize != newSize { - log.Printf("🧹 [EXECUTION] Sanitized output: %d -> %d bytes (%.1f%% reduction)", - originalSize, newSize, float64(originalSize-newSize)/float64(originalSize)*100) - } - - return result -} - -// filterInternalFields recursively removes internal fields from the output map -func filterInternalFields(data map[string]interface{}) map[string]interface{} { - if data == nil { - return nil - } - - result := make(map[string]interface{}) - for key, value := range data { - // Skip internal fields - if internalFieldsToFilter[key] { - continue - } - - // Recursively filter nested maps - if nested, ok := value.(map[string]interface{}); ok { - result[key] = filterInternalFields(nested) - } else if slice, ok := value.([]interface{}); ok { - // Handle arrays - filteredSlice := make([]interface{}, len(slice)) - for i, item := range slice { - if itemMap, ok := item.(map[string]interface{}); ok { - filteredSlice[i] = filterInternalFields(itemMap) - } else { - filteredSlice[i] = item - } - } - result[key] = filteredSlice - } else { - result[key] = value - } - } - - return result -} - -// sanitizeBlockStatesForStorageV2 sanitizes block states using JSON approach -func sanitizeBlockStatesForStorageV2(states map[string]*models.BlockState) map[string]*models.BlockState { - if states == nil { - return nil - } - - // Marshal to JSON - jsonBytes, err := json.Marshal(states) - if err != nil { - log.Printf("⚠️ [EXECUTION] Failed to marshal block states for sanitization: %v", err) - return states - } - - originalSize := len(jsonBytes) - - // Apply regex patterns to strip base64 data - sanitized := stripBase64FromJSON(string(jsonBytes)) - - // Unmarshal back - var result map[string]*models.BlockState - if err := json.Unmarshal([]byte(sanitized), &result); err != nil { - log.Printf("⚠️ [EXECUTION] Failed to unmarshal sanitized block states: %v", err) - return states - } - - // Apply internal field filtering to block state outputs - for blockID, state := range result { - if state != nil && state.Outputs != nil { - result[blockID].Outputs = filterInternalFields(state.Outputs) - } - } - - newSize := len(sanitized) - if originalSize != newSize { - log.Printf("🧹 [EXECUTION] Sanitized block states: %d -> %d bytes (%.1f%% reduction)", - originalSize, newSize, float64(originalSize-newSize)/float64(originalSize)*100) - } - - return result -} - -// stripBase64FromJSON removes base64 image data from JSON string using regex -func stripBase64FromJSON(jsonStr string) string { - // Pattern 1: data URI images (data:image/xxx;base64,...) - dataURIPattern := regexp.MustCompile(`"data:image/[^;]+;base64,[A-Za-z0-9+/=]+"`) - jsonStr = dataURIPattern.ReplaceAllString(jsonStr, `"[BASE64_IMAGE_STRIPPED]"`) - - // Pattern 2: Long base64-like strings in "data" fields - dataFieldPattern := regexp.MustCompile(`"data"\s*:\s*"[A-Za-z0-9+/=]{500,}"`) - jsonStr = dataFieldPattern.ReplaceAllString(jsonStr, `"data":"[BASE64_DATA_STRIPPED]"`) - - // Pattern 3: Long base64-like strings in "image", "plot", "chart" fields - imageFieldPattern := regexp.MustCompile(`"(image|plot|chart|figure|png|jpeg|base64)"\s*:\s*"[A-Za-z0-9+/=]{500,}"`) - jsonStr = imageFieldPattern.ReplaceAllString(jsonStr, `"$1":"[BASE64_IMAGE_STRIPPED]"`) - - // Pattern 4: Any remaining very long strings that look like base64 - // Note: Go RE2 has max repeat count of 1000, so we use {1000,} to catch long strings - longStringPattern := regexp.MustCompile(`"[A-Za-z0-9+/=]{1000,}"`) - jsonStr = longStringPattern.ReplaceAllString(jsonStr, `"[LARGE_DATA_STRIPPED]"`) - - // Pattern 5: Handle nested JSON strings containing base64 (e.g., in "Result" field of tool calls) - // This handles cases where the Result is a JSON string that contains base64 - resultFieldPattern := regexp.MustCompile(`"Result"\s*:\s*"\{[^"]*"data"\s*:\s*\\"[A-Za-z0-9+/=]{100,}\\"[^"]*\}"`) - if resultFieldPattern.MatchString(jsonStr) { - // For Result fields containing JSON with base64, we need to escape the replacement - jsonStr = resultFieldPattern.ReplaceAllStringFunc(jsonStr, func(match string) string { - // Strip the base64 within the nested JSON - innerPattern := regexp.MustCompile(`\\"data\\"\s*:\s*\\"[A-Za-z0-9+/=]+\\"`) - return innerPattern.ReplaceAllString(match, `\"data\":\"[BASE64_STRIPPED]\"`) - }) - } - - return jsonStr -} - -// sanitizeForStorage removes large base64 data from maps to prevent MongoDB document size limit issues -// It replaces base64 image data with a placeholder while preserving metadata -func sanitizeForStorage(data map[string]interface{}) map[string]interface{} { - if data == nil { - return nil - } - - result := make(map[string]interface{}) - for key, value := range data { - result[key] = sanitizeValue(value) - } - return result -} - -// sanitizeBlockStatesForStorage sanitizes all block states -func sanitizeBlockStatesForStorage(states map[string]*models.BlockState) map[string]*models.BlockState { - if states == nil { - return nil - } - - result := make(map[string]*models.BlockState) - for blockID, state := range states { - if state == nil { - continue - } - sanitizedState := &models.BlockState{ - Status: state.Status, - Inputs: sanitizeForStorage(state.Inputs), - Outputs: sanitizeForStorage(state.Outputs), - Error: state.Error, - StartedAt: state.StartedAt, - CompletedAt: state.CompletedAt, - } - result[blockID] = sanitizedState - } - return result -} - -// sanitizeValue recursively sanitizes a value, replacing large base64 strings -func sanitizeValue(value interface{}) interface{} { - if value == nil { - return nil - } - - switch v := value.(type) { - case string: - return sanitizeString(v) - case map[string]interface{}: - return sanitizeMap(v) - case []interface{}: - return sanitizeSlice(v) - default: - return value - } -} - -// sanitizeString checks if a string is base64 image data and replaces it -func sanitizeString(s string) string { - // If string is too short, keep it - if len(s) < 500 { - return s - } - - // Check for data URI prefix (base64 image) - if regexp.MustCompile(`^data:image/[^;]+;base64,`).MatchString(s) { - return "[BASE64_IMAGE_STRIPPED_FOR_STORAGE]" - } - - // Check for long base64-like strings (no spaces, mostly alphanumeric + /+=) - if regexp.MustCompile(`^[A-Za-z0-9+/=]{500,}$`).MatchString(s) { - return "[BASE64_DATA_STRIPPED_FOR_STORAGE]" - } - - // If string is extremely long (>100KB), truncate it - if len(s) > 100000 { - return s[:1000] + "... [TRUNCATED_FOR_STORAGE]" - } - - return s -} - -// internalFieldsToFilter contains fields that should not be exposed to API consumers -// These are internal execution details that add noise to the output -var internalFieldsToFilter = map[string]bool{ - "model": true, // Internal model ID (use _workflowModelId instead) - "__user_id__": true, // Internal user context - "_workflowModelId": true, // Internal workflow model reference - "tokens": true, // Token usage (available in metadata) - "iterations": true, // Internal execution iterations - "start": true, // Internal start state - "value": true, // Redundant with response - "input": true, // Already stored separately - "rawResponse": true, // Huge and redundant -} - -// sanitizeMap recursively sanitizes a map -func sanitizeMap(m map[string]interface{}) map[string]interface{} { - result := make(map[string]interface{}) - - for key, value := range m { - // Skip internal fields that shouldn't be exposed to API consumers - if internalFieldsToFilter[key] { - continue - } - - // Special handling for known artifact/image fields - if key == "artifacts" { - result[key] = sanitizeArtifacts(value) - continue - } - if key == "plots" || key == "images" || key == "base64_images" { - result[key] = sanitizePlots(value) - continue - } - // Handle toolCalls array which contains Result fields with JSON+base64 - if key == "toolCalls" { - result[key] = sanitizeToolCallsForStorage(value) - continue - } - result[key] = sanitizeValue(value) - } - - return result -} - -// sanitizeSlice recursively sanitizes a slice -func sanitizeSlice(s []interface{}) []interface{} { - result := make([]interface{}, len(s)) - for i, v := range s { - result[i] = sanitizeValue(v) - } - return result -} - -// sanitizeArtifacts handles the artifacts array, keeping metadata but removing data -func sanitizeArtifacts(value interface{}) interface{} { - artifacts, ok := value.([]interface{}) - if !ok { - return value - } - - result := make([]interface{}, 0, len(artifacts)) - for _, a := range artifacts { - artifact, ok := a.(map[string]interface{}) - if !ok { - continue - } - - // Keep metadata, remove actual data - sanitized := map[string]interface{}{ - "type": artifact["type"], - "format": artifact["format"], - "title": artifact["title"], - "data": "[BASE64_IMAGE_STRIPPED_FOR_STORAGE]", - } - result = append(result, sanitized) - } - - log.Printf("🧹 [EXECUTION] Sanitized %d artifacts for storage", len(result)) - return result -} - -// sanitizePlots handles plots/images arrays from E2B responses -func sanitizePlots(value interface{}) interface{} { - plots, ok := value.([]interface{}) - if !ok { - // Could be a single plot as map - if plotMap, ok := value.(map[string]interface{}); ok { - return sanitizeSinglePlot(plotMap) - } - return value - } - - result := make([]interface{}, 0, len(plots)) - for _, p := range plots { - if plot, ok := p.(map[string]interface{}); ok { - result = append(result, sanitizeSinglePlot(plot)) - } - } - - log.Printf("🧹 [EXECUTION] Sanitized %d plots for storage", len(result)) - return result -} - -// sanitizeSinglePlot removes base64 data from a single plot while keeping metadata -func sanitizeSinglePlot(plot map[string]interface{}) map[string]interface{} { - result := make(map[string]interface{}) - for k, v := range plot { - if k == "data" || k == "image" || k == "base64" { - result[k] = "[BASE64_IMAGE_STRIPPED_FOR_STORAGE]" - } else { - result[k] = v - } - } - return result -} - -// sanitizeToolCallsForStorage sanitizes tool call results (JSON strings with base64) -func sanitizeToolCallsForStorage(toolCalls interface{}) interface{} { - calls, ok := toolCalls.([]interface{}) - if !ok { - return toolCalls - } - - result := make([]interface{}, 0, len(calls)) - for _, tc := range calls { - call, ok := tc.(map[string]interface{}) - if !ok { - result = append(result, tc) - continue - } - - sanitized := make(map[string]interface{}) - for k, v := range call { - if k == "result" { - // Result is a JSON string - parse, sanitize, re-stringify - if resultStr, ok := v.(string); ok && len(resultStr) > 1000 { - var resultData map[string]interface{} - if err := json.Unmarshal([]byte(resultStr), &resultData); err == nil { - sanitizedResult := sanitizeMap(resultData) - if sanitizedJSON, err := json.Marshal(sanitizedResult); err == nil { - sanitized[k] = string(sanitizedJSON) - continue - } - } - // If parsing fails, just truncate - if len(resultStr) > 5000 { - sanitized[k] = resultStr[:5000] + "... [TRUNCATED]" - continue - } - } - } - sanitized[k] = v - } - result = append(result, sanitized) - } - - return result -} diff --git a/backend/internal/services/execution_service_test.go b/backend/internal/services/execution_service_test.go deleted file mode 100644 index 72704a08..00000000 --- a/backend/internal/services/execution_service_test.go +++ /dev/null @@ -1,241 +0,0 @@ -package services - -import ( - "context" - "testing" - - "go.mongodb.org/mongo-driver/bson/primitive" -) - -func TestNewExecutionService(t *testing.T) { - // Test creation without dependencies (both nil) - service := NewExecutionService(nil, nil) - if service == nil { - t.Fatal("Expected non-nil execution service") - } -} - -func TestExecutionRecord_Structure(t *testing.T) { - // Test that ExecutionRecord can be created with all fields - record := &ExecutionRecord{ - ID: primitive.NewObjectID(), - AgentID: "agent-123", - UserID: "user-456", - WorkflowVersion: 1, - TriggerType: "manual", - Status: "pending", - Input: map[string]interface{}{"topic": "test"}, - } - - if record.AgentID != "agent-123" { - t.Errorf("Expected AgentID 'agent-123', got '%s'", record.AgentID) - } - - if record.TriggerType != "manual" { - t.Errorf("Expected TriggerType 'manual', got '%s'", record.TriggerType) - } -} - -func TestCreateExecutionRequest_Validation(t *testing.T) { - tests := []struct { - name string - req CreateExecutionRequest - wantAgentID string - wantTrigger string - }{ - { - name: "manual trigger", - req: CreateExecutionRequest{ - AgentID: "agent-1", - UserID: "user-1", - WorkflowVersion: 1, - TriggerType: "manual", - }, - wantAgentID: "agent-1", - wantTrigger: "manual", - }, - { - name: "scheduled trigger", - req: CreateExecutionRequest{ - AgentID: "agent-2", - UserID: "user-2", - WorkflowVersion: 2, - TriggerType: "scheduled", - ScheduleID: primitive.NewObjectID(), - }, - wantAgentID: "agent-2", - wantTrigger: "scheduled", - }, - { - name: "api trigger", - req: CreateExecutionRequest{ - AgentID: "agent-3", - UserID: "user-3", - WorkflowVersion: 1, - TriggerType: "api", - APIKeyID: primitive.NewObjectID(), - }, - wantAgentID: "agent-3", - wantTrigger: "api", - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if tt.req.AgentID != tt.wantAgentID { - t.Errorf("Expected AgentID '%s', got '%s'", tt.wantAgentID, tt.req.AgentID) - } - if tt.req.TriggerType != tt.wantTrigger { - t.Errorf("Expected TriggerType '%s', got '%s'", tt.wantTrigger, tt.req.TriggerType) - } - }) - } -} - -func TestListExecutionsOptions_Defaults(t *testing.T) { - opts := &ListExecutionsOptions{} - - // Default values should be zero/empty - if opts.Page != 0 { - t.Errorf("Expected default Page 0, got %d", opts.Page) - } - if opts.Limit != 0 { - t.Errorf("Expected default Limit 0, got %d", opts.Limit) - } - if opts.Status != "" { - t.Errorf("Expected empty default Status, got '%s'", opts.Status) - } -} - -func TestListExecutionsOptions_WithFilters(t *testing.T) { - opts := &ListExecutionsOptions{ - Page: 2, - Limit: 50, - Status: "completed", - TriggerType: "scheduled", - AgentID: "agent-123", - } - - if opts.Page != 2 { - t.Errorf("Expected Page 2, got %d", opts.Page) - } - if opts.Limit != 50 { - t.Errorf("Expected Limit 50, got %d", opts.Limit) - } - if opts.Status != "completed" { - t.Errorf("Expected Status 'completed', got '%s'", opts.Status) - } - if opts.TriggerType != "scheduled" { - t.Errorf("Expected TriggerType 'scheduled', got '%s'", opts.TriggerType) - } - if opts.AgentID != "agent-123" { - t.Errorf("Expected AgentID 'agent-123', got '%s'", opts.AgentID) - } -} - -func TestPaginatedExecutions_Empty(t *testing.T) { - result := &PaginatedExecutions{ - Executions: []ExecutionRecord{}, - Total: 0, - Page: 1, - Limit: 20, - HasMore: false, - } - - if len(result.Executions) != 0 { - t.Errorf("Expected 0 executions, got %d", len(result.Executions)) - } - if result.HasMore { - t.Error("Expected HasMore to be false") - } -} - -func TestExecutionStats_Empty(t *testing.T) { - stats := &ExecutionStats{ - Total: 0, - SuccessCount: 0, - FailedCount: 0, - SuccessRate: 0, - ByStatus: make(map[string]StatusStats), - } - - if stats.Total != 0 { - t.Errorf("Expected Total 0, got %d", stats.Total) - } - if stats.SuccessRate != 0 { - t.Errorf("Expected SuccessRate 0, got %f", stats.SuccessRate) - } -} - -func TestExecutionStats_Calculations(t *testing.T) { - // Simulate stats calculation - stats := &ExecutionStats{ - Total: 100, - SuccessCount: 85, - FailedCount: 15, - SuccessRate: 85.0, - ByStatus: map[string]StatusStats{ - "completed": {Count: 85, AvgDuration: 1500}, - "failed": {Count: 15, AvgDuration: 2000}, - }, - } - - if stats.SuccessRate != 85.0 { - t.Errorf("Expected SuccessRate 85.0, got %f", stats.SuccessRate) - } - - completedStats, ok := stats.ByStatus["completed"] - if !ok { - t.Fatal("Expected 'completed' status in ByStatus") - } - if completedStats.Count != 85 { - t.Errorf("Expected completed count 85, got %d", completedStats.Count) - } -} - -func TestExecutionCompleteRequest_Fields(t *testing.T) { - req := &ExecutionCompleteRequest{ - Status: "completed", - Output: map[string]interface{}{ - "result": "success", - }, - Error: "", - } - - if req.Status != "completed" { - t.Errorf("Expected Status 'completed', got '%s'", req.Status) - } - if req.Error != "" { - t.Errorf("Expected empty Error, got '%s'", req.Error) - } -} - -func TestExecutionCompleteRequest_WithError(t *testing.T) { - req := &ExecutionCompleteRequest{ - Status: "failed", - Output: nil, - Error: "execution timeout", - } - - if req.Status != "failed" { - t.Errorf("Expected Status 'failed', got '%s'", req.Status) - } - if req.Error != "execution timeout" { - t.Errorf("Expected Error 'execution timeout', got '%s'", req.Error) - } -} - -// Integration test helper - would need MongoDB to run actual tests -func TestExecutionService_Integration(t *testing.T) { - if testing.Short() { - t.Skip("Skipping integration test in short mode") - } - - // This test would require a running MongoDB instance - // For now, just verify the service can be created - _ = context.Background() - service := NewExecutionService(nil, nil) - if service == nil { - t.Fatal("Expected non-nil service") - } -} diff --git a/backend/internal/services/file_cache.go b/backend/internal/services/file_cache.go deleted file mode 100644 index 575836e7..00000000 --- a/backend/internal/services/file_cache.go +++ /dev/null @@ -1,437 +0,0 @@ -package services - -import ( - "claraverse/internal/security" - "fmt" - "log" - "os" - "sync" - "time" - - "github.com/patrickmn/go-cache" -) - -// CachedFile represents a file stored in memory cache -type CachedFile struct { - FileID string - UserID string - ConversationID string - ExtractedText *security.SecureString // For PDFs - FileHash security.Hash - Filename string - MimeType string - Size int64 - PageCount int // For PDFs - WordCount int // For PDFs - FilePath string // For images (disk location) - UploadedAt time.Time -} - -// FileCacheService manages uploaded files in memory -type FileCacheService struct { - cache *cache.Cache - mu sync.RWMutex -} - -var ( - fileCacheInstance *FileCacheService - fileCacheOnce sync.Once -) - -// GetFileCacheService returns the singleton file cache service -func GetFileCacheService() *FileCacheService { - fileCacheOnce.Do(func() { - fileCacheInstance = NewFileCacheService() - }) - return fileCacheInstance -} - -// NewFileCacheService creates a new file cache service -func NewFileCacheService() *FileCacheService { - c := cache.New(30*time.Minute, 10*time.Minute) - - // Set eviction handler for secure wiping - c.OnEvicted(func(key string, value interface{}) { - if file, ok := value.(*CachedFile); ok { - log.Printf("🗑️ [FILE-CACHE] Evicting file %s (%s) - secure wiping memory", file.FileID, file.Filename) - file.SecureWipe() - } - }) - - return &FileCacheService{ - cache: c, - } -} - -// Store stores a file in the cache -func (s *FileCacheService) Store(file *CachedFile) { - s.mu.Lock() - defer s.mu.Unlock() - s.cache.Set(file.FileID, file, cache.DefaultExpiration) - log.Printf("📦 [FILE-CACHE] Stored file %s (%s) - %d bytes, %d words", - file.FileID, file.Filename, file.Size, file.WordCount) -} - -// Get retrieves a file from the cache -func (s *FileCacheService) Get(fileID string) (*CachedFile, bool) { - s.mu.RLock() - defer s.mu.RUnlock() - - value, found := s.cache.Get(fileID) - if !found { - return nil, false - } - - file, ok := value.(*CachedFile) - if !ok { - return nil, false - } - - return file, true -} - -// GetByUserAndConversation retrieves a file if it belongs to the user and conversation -func (s *FileCacheService) GetByUserAndConversation(fileID, userID, conversationID string) (*CachedFile, error) { - file, found := s.Get(fileID) - if !found { - return nil, fmt.Errorf("file not found or expired") - } - - // Verify ownership - if file.UserID != userID { - return nil, fmt.Errorf("access denied: file belongs to different user") - } - - // Verify conversation - if file.ConversationID != conversationID { - return nil, fmt.Errorf("file belongs to different conversation") - } - - return file, nil -} - -// GetFilesForConversation returns all files for a conversation -func (s *FileCacheService) GetFilesForConversation(conversationID string) []*CachedFile { - s.mu.RLock() - defer s.mu.RUnlock() - - var files []*CachedFile - for _, item := range s.cache.Items() { - if file, ok := item.Object.(*CachedFile); ok { - if file.ConversationID == conversationID { - files = append(files, file) - } - } - } - - return files -} - -// GetConversationFiles returns all file IDs for a conversation -func (s *FileCacheService) GetConversationFiles(conversationID string) []string { - files := s.GetFilesForConversation(conversationID) - fileIDs := make([]string, 0, len(files)) - for _, file := range files { - fileIDs = append(fileIDs, file.FileID) - } - return fileIDs -} - -// Delete removes a file from the cache and securely wipes it -func (s *FileCacheService) Delete(fileID string) { - s.mu.Lock() - defer s.mu.Unlock() - - // Get the file first to wipe it - if value, found := s.cache.Get(fileID); found { - if file, ok := value.(*CachedFile); ok { - log.Printf("🗑️ [FILE-CACHE] Deleting file %s (%s)", file.FileID, file.Filename) - file.SecureWipe() - } - } - - s.cache.Delete(fileID) -} - -// DeleteConversationFiles deletes all files for a conversation -func (s *FileCacheService) DeleteConversationFiles(conversationID string) { - s.mu.Lock() - defer s.mu.Unlock() - - log.Printf("🗑️ [FILE-CACHE] Deleting all files for conversation %s", conversationID) - - for key, item := range s.cache.Items() { - if file, ok := item.Object.(*CachedFile); ok { - if file.ConversationID == conversationID { - file.SecureWipe() - s.cache.Delete(key) - } - } - } -} - -// ExtendTTL extends the TTL of a file to match conversation lifetime -func (s *FileCacheService) ExtendTTL(fileID string, duration time.Duration) { - s.mu.Lock() - defer s.mu.Unlock() - - if value, found := s.cache.Get(fileID); found { - s.cache.Set(fileID, value, duration) - log.Printf("⏰ [FILE-CACHE] Extended TTL for file %s to %v", fileID, duration) - } -} - -// SecureWipe securely wipes the file's sensitive data -func (f *CachedFile) SecureWipe() { - if f.ExtractedText != nil { - f.ExtractedText.Wipe() - f.ExtractedText = nil - } - - // Delete physical file if it exists (for images) - if f.FilePath != "" { - if err := os.Remove(f.FilePath); err != nil && !os.IsNotExist(err) { - log.Printf("⚠️ Failed to delete file %s: %v", f.FilePath, err) - } else { - log.Printf("🗑️ Deleted file from disk: %s", f.FilePath) - } - } - - // Wipe hash - for i := range f.FileHash { - f.FileHash[i] = 0 - } - - // Clear other fields - f.FileID = "" - f.UserID = "" - f.ConversationID = "" - f.Filename = "" - f.FilePath = "" -} - -// CleanupExpiredFiles deletes files (images, CSV, Excel, JSON, etc.) older than 1 hour -// This handles all file types stored on disk, not just images -func (s *FileCacheService) CleanupExpiredFiles() { - s.mu.Lock() - defer s.mu.Unlock() - - now := time.Now() - expiredCount := 0 - - for key, item := range s.cache.Items() { - if file, ok := item.Object.(*CachedFile); ok { - // Cleanup all files with disk storage (images, CSV, Excel, JSON, etc.) - if file.FilePath != "" { - // Delete if older than 1 hour - if now.Sub(file.UploadedAt) > 1*time.Hour { - log.Printf("🗑️ [FILE-CACHE] Deleting expired file: %s (uploaded %v ago)", - file.Filename, now.Sub(file.UploadedAt)) - - // Delete from disk - if err := os.Remove(file.FilePath); err != nil && !os.IsNotExist(err) { - log.Printf("⚠️ Failed to delete expired file %s: %v", file.FilePath, err) - } - - // Remove from cache - s.cache.Delete(key) - expiredCount++ - } - } - } - } - - if expiredCount > 0 { - log.Printf("✅ [FILE-CACHE] Cleaned up %d expired files", expiredCount) - } -} - -// CleanupOrphanedFiles scans the uploads directory and deletes files that: -// 1. Are not tracked in the cache (orphaned after server restart) -// 2. Are older than the maxAge duration -// This ensures zero retention policy is enforced even after server restarts -func (s *FileCacheService) CleanupOrphanedFiles(uploadDir string, maxAge time.Duration) { - s.mu.RLock() - // Build a set of tracked file paths - trackedFiles := make(map[string]bool) - for _, item := range s.cache.Items() { - if file, ok := item.Object.(*CachedFile); ok { - if file.FilePath != "" { - trackedFiles[file.FilePath] = true - } - } - } - s.mu.RUnlock() - - // Scan uploads directory - entries, err := os.ReadDir(uploadDir) - if err != nil { - log.Printf("⚠️ [CLEANUP] Failed to read uploads directory: %v", err) - return - } - - now := time.Now() - orphanedCount := 0 - expiredCount := 0 - - for _, entry := range entries { - if entry.IsDir() { - continue - } - - filePath := fmt.Sprintf("%s/%s", uploadDir, entry.Name()) - - // Get file info - info, err := entry.Info() - if err != nil { - log.Printf("⚠️ [CLEANUP] Failed to get file info for %s: %v", entry.Name(), err) - continue - } - - fileAge := now.Sub(info.ModTime()) - - // Check if file is tracked in cache - if !trackedFiles[filePath] { - // File is orphaned (not in cache) - delete if older than 5 minutes - // Give a small grace period for files being uploaded - if fileAge > 5*time.Minute { - log.Printf("🗑️ [CLEANUP] Deleting orphaned file: %s (age: %v)", entry.Name(), fileAge) - if err := os.Remove(filePath); err != nil && !os.IsNotExist(err) { - log.Printf("⚠️ [CLEANUP] Failed to delete orphaned file %s: %v", entry.Name(), err) - } else { - orphanedCount++ - } - } - } else if fileAge > maxAge { - // File is tracked but expired - will be cleaned by cache eviction - // Just log for now - expiredCount++ - } - } - - if orphanedCount > 0 || expiredCount > 0 { - log.Printf("✅ [CLEANUP] Scan complete: deleted %d orphaned files, found %d expired tracked files", - orphanedCount, expiredCount) - } -} - -// RunStartupCleanup performs initial cleanup when server starts -// This is critical for enforcing zero retention policy after restarts -func (s *FileCacheService) RunStartupCleanup(uploadDir string) { - log.Printf("🧹 [STARTUP] Running startup file cleanup in %s...", uploadDir) - - entries, err := os.ReadDir(uploadDir) - if err != nil { - log.Printf("⚠️ [STARTUP] Failed to read uploads directory: %v", err) - return - } - - now := time.Now() - deletedCount := 0 - - for _, entry := range entries { - if entry.IsDir() { - continue - } - - filePath := fmt.Sprintf("%s/%s", uploadDir, entry.Name()) - - // Get file info - info, err := entry.Info() - if err != nil { - continue - } - - // Delete any file older than 1 hour (matching our retention policy) - // On startup, all files are orphaned since cache is empty - if now.Sub(info.ModTime()) > 1*time.Hour { - log.Printf("🗑️ [STARTUP] Deleting stale file: %s (modified: %v ago)", - entry.Name(), now.Sub(info.ModTime())) - if err := os.Remove(filePath); err != nil && !os.IsNotExist(err) { - log.Printf("⚠️ [STARTUP] Failed to delete file %s: %v", entry.Name(), err) - } else { - deletedCount++ - } - } - } - - log.Printf("✅ [STARTUP] Startup cleanup complete: deleted %d stale files", deletedCount) -} - -// GetStats returns cache statistics -func (s *FileCacheService) GetStats() map[string]interface{} { - s.mu.RLock() - defer s.mu.RUnlock() - - items := s.cache.Items() - totalSize := int64(0) - totalWords := 0 - - for _, item := range items { - if file, ok := item.Object.(*CachedFile); ok { - totalSize += file.Size - totalWords += file.WordCount - } - } - - return map[string]interface{}{ - "total_files": len(items), - "total_size": totalSize, - "total_words": totalWords, - } -} - -// GetAllFilesByUser returns metadata for all files owned by a user (for GDPR data export) -func (s *FileCacheService) GetAllFilesByUser(userID string) []map[string]interface{} { - s.mu.RLock() - defer s.mu.RUnlock() - - var fileMetadata []map[string]interface{} - - for _, item := range s.cache.Items() { - if file, ok := item.Object.(*CachedFile); ok { - if file.UserID == userID { - metadata := map[string]interface{}{ - "file_id": file.FileID, - "filename": file.Filename, - "mime_type": file.MimeType, - "size": file.Size, - "uploaded_at": file.UploadedAt.Format(time.RFC3339), - "conversation_id": file.ConversationID, - } - - // Add PDF-specific fields if applicable - if file.MimeType == "application/pdf" { - metadata["page_count"] = file.PageCount - metadata["word_count"] = file.WordCount - } - - fileMetadata = append(fileMetadata, metadata) - } - } - } - - return fileMetadata -} - -// DeleteAllFilesByUser deletes all files owned by a user (for GDPR compliance) -func (s *FileCacheService) DeleteAllFilesByUser(userID string) (int, error) { - s.mu.Lock() - defer s.mu.Unlock() - - deletedCount := 0 - - for key, item := range s.cache.Items() { - if file, ok := item.Object.(*CachedFile); ok { - if file.UserID == userID { - log.Printf("🗑️ [GDPR] Deleting file %s (%s) for user %s", file.FileID, file.Filename, userID) - file.SecureWipe() - s.cache.Delete(key) - deletedCount++ - } - } - } - - log.Printf("✅ [GDPR] Deleted %d files for user %s", deletedCount, userID) - return deletedCount, nil -} diff --git a/backend/internal/services/image_edit_provider_service.go b/backend/internal/services/image_edit_provider_service.go deleted file mode 100644 index a895762a..00000000 --- a/backend/internal/services/image_edit_provider_service.go +++ /dev/null @@ -1,94 +0,0 @@ -package services - -import ( - "claraverse/internal/models" - "log" - "sync" -) - -// ImageEditProviderConfig holds the configuration for an image editing provider -type ImageEditProviderConfig struct { - Name string - BaseURL string - APIKey string - Favicon string -} - -// ImageEditProviderService manages image editing providers -type ImageEditProviderService struct { - providers []ImageEditProviderConfig - mutex sync.RWMutex -} - -var ( - imageEditProviderInstance *ImageEditProviderService - imageEditProviderOnce sync.Once -) - -// GetImageEditProviderService returns the singleton image edit provider service -func GetImageEditProviderService() *ImageEditProviderService { - imageEditProviderOnce.Do(func() { - imageEditProviderInstance = &ImageEditProviderService{ - providers: make([]ImageEditProviderConfig, 0), - } - }) - return imageEditProviderInstance -} - -// LoadFromProviders loads image edit providers from the providers config -// This is called during provider sync -func (s *ImageEditProviderService) LoadFromProviders(providers []models.ProviderConfig) { - s.mutex.Lock() - defer s.mutex.Unlock() - - // Clear existing providers - s.providers = make([]ImageEditProviderConfig, 0) - - for _, p := range providers { - // Only load enabled providers with image_edit_only flag - if p.Enabled && p.ImageEditOnly { - config := ImageEditProviderConfig{ - Name: p.Name, - BaseURL: p.BaseURL, - APIKey: p.APIKey, - Favicon: p.Favicon, - } - s.providers = append(s.providers, config) - log.Printf("🖌️ [IMAGE-EDIT-PROVIDER] Loaded image edit provider: %s", p.Name) - } - } - - log.Printf("🖌️ [IMAGE-EDIT-PROVIDER] Total image edit providers loaded: %d", len(s.providers)) -} - -// GetProvider returns the first enabled image edit provider -// Returns nil if no image edit providers are configured -func (s *ImageEditProviderService) GetProvider() *ImageEditProviderConfig { - s.mutex.RLock() - defer s.mutex.RUnlock() - - if len(s.providers) == 0 { - return nil - } - - // Return the first provider - return &s.providers[0] -} - -// GetAllProviders returns all configured image edit providers -func (s *ImageEditProviderService) GetAllProviders() []ImageEditProviderConfig { - s.mutex.RLock() - defer s.mutex.RUnlock() - - // Return a copy to prevent external modification - result := make([]ImageEditProviderConfig, len(s.providers)) - copy(result, s.providers) - return result -} - -// HasProvider checks if any image edit provider is configured -func (s *ImageEditProviderService) HasProvider() bool { - s.mutex.RLock() - defer s.mutex.RUnlock() - return len(s.providers) > 0 -} diff --git a/backend/internal/services/image_provider_service.go b/backend/internal/services/image_provider_service.go deleted file mode 100644 index 86eff88b..00000000 --- a/backend/internal/services/image_provider_service.go +++ /dev/null @@ -1,96 +0,0 @@ -package services - -import ( - "claraverse/internal/models" - "log" - "sync" -) - -// ImageProviderConfig holds the configuration for an image generation provider -type ImageProviderConfig struct { - Name string - BaseURL string - APIKey string - DefaultModel string - Favicon string -} - -// ImageProviderService manages image generation providers -type ImageProviderService struct { - providers []ImageProviderConfig - mutex sync.RWMutex -} - -var ( - imageProviderInstance *ImageProviderService - imageProviderOnce sync.Once -) - -// GetImageProviderService returns the singleton image provider service -func GetImageProviderService() *ImageProviderService { - imageProviderOnce.Do(func() { - imageProviderInstance = &ImageProviderService{ - providers: make([]ImageProviderConfig, 0), - } - }) - return imageProviderInstance -} - -// LoadFromProviders loads image providers from the providers config -// This is called during provider sync -func (s *ImageProviderService) LoadFromProviders(providers []models.ProviderConfig) { - s.mutex.Lock() - defer s.mutex.Unlock() - - // Clear existing providers - s.providers = make([]ImageProviderConfig, 0) - - for _, p := range providers { - // Only load enabled providers with image_only flag - if p.Enabled && p.ImageOnly { - config := ImageProviderConfig{ - Name: p.Name, - BaseURL: p.BaseURL, - APIKey: p.APIKey, - DefaultModel: p.DefaultModel, - Favicon: p.Favicon, - } - s.providers = append(s.providers, config) - log.Printf("🎨 [IMAGE-PROVIDER] Loaded image provider: %s (model: %s)", p.Name, p.DefaultModel) - } - } - - log.Printf("🎨 [IMAGE-PROVIDER] Total image providers loaded: %d", len(s.providers)) -} - -// GetProvider returns the first enabled image provider -// Returns nil if no image providers are configured -func (s *ImageProviderService) GetProvider() *ImageProviderConfig { - s.mutex.RLock() - defer s.mutex.RUnlock() - - if len(s.providers) == 0 { - return nil - } - - // Return the first provider (could be enhanced to support multiple providers) - return &s.providers[0] -} - -// GetAllProviders returns all configured image providers -func (s *ImageProviderService) GetAllProviders() []ImageProviderConfig { - s.mutex.RLock() - defer s.mutex.RUnlock() - - // Return a copy to prevent external modification - result := make([]ImageProviderConfig, len(s.providers)) - copy(result, s.providers) - return result -} - -// HasProvider checks if any image provider is configured -func (s *ImageProviderService) HasProvider() bool { - s.mutex.RLock() - defer s.mutex.RUnlock() - return len(s.providers) > 0 -} diff --git a/backend/internal/services/image_registry_service.go b/backend/internal/services/image_registry_service.go deleted file mode 100644 index aabfcb86..00000000 --- a/backend/internal/services/image_registry_service.go +++ /dev/null @@ -1,344 +0,0 @@ -package services - -import ( - "fmt" - "log" - "strings" - "sync" - "time" -) - -// ImageEntry represents a registered image in a conversation -type ImageEntry struct { - Handle string // "img-1", "img-2", etc. - FileID string // UUID from filecache - Filename string // Original or generated name - Source string // "uploaded", "generated", "edited" - SourceHandle string // For edited images, which image was source - Prompt string // For generated/edited images, the prompt used - Width int // Image width (if known) - Height int // Image height (if known) - CreatedAt time.Time // When this entry was created -} - -// ImageRegistry holds the image entries for a single conversation -type ImageRegistry struct { - entries map[string]*ImageEntry // handle -> entry - byFileID map[string]string // fileID -> handle (reverse lookup) - counter int // for generating handles - mutex sync.RWMutex -} - -// ImageRegistryService manages per-conversation image registries -type ImageRegistryService struct { - registries map[string]*ImageRegistry // conversationID -> registry - mutex sync.RWMutex -} - -var ( - imageRegistryInstance *ImageRegistryService - imageRegistryOnce sync.Once -) - -// GetImageRegistryService returns the singleton image registry service -func GetImageRegistryService() *ImageRegistryService { - imageRegistryOnce.Do(func() { - imageRegistryInstance = &ImageRegistryService{ - registries: make(map[string]*ImageRegistry), - } - log.Printf("📸 [IMAGE-REGISTRY] Service initialized") - }) - return imageRegistryInstance -} - -// getOrCreateRegistry gets or creates a registry for a conversation -func (s *ImageRegistryService) getOrCreateRegistry(conversationID string) *ImageRegistry { - s.mutex.Lock() - defer s.mutex.Unlock() - - if registry, exists := s.registries[conversationID]; exists { - return registry - } - - registry := &ImageRegistry{ - entries: make(map[string]*ImageEntry), - byFileID: make(map[string]string), - counter: 0, - } - s.registries[conversationID] = registry - return registry -} - -// generateHandle creates the next handle for a registry -func (r *ImageRegistry) generateHandle() string { - r.counter++ - return fmt.Sprintf("img-%d", r.counter) -} - -// RegisterUploadedImage registers an uploaded image and returns its handle -func (s *ImageRegistryService) RegisterUploadedImage(conversationID, fileID, filename string, width, height int) string { - registry := s.getOrCreateRegistry(conversationID) - - registry.mutex.Lock() - defer registry.mutex.Unlock() - - // Check if already registered - if handle, exists := registry.byFileID[fileID]; exists { - log.Printf("📸 [IMAGE-REGISTRY] Image already registered: %s -> %s", fileID, handle) - return handle - } - - handle := registry.generateHandle() - entry := &ImageEntry{ - Handle: handle, - FileID: fileID, - Filename: filename, - Source: "uploaded", - Width: width, - Height: height, - CreatedAt: time.Now(), - } - - registry.entries[handle] = entry - registry.byFileID[fileID] = handle - - log.Printf("📸 [IMAGE-REGISTRY] Registered uploaded image: %s -> %s (%s)", handle, fileID, filename) - return handle -} - -// RegisterGeneratedImage registers a generated image and returns its handle -func (s *ImageRegistryService) RegisterGeneratedImage(conversationID, fileID, prompt string) string { - registry := s.getOrCreateRegistry(conversationID) - - registry.mutex.Lock() - defer registry.mutex.Unlock() - - // Check if already registered - if handle, exists := registry.byFileID[fileID]; exists { - return handle - } - - handle := registry.generateHandle() - - // Create a short filename from prompt - filename := truncatePromptForFilename(prompt) + ".png" - - entry := &ImageEntry{ - Handle: handle, - FileID: fileID, - Filename: filename, - Source: "generated", - Prompt: prompt, - Width: 1024, // Default generation size - Height: 1024, - CreatedAt: time.Now(), - } - - registry.entries[handle] = entry - registry.byFileID[fileID] = handle - - log.Printf("📸 [IMAGE-REGISTRY] Registered generated image: %s -> %s", handle, fileID) - return handle -} - -// RegisterEditedImage registers an edited image and returns its handle -func (s *ImageRegistryService) RegisterEditedImage(conversationID, fileID, sourceHandle, prompt string) string { - registry := s.getOrCreateRegistry(conversationID) - - registry.mutex.Lock() - defer registry.mutex.Unlock() - - // Check if already registered - if handle, exists := registry.byFileID[fileID]; exists { - return handle - } - - handle := registry.generateHandle() - - // Create filename based on source and edit prompt - filename := fmt.Sprintf("edited_%s_%s.png", sourceHandle, truncatePromptForFilename(prompt)) - - entry := &ImageEntry{ - Handle: handle, - FileID: fileID, - Filename: filename, - Source: "edited", - SourceHandle: sourceHandle, - Prompt: prompt, - Width: 1024, // Edited images typically same size - Height: 1024, - CreatedAt: time.Now(), - } - - registry.entries[handle] = entry - registry.byFileID[fileID] = handle - - log.Printf("📸 [IMAGE-REGISTRY] Registered edited image: %s -> %s (from %s)", handle, fileID, sourceHandle) - return handle -} - -// GetByHandle returns an image entry by its handle -func (s *ImageRegistryService) GetByHandle(conversationID, handle string) *ImageEntry { - s.mutex.RLock() - registry, exists := s.registries[conversationID] - s.mutex.RUnlock() - - if !exists { - return nil - } - - registry.mutex.RLock() - defer registry.mutex.RUnlock() - - return registry.entries[handle] -} - -// GetByFileID returns an image entry by its file ID -func (s *ImageRegistryService) GetByFileID(conversationID, fileID string) *ImageEntry { - s.mutex.RLock() - registry, exists := s.registries[conversationID] - s.mutex.RUnlock() - - if !exists { - return nil - } - - registry.mutex.RLock() - defer registry.mutex.RUnlock() - - handle, exists := registry.byFileID[fileID] - if !exists { - return nil - } - - return registry.entries[handle] -} - -// ListImages returns all images in a conversation -func (s *ImageRegistryService) ListImages(conversationID string) []*ImageEntry { - s.mutex.RLock() - registry, exists := s.registries[conversationID] - s.mutex.RUnlock() - - if !exists { - return nil - } - - registry.mutex.RLock() - defer registry.mutex.RUnlock() - - result := make([]*ImageEntry, 0, len(registry.entries)) - for _, entry := range registry.entries { - result = append(result, entry) - } - - return result -} - -// ListHandles returns all available handles for a conversation (for error messages) -func (s *ImageRegistryService) ListHandles(conversationID string) []string { - s.mutex.RLock() - registry, exists := s.registries[conversationID] - s.mutex.RUnlock() - - if !exists { - return nil - } - - registry.mutex.RLock() - defer registry.mutex.RUnlock() - - handles := make([]string, 0, len(registry.entries)) - for handle := range registry.entries { - handles = append(handles, handle) - } - - return handles -} - -// BuildSystemContext creates the system prompt injection for available images -func (s *ImageRegistryService) BuildSystemContext(conversationID string) string { - images := s.ListImages(conversationID) - if len(images) == 0 { - return "" - } - - var sb strings.Builder - sb.WriteString("[Available Images]\n") - sb.WriteString("You have access to the following images in this conversation. Use the image ID (e.g., 'img-1') when calling the edit_image tool:\n") - - for _, img := range images { - sb.WriteString(fmt.Sprintf("- %s: \"%s\"", img.Handle, img.Filename)) - - // Add source info - switch img.Source { - case "uploaded": - sb.WriteString(" (uploaded by user") - case "generated": - sb.WriteString(" (generated") - case "edited": - sb.WriteString(fmt.Sprintf(" (edited from %s", img.SourceHandle)) - } - - // Add dimensions if known - if img.Width > 0 && img.Height > 0 { - sb.WriteString(fmt.Sprintf(", %dx%d", img.Width, img.Height)) - } - - sb.WriteString(")\n") - } - - return sb.String() -} - -// CleanupConversation removes all registry data for a conversation -func (s *ImageRegistryService) CleanupConversation(conversationID string) { - s.mutex.Lock() - defer s.mutex.Unlock() - - if _, exists := s.registries[conversationID]; exists { - delete(s.registries, conversationID) - log.Printf("📸 [IMAGE-REGISTRY] Cleaned up conversation: %s", conversationID) - } -} - -// HasImages checks if a conversation has any registered images -func (s *ImageRegistryService) HasImages(conversationID string) bool { - s.mutex.RLock() - registry, exists := s.registries[conversationID] - s.mutex.RUnlock() - - if !exists { - return false - } - - registry.mutex.RLock() - defer registry.mutex.RUnlock() - - return len(registry.entries) > 0 -} - -// truncatePromptForFilename creates a short filename-safe string from a prompt -func truncatePromptForFilename(prompt string) string { - // Take first 30 chars, make filename-safe - if len(prompt) > 30 { - prompt = prompt[:30] - } - - // Replace unsafe characters - safe := strings.Map(func(r rune) rune { - if (r >= 'a' && r <= 'z') || (r >= 'A' && r <= 'Z') || (r >= '0' && r <= '9') || r == '_' || r == '-' { - return r - } - if r == ' ' { - return '_' - } - return -1 // Remove other characters - }, prompt) - - if safe == "" { - safe = "image" - } - - return safe -} diff --git a/backend/internal/services/mcp_bridge_service.go b/backend/internal/services/mcp_bridge_service.go deleted file mode 100644 index 292961d6..00000000 --- a/backend/internal/services/mcp_bridge_service.go +++ /dev/null @@ -1,296 +0,0 @@ -package services - -import ( - "encoding/json" - "fmt" - "log" - "sync" - "time" - - "claraverse/internal/database" - "claraverse/internal/models" - "claraverse/internal/tools" - "github.com/google/uuid" -) - -// MCPBridgeService manages MCP client connections and tool routing -type MCPBridgeService struct { - db *database.DB - connections map[string]*models.MCPConnection // clientID -> connection - userConns map[string]string // userID -> clientID - registry *tools.Registry - mutex sync.RWMutex -} - -// NewMCPBridgeService creates a new MCP bridge service -func NewMCPBridgeService(db *database.DB, registry *tools.Registry) *MCPBridgeService { - return &MCPBridgeService{ - db: db, - connections: make(map[string]*models.MCPConnection), - userConns: make(map[string]string), - registry: registry, - } -} - -// RegisterClient registers a new MCP client connection -func (s *MCPBridgeService) RegisterClient(userID string, registration *models.MCPToolRegistration) (*models.MCPConnection, error) { - s.mutex.Lock() - defer s.mutex.Unlock() - - // Check if user already has a connection - if existingClientID, exists := s.userConns[userID]; exists { - // Disconnect existing connection - if existingConn, ok := s.connections[existingClientID]; ok { - log.Printf("Disconnecting existing MCP client for user %s", userID) - s.disconnectClientLocked(existingClientID, existingConn) - } - } - - // Create new connection - conn := &models.MCPConnection{ - ID: uuid.New().String(), - UserID: userID, - ClientID: registration.ClientID, - ClientVersion: registration.ClientVersion, - Platform: registration.Platform, - ConnectedAt: time.Now(), - LastHeartbeat: time.Now(), - IsActive: true, - Tools: registration.Tools, - WriteChan: make(chan models.MCPServerMessage, 100), - StopChan: make(chan bool, 1), - PendingResults: make(map[string]chan models.MCPToolResult), - } - - // Store in memory - s.connections[registration.ClientID] = conn - s.userConns[userID] = registration.ClientID - - // Store in database - _, err := s.db.Exec(` - INSERT INTO mcp_connections (user_id, client_id, client_version, platform, connected_at, last_heartbeat, is_active) - VALUES (?, ?, ?, ?, ?, ?, ?) - `, userID, registration.ClientID, registration.ClientVersion, registration.Platform, conn.ConnectedAt, conn.LastHeartbeat, true) - - if err != nil { - delete(s.connections, registration.ClientID) - delete(s.userConns, userID) - return nil, fmt.Errorf("failed to store connection in database: %w", err) - } - - // Get connection ID from database - var dbConnID int64 - err = s.db.QueryRow("SELECT id FROM mcp_connections WHERE client_id = ?", registration.ClientID).Scan(&dbConnID) - if err != nil { - log.Printf("Warning: Failed to get connection ID from database: %v", err) - } - - // Register tools in registry and database - for _, tool := range registration.Tools { - // Register in registry - err := s.registry.RegisterUserTool(userID, &tools.Tool{ - Name: tool.Name, - Description: tool.Description, - Parameters: tool.Parameters, - Source: tools.ToolSourceMCPLocal, - UserID: userID, - Execute: nil, // MCP tools don't have direct execute functions - }) - - if err != nil { - log.Printf("Warning: Failed to register tool %s: %v", tool.Name, err) - continue - } - - // Store tool in database - toolDefJSON, _ := json.Marshal(tool) - _, err = s.db.Exec(` - INSERT OR REPLACE INTO mcp_tools (user_id, connection_id, tool_name, tool_definition) - VALUES (?, ?, ?, ?) - `, userID, dbConnID, tool.Name, string(toolDefJSON)) - - if err != nil { - log.Printf("Warning: Failed to store tool %s in database: %v", tool.Name, err) - } - } - - log.Printf("✅ MCP client registered: user=%s, client=%s, tools=%d", userID, registration.ClientID, len(registration.Tools)) - - // Send acknowledgment - go func() { - conn.WriteChan <- models.MCPServerMessage{ - Type: "ack", - Payload: map[string]interface{}{ - "status": "connected", - "tools_registered": len(registration.Tools), - }, - } - }() - - return conn, nil -} - -// DisconnectClient handles client disconnection -func (s *MCPBridgeService) DisconnectClient(clientID string) error { - s.mutex.Lock() - defer s.mutex.Unlock() - - conn, exists := s.connections[clientID] - if !exists { - return fmt.Errorf("client %s not found", clientID) - } - - s.disconnectClientLocked(clientID, conn) - return nil -} - -// disconnectClientLocked handles disconnection (must be called with lock held) -func (s *MCPBridgeService) disconnectClientLocked(clientID string, conn *models.MCPConnection) { - // Mark as inactive in database - _, err := s.db.Exec("UPDATE mcp_connections SET is_active = 0 WHERE client_id = ?", clientID) - if err != nil { - log.Printf("Warning: Failed to mark connection as inactive: %v", err) - } - - // Unregister all tools - s.registry.UnregisterAllUserTools(conn.UserID) - - // Clean up memory - delete(s.connections, clientID) - delete(s.userConns, conn.UserID) - - // Close channels - close(conn.StopChan) - close(conn.WriteChan) - - log.Printf("🔌 MCP client disconnected: user=%s, client=%s", conn.UserID, clientID) -} - -// UpdateHeartbeat updates the last heartbeat time for a client -func (s *MCPBridgeService) UpdateHeartbeat(clientID string) error { - s.mutex.Lock() - defer s.mutex.Unlock() - - conn, exists := s.connections[clientID] - if !exists { - return fmt.Errorf("client %s not found", clientID) - } - - conn.LastHeartbeat = time.Now() - - // Update in database - _, err := s.db.Exec("UPDATE mcp_connections SET last_heartbeat = ? WHERE client_id = ?", conn.LastHeartbeat, clientID) - return err -} - -// ExecuteToolOnClient sends a tool execution request to the MCP client -func (s *MCPBridgeService) ExecuteToolOnClient(userID string, toolName string, args map[string]interface{}, timeout time.Duration) (string, error) { - s.mutex.RLock() - clientID, exists := s.userConns[userID] - if !exists { - s.mutex.RUnlock() - return "", fmt.Errorf("no MCP client connected for user %s", userID) - } - - conn, connExists := s.connections[clientID] - s.mutex.RUnlock() - - if !connExists { - return "", fmt.Errorf("MCP client connection not found") - } - - // Generate unique call ID - callID := uuid.New().String() - - // Create result channel for this call - resultChan := make(chan models.MCPToolResult, 1) - conn.PendingResults[callID] = resultChan - - // Create tool call message - toolCall := models.MCPToolCall{ - CallID: callID, - ToolName: toolName, - Arguments: args, - Timeout: int(timeout.Seconds()), - } - - // Send to client - select { - case conn.WriteChan <- models.MCPServerMessage{ - Type: "tool_call", - Payload: map[string]interface{}{ - "call_id": toolCall.CallID, - "tool_name": toolCall.ToolName, - "arguments": toolCall.Arguments, - "timeout": toolCall.Timeout, - }, - }: - // Message sent successfully - case <-time.After(5 * time.Second): - delete(conn.PendingResults, callID) - return "", fmt.Errorf("timeout sending tool call to client") - } - - // Wait for result with timeout - select { - case result := <-resultChan: - delete(conn.PendingResults, callID) - if result.Success { - return result.Result, nil - } else { - return "", fmt.Errorf("%s", result.Error) - } - case <-time.After(timeout): - delete(conn.PendingResults, callID) - return "", fmt.Errorf("tool execution timeout after %v", timeout) - } -} - -// GetConnection retrieves a connection by client ID -func (s *MCPBridgeService) GetConnection(clientID string) (*models.MCPConnection, bool) { - s.mutex.RLock() - defer s.mutex.RUnlock() - conn, exists := s.connections[clientID] - return conn, exists -} - -// GetUserConnection retrieves a connection by user ID -func (s *MCPBridgeService) GetUserConnection(userID string) (*models.MCPConnection, bool) { - s.mutex.RLock() - defer s.mutex.RUnlock() - - clientID, exists := s.userConns[userID] - if !exists { - return nil, false - } - - conn, connExists := s.connections[clientID] - return conn, connExists -} - -// IsUserConnected checks if a user has an active MCP client -func (s *MCPBridgeService) IsUserConnected(userID string) bool { - s.mutex.RLock() - defer s.mutex.RUnlock() - _, exists := s.userConns[userID] - return exists -} - -// GetConnectionCount returns the number of active connections -func (s *MCPBridgeService) GetConnectionCount() int { - s.mutex.RLock() - defer s.mutex.RUnlock() - return len(s.connections) -} - -// LogToolExecution logs a tool execution for audit purposes -func (s *MCPBridgeService) LogToolExecution(userID, toolName, conversationID string, executionTimeMs int, success bool, errorMsg string) { - _, err := s.db.Exec(` - INSERT INTO mcp_audit_log (user_id, tool_name, conversation_id, execution_time_ms, success, error_message) - VALUES (?, ?, ?, ?, ?, ?) - `, userID, toolName, conversationID, executionTimeMs, success, errorMsg) - - if err != nil { - log.Printf("Warning: Failed to log tool execution: %v", err) - } -} diff --git a/backend/internal/services/memory_decay_service.go b/backend/internal/services/memory_decay_service.go deleted file mode 100644 index b642124a..00000000 --- a/backend/internal/services/memory_decay_service.go +++ /dev/null @@ -1,304 +0,0 @@ -package services - -import ( - "context" - "fmt" - "log" - "math" - "time" - - "claraverse/internal/database" - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/bson/primitive" - "go.mongodb.org/mongo-driver/mongo" -) - -// MemoryDecayService handles memory scoring and archival using PageRank-like algorithm -type MemoryDecayService struct { - mongodb *database.MongoDB - collection *mongo.Collection -} - -// DecayConfig holds the decay algorithm configuration -type DecayConfig struct { - RecencyWeight float64 // Default: 0.4 - FrequencyWeight float64 // Default: 0.3 - EngagementWeight float64 // Default: 0.3 - RecencyDecayRate float64 // Default: 0.05 - FrequencyMax int64 // Default: 20 - ArchiveThreshold float64 // Default: 0.15 -} - -// DefaultDecayConfig returns the default decay configuration -func DefaultDecayConfig() DecayConfig { - return DecayConfig{ - RecencyWeight: 0.4, - FrequencyWeight: 0.3, - EngagementWeight: 0.3, - RecencyDecayRate: 0.05, - FrequencyMax: 20, - ArchiveThreshold: 0.15, - } -} - -// NewMemoryDecayService creates a new memory decay service -func NewMemoryDecayService(mongodb *database.MongoDB) *MemoryDecayService { - return &MemoryDecayService{ - mongodb: mongodb, - collection: mongodb.Collection(database.CollectionMemories), - } -} - -// RunDecayJob runs the full decay job for all users -func (s *MemoryDecayService) RunDecayJob(ctx context.Context) error { - log.Printf("🔄 [MEMORY-DECAY] Starting decay job") - - config := DefaultDecayConfig() - - // Get all unique user IDs with active memories - userIDs, err := s.getActiveUserIDs(ctx) - if err != nil { - return fmt.Errorf("failed to get active user IDs: %w", err) - } - - log.Printf("📊 [MEMORY-DECAY] Processing %d users with active memories", len(userIDs)) - - // Process each user - totalRecalculated := 0 - totalArchived := 0 - - for _, userID := range userIDs { - recalculated, archived, err := s.RunDecayJobForUser(ctx, userID, config) - if err != nil { - log.Printf("⚠️ [MEMORY-DECAY] Failed to process user %s: %v", userID, err) - continue - } - - totalRecalculated += recalculated - totalArchived += archived - } - - log.Printf("✅ [MEMORY-DECAY] Decay job completed: %d memories recalculated, %d archived", totalRecalculated, totalArchived) - return nil -} - -// RunDecayJobForUser runs decay job for a specific user -func (s *MemoryDecayService) RunDecayJobForUser(ctx context.Context, userID string, config DecayConfig) (int, int, error) { - // Get all active memories for user - filter := bson.M{ - "userId": userID, - "isArchived": false, - } - - cursor, err := s.collection.Find(ctx, filter) - if err != nil { - return 0, 0, fmt.Errorf("failed to find memories: %w", err) - } - defer cursor.Close(ctx) - - var memories []struct { - ID primitive.ObjectID `bson:"_id"` - AccessCount int64 `bson:"accessCount"` - LastAccessedAt *time.Time `bson:"lastAccessedAt"` - SourceEngagement float64 `bson:"sourceEngagement"` - CreatedAt time.Time `bson:"createdAt"` - } - - if err := cursor.All(ctx, &memories); err != nil { - return 0, 0, fmt.Errorf("failed to decode memories: %w", err) - } - - if len(memories) == 0 { - return 0, 0, nil - } - - // Calculate scores for all memories - now := time.Now() - memoriesToArchive := []primitive.ObjectID{} - memoriesToUpdate := []mongo.WriteModel{} - - for _, mem := range memories { - newScore := s.calculateMemoryScore(mem.AccessCount, mem.LastAccessedAt, mem.SourceEngagement, mem.CreatedAt, now, config) - - // Check if should be archived - if newScore < config.ArchiveThreshold { - memoriesToArchive = append(memoriesToArchive, mem.ID) - } else { - // Update score - update := mongo.NewUpdateOneModel(). - SetFilter(bson.M{"_id": mem.ID}). - SetUpdate(bson.M{ - "$set": bson.M{ - "score": newScore, - "updatedAt": now, - }, - }) - memoriesToUpdate = append(memoriesToUpdate, update) - } - } - - // Bulk update scores - recalculated := 0 - if len(memoriesToUpdate) > 0 { - result, err := s.collection.BulkWrite(ctx, memoriesToUpdate) - if err != nil { - log.Printf("⚠️ [MEMORY-DECAY] Failed to update scores for user %s: %v", userID, err) - } else { - recalculated = int(result.ModifiedCount) - } - } - - // Archive low-score memories - archived := 0 - if len(memoriesToArchive) > 0 { - archived, err = s.archiveMemoriesBulk(ctx, memoriesToArchive, now) - if err != nil { - log.Printf("⚠️ [MEMORY-DECAY] Failed to archive memories for user %s: %v", userID, err) - } - } - - log.Printf("📊 [MEMORY-DECAY] User %s: %d memories recalculated, %d archived", userID, recalculated, archived) - return recalculated, archived, nil -} - -// calculateMemoryScore calculates the PageRank-like score for a memory -func (s *MemoryDecayService) calculateMemoryScore( - accessCount int64, - lastAccessedAt *time.Time, - sourceEngagement float64, - createdAt time.Time, - now time.Time, - config DecayConfig, -) float64 { - // Calculate recency score - recencyScore := s.calculateRecencyScore(lastAccessedAt, createdAt, now, config.RecencyDecayRate) - - // Calculate frequency score - frequencyScore := s.calculateFrequencyScore(accessCount, config.FrequencyMax) - - // Engagement score is directly from source conversation - engagementScore := sourceEngagement - - // Weighted combination (PageRank-like) - finalScore := (config.RecencyWeight * recencyScore) + - (config.FrequencyWeight * frequencyScore) + - (config.EngagementWeight * engagementScore) - - return finalScore -} - -// calculateRecencyScore calculates recency score using exponential decay -// RecencyScore = exp(-0.05 × days_since_last_access) -// - Recent: 1.0 -// - 1 week: ~0.70 -// - 1 month: ~0.22 -// - 3 months: ~0.01 -func (s *MemoryDecayService) calculateRecencyScore(lastAccessedAt *time.Time, createdAt time.Time, now time.Time, decayRate float64) float64 { - var referenceTime time.Time - - // Use last accessed time if available, otherwise use creation time - if lastAccessedAt != nil { - referenceTime = *lastAccessedAt - } else { - referenceTime = createdAt - } - - // Calculate days since last access/creation - daysSince := now.Sub(referenceTime).Hours() / 24.0 - - // Exponential decay: exp(-decayRate × days) - recencyScore := math.Exp(-decayRate * daysSince) - - return recencyScore -} - -// calculateFrequencyScore calculates frequency score based on access count -// FrequencyScore = min(1.0, access_count / max) -// - 0 accesses: 0.0 -// - 10 accesses: 0.5 (if max=20) -// - 20+ accesses: 1.0 -func (s *MemoryDecayService) calculateFrequencyScore(accessCount int64, frequencyMax int64) float64 { - if accessCount <= 0 { - return 0.0 - } - - frequencyScore := float64(accessCount) / float64(frequencyMax) - - // Cap at 1.0 - if frequencyScore > 1.0 { - frequencyScore = 1.0 - } - - return frequencyScore -} - -// archiveMemoriesBulk archives multiple memories at once -func (s *MemoryDecayService) archiveMemoriesBulk(ctx context.Context, memoryIDs []primitive.ObjectID, now time.Time) (int, error) { - filter := bson.M{ - "_id": bson.M{"$in": memoryIDs}, - } - - update := bson.M{ - "$set": bson.M{ - "isArchived": true, - "archivedAt": now, - "updatedAt": now, - }, - } - - result, err := s.collection.UpdateMany(ctx, filter, update) - if err != nil { - return 0, fmt.Errorf("failed to archive memories: %w", err) - } - - log.Printf("📦 [MEMORY-DECAY] Archived %d memories", result.ModifiedCount) - return int(result.ModifiedCount), nil -} - -// getActiveUserIDs gets all unique user IDs with active memories -func (s *MemoryDecayService) getActiveUserIDs(ctx context.Context) ([]string, error) { - filter := bson.M{"isArchived": false} - - distinctUserIDs, err := s.collection.Distinct(ctx, "userId", filter) - if err != nil { - return nil, fmt.Errorf("failed to get distinct user IDs: %w", err) - } - - userIDs := make([]string, 0, len(distinctUserIDs)) - for _, id := range distinctUserIDs { - if userID, ok := id.(string); ok { - userIDs = append(userIDs, userID) - } - } - - return userIDs, nil -} - -// GetMemoryScore calculates and returns the current score for a specific memory (for testing/debugging) -func (s *MemoryDecayService) GetMemoryScore(ctx context.Context, memoryID primitive.ObjectID) (float64, error) { - var memory struct { - AccessCount int64 `bson:"accessCount"` - LastAccessedAt *time.Time `bson:"lastAccessedAt"` - SourceEngagement float64 `bson:"sourceEngagement"` - CreatedAt time.Time `bson:"createdAt"` - } - - err := s.collection.FindOne(ctx, bson.M{"_id": memoryID}).Decode(&memory) - if err != nil { - return 0, fmt.Errorf("failed to find memory: %w", err) - } - - config := DefaultDecayConfig() - now := time.Now() - - score := s.calculateMemoryScore( - memory.AccessCount, - memory.LastAccessedAt, - memory.SourceEngagement, - memory.CreatedAt, - now, - config, - ) - - return score, nil -} diff --git a/backend/internal/services/memory_decay_service_test.go b/backend/internal/services/memory_decay_service_test.go deleted file mode 100644 index bae9cf9c..00000000 --- a/backend/internal/services/memory_decay_service_test.go +++ /dev/null @@ -1,352 +0,0 @@ -package services - -import ( - "math" - "testing" - "time" -) - -// TestCalculateRecencyScore tests the recency score calculation -func TestCalculateRecencyScore(t *testing.T) { - service := &MemoryDecayService{} - config := DefaultDecayConfig() - now := time.Now() - - tests := []struct { - name string - lastAccessedAt *time.Time - createdAt time.Time - expectedScore float64 - tolerance float64 - }{ - { - name: "Just accessed (0 days)", - lastAccessedAt: &now, - createdAt: now.AddDate(0, 0, -30), - expectedScore: 1.0, - tolerance: 0.01, - }, - { - name: "1 week ago (~0.70)", - lastAccessedAt: timePtr(now.AddDate(0, 0, -7)), - createdAt: now.AddDate(0, 0, -30), - expectedScore: 0.70, - tolerance: 0.05, - }, - { - name: "1 month ago (~0.22)", - lastAccessedAt: timePtr(now.AddDate(0, 0, -30)), - createdAt: now.AddDate(0, 0, -60), - expectedScore: 0.22, - tolerance: 0.05, - }, - { - name: "3 months ago (~0.01)", - lastAccessedAt: timePtr(now.AddDate(0, 0, -90)), - createdAt: now.AddDate(0, 0, -120), - expectedScore: 0.01, - tolerance: 0.02, - }, - { - name: "Never accessed (use createdAt)", - lastAccessedAt: nil, - createdAt: now.AddDate(0, 0, -7), - expectedScore: 0.70, - tolerance: 0.05, - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - score := service.calculateRecencyScore(tt.lastAccessedAt, tt.createdAt, now, config.RecencyDecayRate) - if math.Abs(score-tt.expectedScore) > tt.tolerance { - t.Errorf("Expected score ~%.2f, got %.2f (tolerance: %.2f)", tt.expectedScore, score, tt.tolerance) - } - }) - } -} - -// TestCalculateFrequencyScore tests the frequency score calculation -func TestCalculateFrequencyScore(t *testing.T) { - service := &MemoryDecayService{} - config := DefaultDecayConfig() - - tests := []struct { - name string - accessCount int64 - expectedScore float64 - }{ - { - name: "0 accesses", - accessCount: 0, - expectedScore: 0.0, - }, - { - name: "10 accesses (50%)", - accessCount: 10, - expectedScore: 0.5, - }, - { - name: "20 accesses (100%)", - accessCount: 20, - expectedScore: 1.0, - }, - { - name: "40 accesses (capped at 100%)", - accessCount: 40, - expectedScore: 1.0, - }, - { - name: "5 accesses (25%)", - accessCount: 5, - expectedScore: 0.25, - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - score := service.calculateFrequencyScore(tt.accessCount, config.FrequencyMax) - if math.Abs(score-tt.expectedScore) > 0.01 { - t.Errorf("Expected score %.2f, got %.2f", tt.expectedScore, score) - } - }) - } -} - -// TestCalculateMemoryScore tests the complete PageRank algorithm -func TestCalculateMemoryScore(t *testing.T) { - service := &MemoryDecayService{} - config := DefaultDecayConfig() - now := time.Now() - - tests := []struct { - name string - accessCount int64 - lastAccessedAt *time.Time - sourceEngagement float64 - createdAt time.Time - minScore float64 - maxScore float64 - description string - }{ - { - name: "High quality memory (recent, frequent, engaging)", - accessCount: 25, - lastAccessedAt: &now, - sourceEngagement: 0.9, - createdAt: now.AddDate(0, 0, -30), - minScore: 0.85, - maxScore: 1.0, - description: "Should have very high score", - }, - { - name: "Medium quality memory", - accessCount: 10, - lastAccessedAt: timePtr(now.AddDate(0, 0, -7)), - sourceEngagement: 0.6, - createdAt: now.AddDate(0, 0, -30), - minScore: 0.50, - maxScore: 0.70, - description: "Should have medium score", - }, - { - name: "Low quality memory (old, never accessed, low engagement)", - accessCount: 0, - lastAccessedAt: nil, - sourceEngagement: 0.2, - createdAt: now.AddDate(0, 0, -90), - minScore: 0.0, - maxScore: 0.15, - description: "Should be below archive threshold", - }, - { - name: "Decaying memory (moderately old, few accesses)", - accessCount: 3, - lastAccessedAt: timePtr(now.AddDate(0, 0, -30)), - sourceEngagement: 0.5, - createdAt: now.AddDate(0, 0, -60), - minScore: 0.15, - maxScore: 0.35, - description: "Should be approaching archive threshold", - }, - { - name: "High engagement saves old memory", - accessCount: 5, - lastAccessedAt: timePtr(now.AddDate(0, 0, -60)), - sourceEngagement: 0.95, - createdAt: now.AddDate(0, 0, -90), - minScore: 0.30, - maxScore: 0.50, - description: "High engagement keeps it above archive threshold", - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - score := service.calculateMemoryScore( - tt.accessCount, - tt.lastAccessedAt, - tt.sourceEngagement, - tt.createdAt, - now, - config, - ) - - if score < tt.minScore || score > tt.maxScore { - t.Errorf("%s: Expected score between %.2f and %.2f, got %.2f", - tt.description, tt.minScore, tt.maxScore, score) - } - - t.Logf("Score: %.3f (recency: %.2f, frequency: %.2f, engagement: %.2f)", - score, - service.calculateRecencyScore(tt.lastAccessedAt, tt.createdAt, now, config.RecencyDecayRate), - service.calculateFrequencyScore(tt.accessCount, config.FrequencyMax), - tt.sourceEngagement, - ) - }) - } -} - -// TestArchiveThreshold ensures memories below threshold get archived -func TestArchiveThreshold(t *testing.T) { - service := &MemoryDecayService{} - config := DefaultDecayConfig() - now := time.Now() - - // Create a memory that should be archived - accessCount := int64(0) - lastAccessedAt := (*time.Time)(nil) - sourceEngagement := 0.2 - createdAt := now.AddDate(0, 0, -90) - - score := service.calculateMemoryScore( - accessCount, - lastAccessedAt, - sourceEngagement, - createdAt, - now, - config, - ) - - if score >= config.ArchiveThreshold { - t.Errorf("Expected score below archive threshold (%.2f), got %.2f", config.ArchiveThreshold, score) - } -} - -// TestDecayConfigWeights ensures weights add up to 1.0 -func TestDecayConfigWeights(t *testing.T) { - config := DefaultDecayConfig() - - totalWeight := config.RecencyWeight + config.FrequencyWeight + config.EngagementWeight - - if math.Abs(totalWeight-1.0) > 0.001 { - t.Errorf("Weights should add up to 1.0, got %.3f", totalWeight) - } -} - -// TestRecencyDecayFormula verifies the exponential decay formula -func TestRecencyDecayFormula(t *testing.T) { - service := &MemoryDecayService{} - now := time.Now() - - // Test known values - tests := []struct { - daysAgo int - decayRate float64 - expectedScore float64 - tolerance float64 - }{ - {daysAgo: 0, decayRate: 0.05, expectedScore: 1.0, tolerance: 0.01}, - {daysAgo: 7, decayRate: 0.05, expectedScore: 0.704, tolerance: 0.01}, - {daysAgo: 14, decayRate: 0.05, expectedScore: 0.496, tolerance: 0.01}, - {daysAgo: 30, decayRate: 0.05, expectedScore: 0.223, tolerance: 0.01}, - {daysAgo: 60, decayRate: 0.05, expectedScore: 0.050, tolerance: 0.01}, - {daysAgo: 90, decayRate: 0.05, expectedScore: 0.011, tolerance: 0.01}, - } - - for _, tt := range tests { - lastAccessed := now.AddDate(0, 0, -tt.daysAgo) - createdAt := now.AddDate(0, 0, -tt.daysAgo-30) - score := service.calculateRecencyScore(&lastAccessed, createdAt, now, tt.decayRate) - - if math.Abs(score-tt.expectedScore) > tt.tolerance { - t.Errorf("Day %d: Expected %.3f, got %.3f (diff: %.3f)", - tt.daysAgo, tt.expectedScore, score, math.Abs(score-tt.expectedScore)) - } - } -} - -// TestFrequencyScoreLinear ensures frequency score is linear up to max -func TestFrequencyScoreLinear(t *testing.T) { - service := &MemoryDecayService{} - frequencyMax := int64(20) - - for i := int64(0); i <= frequencyMax*2; i += 2 { - score := service.calculateFrequencyScore(i, frequencyMax) - - expected := math.Min(1.0, float64(i)/float64(frequencyMax)) - if math.Abs(score-expected) > 0.001 { - t.Errorf("Access count %d: Expected %.3f, got %.3f", i, expected, score) - } - } -} - -// TestMemoryLifecycle tests a realistic memory lifecycle -func TestMemoryLifecycle(t *testing.T) { - service := &MemoryDecayService{} - config := DefaultDecayConfig() - now := time.Now() - - // Day 0: Memory created with high engagement - createdAt := now.AddDate(0, 0, -90) - accessCount := int64(0) - lastAccessed := (*time.Time)(nil) - sourceEngagement := 0.85 - - // Initial score (only engagement matters) - score0 := service.calculateMemoryScore(accessCount, lastAccessed, sourceEngagement, createdAt, createdAt, config) - t.Logf("Day 0: Score %.3f", score0) - - // Day 7: Accessed once - day7 := createdAt.AddDate(0, 0, 7) - accessCount = 1 - lastAccessed = &day7 - score7 := service.calculateMemoryScore(accessCount, lastAccessed, sourceEngagement, createdAt, day7, config) - t.Logf("Day 7: Score %.3f (accessed once)", score7) - - // Day 30: Accessed 5 more times - day30 := createdAt.AddDate(0, 0, 30) - accessCount = 6 - lastAccessed = &day30 - score30 := service.calculateMemoryScore(accessCount, lastAccessed, sourceEngagement, createdAt, day30, config) - t.Logf("Day 30: Score %.3f (accessed 6 times total)", score30) - - // Day 60: No new accesses (recency drops) - day60 := createdAt.AddDate(0, 0, 60) - score60 := service.calculateMemoryScore(accessCount, lastAccessed, sourceEngagement, createdAt, day60, config) - t.Logf("Day 60: Score %.3f (no new accesses, recency drops)", score60) - - // Day 90: Still no accesses (further decay) - day90 := createdAt.AddDate(0, 0, 90) - score90 := service.calculateMemoryScore(accessCount, lastAccessed, sourceEngagement, createdAt, day90, config) - t.Logf("Day 90: Score %.3f (continued decay)", score90) - - // Score should decrease over time without accesses - if score60 >= score30 { - t.Error("Score should decrease from day 30 to day 60 without accesses") - } - if score90 >= score60 { - t.Error("Score should continue decreasing from day 60 to day 90") - } - - // Should still be above archive threshold due to high engagement - if score90 < config.ArchiveThreshold { - t.Errorf("High engagement memory should stay above archive threshold (%.2f), got %.3f", - config.ArchiveThreshold, score90) - } -} - -// Helper function to create time pointer -func timePtr(t time.Time) *time.Time { - return &t -} diff --git a/backend/internal/services/memory_extraction_service.go b/backend/internal/services/memory_extraction_service.go deleted file mode 100644 index 4c885b14..00000000 --- a/backend/internal/services/memory_extraction_service.go +++ /dev/null @@ -1,717 +0,0 @@ -package services - -import ( - "bytes" - "context" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "strings" - "time" - - "claraverse/internal/crypto" - "claraverse/internal/database" - "claraverse/internal/models" - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/bson/primitive" - "go.mongodb.org/mongo-driver/mongo" - "go.mongodb.org/mongo-driver/mongo/options" -) - -// MemoryExtractionService handles extraction of memories from conversations using LLMs -type MemoryExtractionService struct { - mongodb *database.MongoDB - jobCollection *mongo.Collection - engagementCollection *mongo.Collection - encryptionService *crypto.EncryptionService - providerService *ProviderService - memoryStorageService *MemoryStorageService - chatService *ChatService - modelPool *MemoryModelPool // Dynamic model pool with round-robin and failover -} - -// Rate limiting constants for extraction to prevent abuse -const ( - MaxExtractionsPerHour = 20 // Maximum extractions per user per hour - MaxPendingJobsPerUser = 50 // Maximum pending jobs per user -) - -// Memory extraction system prompt -const MemoryExtractionSystemPrompt = `You are a memory extraction system for Clara AI. Analyze this conversation and extract important information to remember about the user. - -WHAT TO EXTRACT: -1. **Personal Information**: Name, location, occupation, family, age, background -2. **Preferences**: Likes, dislikes, communication style, how they want to be addressed -3. **Important Context**: Ongoing projects, goals, constraints, responsibilities -4. **Facts**: Skills, experiences, knowledge areas, technical expertise -5. **Instructions**: Specific guidelines the user wants you to follow (e.g., "always use TypeScript", "keep responses brief") - -RULES: -- Be concise (1-2 sentences per memory) -- Only extract FACTUAL information explicitly stated by the user -- Ignore small talk and pleasantries -- Avoid redundant or obvious information -- Each memory should be atomic (one piece of information) -- Categorize each memory correctly -- Add relevant tags for searchability -- **CRITICAL**: DO NOT extract information that is already captured in EXISTING MEMORIES (provided below) -- Only extract NEW information not present in existing memories -- If conversation contains no new memorable information, return empty array - -CATEGORIES: -- "personal_info": Name, location, occupation, family, age -- "preferences": Likes, dislikes, style, communication preferences -- "context": Ongoing projects, goals, responsibilities -- "fact": Skills, knowledge, experiences -- "instruction": Guidelines to follow - -Return JSON with array of memories.` - -// memoryExtractionSchema defines structured output for memory extraction -var memoryExtractionSchema = map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "memories": map[string]interface{}{ - "type": "array", - "items": map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "content": map[string]interface{}{ - "type": "string", - "description": "The memory content (concise, factual)", - }, - "category": map[string]interface{}{ - "type": "string", - "enum": []string{"personal_info", "preferences", "context", "fact", "instruction"}, - "description": "Memory category", - }, - "tags": map[string]interface{}{ - "type": "array", - "items": map[string]interface{}{ - "type": "string", - }, - "description": "Relevant tags for this memory", - }, - }, - "required": []string{"content", "category", "tags"}, - "additionalProperties": false, - }, - }, - }, - "required": []string{"memories"}, - "additionalProperties": false, -} - -// NewMemoryExtractionService creates a new memory extraction service -func NewMemoryExtractionService( - mongodb *database.MongoDB, - encryptionService *crypto.EncryptionService, - providerService *ProviderService, - memoryStorageService *MemoryStorageService, - chatService *ChatService, - modelPool *MemoryModelPool, -) *MemoryExtractionService { - return &MemoryExtractionService{ - mongodb: mongodb, - jobCollection: mongodb.Collection(database.CollectionMemoryExtractionJobs), - engagementCollection: mongodb.Collection(database.CollectionConversationEngagement), - encryptionService: encryptionService, - providerService: providerService, - memoryStorageService: memoryStorageService, - chatService: chatService, - modelPool: modelPool, - } -} - -// EnqueueExtraction creates a new extraction job (non-blocking) -// SECURITY: Includes rate limiting to prevent abuse and DoS attacks -func (s *MemoryExtractionService) EnqueueExtraction( - ctx context.Context, - userID string, - conversationID string, - messages []map[string]interface{}, -) error { - if userID == "" || conversationID == "" { - return fmt.Errorf("user ID and conversation ID are required") - } - - // SECURITY: Check pending jobs limit to prevent queue flooding - pendingCount, err := s.jobCollection.CountDocuments(ctx, bson.M{ - "userId": userID, - "status": models.JobStatusPending, - }) - if err != nil { - log.Printf("⚠️ [MEMORY-EXTRACTION] Failed to count pending jobs: %v", err) - } else if pendingCount >= MaxPendingJobsPerUser { - log.Printf("⚠️ [MEMORY-EXTRACTION] User %s has %d pending jobs (max: %d), skipping", userID, pendingCount, MaxPendingJobsPerUser) - return fmt.Errorf("too many pending extraction jobs (%d), please wait", pendingCount) - } - - // SECURITY: Check hourly extraction rate (last hour completed + pending) - oneHourAgo := time.Now().Add(-1 * time.Hour) - recentCount, err := s.jobCollection.CountDocuments(ctx, bson.M{ - "userId": userID, - "$or": []bson.M{ - {"status": models.JobStatusPending}, - {"status": models.JobStatusProcessing}, - { - "status": models.JobStatusCompleted, - "processedAt": bson.M{"$gte": oneHourAgo}, - }, - }, - }) - if err != nil { - log.Printf("⚠️ [MEMORY-EXTRACTION] Failed to count recent jobs: %v", err) - } else if recentCount >= MaxExtractionsPerHour { - log.Printf("⚠️ [MEMORY-EXTRACTION] User %s exceeded hourly extraction limit (%d/%d)", userID, recentCount, MaxExtractionsPerHour) - return fmt.Errorf("extraction rate limit exceeded (%d extractions in last hour), please wait", recentCount) - } - - // Encrypt messages - messagesJSON, err := json.Marshal(messages) - if err != nil { - return fmt.Errorf("failed to marshal messages: %w", err) - } - - encryptedMessages, err := s.encryptionService.Encrypt(userID, messagesJSON) - if err != nil { - return fmt.Errorf("failed to encrypt messages: %w", err) - } - - // Create job - job := &models.MemoryExtractionJob{ - ID: primitive.NewObjectID(), - UserID: userID, - ConversationID: conversationID, - MessageCount: len(messages), - EncryptedMessages: encryptedMessages, - Status: models.JobStatusPending, - AttemptCount: 0, - CreatedAt: time.Now(), - } - - // Insert job - _, err = s.jobCollection.InsertOne(ctx, job) - if err != nil { - return fmt.Errorf("failed to insert extraction job: %w", err) - } - - log.Printf("📥 [MEMORY-EXTRACTION] Enqueued job for conversation %s (%d messages)", conversationID, len(messages)) - return nil -} - -// ProcessPendingJobs processes all pending extraction jobs (background worker) -func (s *MemoryExtractionService) ProcessPendingJobs(ctx context.Context) error { - // Find pending jobs - filter := bson.M{"status": models.JobStatusPending} - cursor, err := s.jobCollection.Find(ctx, filter) - if err != nil { - return fmt.Errorf("failed to find pending jobs: %w", err) - } - defer cursor.Close(ctx) - - var jobs []models.MemoryExtractionJob - if err := cursor.All(ctx, &jobs); err != nil { - return fmt.Errorf("failed to decode jobs: %w", err) - } - - if len(jobs) == 0 { - return nil // No pending jobs - } - - log.Printf("⚙️ [MEMORY-EXTRACTION] Processing %d pending jobs", len(jobs)) - - // Process each job - for _, job := range jobs { - if err := s.processJob(ctx, &job); err != nil { - log.Printf("⚠️ [MEMORY-EXTRACTION] Job %s failed: %v", job.ID.Hex(), err) - s.markJobFailed(ctx, job.ID, err.Error()) - } - } - - return nil -} - -// processJob processes a single extraction job -func (s *MemoryExtractionService) processJob(ctx context.Context, job *models.MemoryExtractionJob) error { - // Mark as processing - s.updateJobStatus(ctx, job.ID, models.JobStatusProcessing) - - // Decrypt messages - messagesBytes, err := s.encryptionService.Decrypt(job.UserID, job.EncryptedMessages) - if err != nil { - return fmt.Errorf("failed to decrypt messages: %w", err) - } - - var messages []map[string]interface{} - if err := json.Unmarshal(messagesBytes, &messages); err != nil { - return fmt.Errorf("failed to unmarshal messages: %w", err) - } - - log.Printf("🔍 [MEMORY-EXTRACTION] Processing job %s (%d messages)", job.ID.Hex(), len(messages)) - - // Calculate conversation engagement - engagement := s.calculateEngagement(messages) - - // Store engagement in database - if err := s.storeEngagement(ctx, job.UserID, job.ConversationID, messages, engagement); err != nil { - log.Printf("⚠️ [MEMORY-EXTRACTION] Failed to store engagement: %v", err) - } - - // Fetch existing memories to avoid duplicates - existingMemories, _, err := s.memoryStorageService.ListMemories( - ctx, - job.UserID, - "", // category (empty = all categories) - nil, // tags (nil = all tags) - false, // includeArchived (false = only active) - 1, // page - 100, // pageSize (get recent 100 memories for context) - ) - if err != nil { - log.Printf("⚠️ [MEMORY-EXTRACTION] Failed to fetch existing memories: %v, continuing without context", err) - existingMemories = []models.DecryptedMemory{} // Continue with empty list - } - - log.Printf("📚 [MEMORY-EXTRACTION] Found %d existing memories to avoid duplicates", len(existingMemories)) - - // Extract memories via LLM (with existing memories for context) - extractedMemories, err := s.extractMemories(ctx, job.UserID, messages, existingMemories) - if err != nil { - return fmt.Errorf("failed to extract memories: %w", err) - } - - log.Printf("🧠 [MEMORY-EXTRACTION] Extracted %d memories", len(extractedMemories.Memories)) - - // Store each memory - for _, mem := range extractedMemories.Memories { - _, err := s.memoryStorageService.CreateMemory( - ctx, - job.UserID, - mem.Content, - mem.Category, - mem.Tags, - engagement, - job.ConversationID, - ) - if err != nil { - log.Printf("⚠️ [MEMORY-EXTRACTION] Failed to store memory: %v", err) - } - } - - // Mark job as completed - s.markJobCompleted(ctx, job.ID) - - log.Printf("✅ [MEMORY-EXTRACTION] Job %s completed successfully", job.ID.Hex()) - return nil -} - -// extractMemories calls LLM to extract memories from conversation with automatic failover -func (s *MemoryExtractionService) extractMemories( - ctx context.Context, - userID string, - messages []map[string]interface{}, - existingMemories []models.DecryptedMemory, -) (*models.ExtractedMemoryFromLLM, error) { - - // Check if user has a custom extractor model preference - userPreferredModel, err := s.getExtractorModelForUser(ctx, userID) - var extractorModelID string - - if err == nil && userPreferredModel != "" { - // User has a preference, use it - extractorModelID = userPreferredModel - log.Printf("👤 [MEMORY-EXTRACTION] Using user-preferred model: %s", extractorModelID) - } else { - // No user preference, get from model pool - extractorModelID, err = s.modelPool.GetNextExtractor() - if err != nil { - return nil, fmt.Errorf("no extractor models available: %w", err) - } - } - - // Try extraction with automatic failover (max 3 attempts) - maxAttempts := 3 - var lastError error - - for attempt := 1; attempt <= maxAttempts; attempt++ { - result, err := s.tryExtraction(ctx, userID, extractorModelID, messages, existingMemories) - - if err == nil { - // Success! - s.modelPool.MarkSuccess(extractorModelID) - return result, nil - } - - // Extraction failed - lastError = err - s.modelPool.MarkFailure(extractorModelID) - log.Printf("⚠️ [MEMORY-EXTRACTION] Attempt %d/%d failed with model %s: %v", - attempt, maxAttempts, extractorModelID, err) - - // If not last attempt, get next model from pool - if attempt < maxAttempts { - extractorModelID, err = s.modelPool.GetNextExtractor() - if err != nil { - return nil, fmt.Errorf("no more extractors available after %d attempts: %w", attempt, err) - } - log.Printf("🔄 [MEMORY-EXTRACTION] Retrying with next model: %s", extractorModelID) - } - } - - return nil, fmt.Errorf("extraction failed after %d attempts, last error: %w", maxAttempts, lastError) -} - -// tryExtraction attempts extraction with a specific model (internal helper) -func (s *MemoryExtractionService) tryExtraction( - ctx context.Context, - userID string, - extractorModelID string, - messages []map[string]interface{}, - existingMemories []models.DecryptedMemory, -) (*models.ExtractedMemoryFromLLM, error) { - - // Get provider and model - provider, actualModel, err := s.getProviderAndModel(extractorModelID) - if err != nil { - return nil, fmt.Errorf("failed to get provider for extractor: %w", err) - } - - log.Printf("🤖 [MEMORY-EXTRACTION] Using model: %s (%s)", extractorModelID, actualModel) - - // Build conversation transcript - conversationTranscript := s.buildConversationTranscript(messages) - - // Build existing memories context (decrypted) - existingMemoriesContext := s.buildExistingMemoriesContext(ctx, userID, existingMemories) - - // Build user prompt with existing memories - userPrompt := fmt.Sprintf(`EXISTING MEMORIES: -%s - -CONVERSATION: -%s - -Analyze this conversation and extract ONLY NEW memories that are NOT already captured in the existing memories above. Return JSON with array of memories. If no new information, return empty array.`, existingMemoriesContext, conversationTranscript) - - // Build messages - llmMessages := []map[string]interface{}{ - { - "role": "system", - "content": MemoryExtractionSystemPrompt, - }, - { - "role": "user", - "content": userPrompt, - }, - } - - // Build request with structured output - requestBody := map[string]interface{}{ - "model": actualModel, - "messages": llmMessages, - "stream": false, - "temperature": 0.3, // Low temp for consistency - "response_format": map[string]interface{}{ - "type": "json_schema", - "json_schema": map[string]interface{}{ - "name": "memory_extraction", - "strict": true, - "schema": memoryExtractionSchema, - }, - }, - } - - reqBody, err := json.Marshal(requestBody) - if err != nil { - return nil, fmt.Errorf("failed to marshal request: %w", err) - } - - // Create HTTP request with timeout - httpReq, err := http.NewRequestWithContext(ctx, "POST", provider.BaseURL+"/chat/completions", bytes.NewBuffer(reqBody)) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - httpReq.Header.Set("Content-Type", "application/json") - httpReq.Header.Set("Authorization", "Bearer "+provider.APIKey) - - // Send request with 60s timeout - client := &http.Client{Timeout: 60 * time.Second} - resp, err := client.Do(httpReq) - if err != nil { - return nil, fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - body, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - if resp.StatusCode != http.StatusOK { - log.Printf("⚠️ [MEMORY-EXTRACTION] API error: %s", string(body)) - return nil, fmt.Errorf("API error (status %d): %s", resp.StatusCode, string(body)) - } - - // Parse response - var apiResponse struct { - Choices []struct { - Message struct { - Content string `json:"content"` - } `json:"message"` - } `json:"choices"` - } - - if err := json.Unmarshal(body, &apiResponse); err != nil { - return nil, fmt.Errorf("failed to parse API response: %w", err) - } - - if len(apiResponse.Choices) == 0 { - return nil, fmt.Errorf("no response from extractor model") - } - - // Parse the extraction result - var result models.ExtractedMemoryFromLLM - content := apiResponse.Choices[0].Message.Content - - if err := json.Unmarshal([]byte(content), &result); err != nil { - // SECURITY: Don't log decrypted content - only log length - log.Printf("⚠️ [MEMORY-EXTRACTION] Failed to parse extraction: %v (response length: %d bytes)", err, len(content)) - return nil, fmt.Errorf("failed to parse extraction: %w", err) - } - - return &result, nil -} - -// calculateEngagement calculates conversation engagement score -func (s *MemoryExtractionService) calculateEngagement(messages []map[string]interface{}) float64 { - if len(messages) == 0 { - return 0.0 - } - - userMessageCount := 0 - totalResponseLength := 0 - - for _, msg := range messages { - role, _ := msg["role"].(string) - content, _ := msg["content"].(string) - - if role == "user" { - userMessageCount++ - totalResponseLength += len(content) - } - } - - if userMessageCount == 0 { - return 0.0 - } - - // Turn ratio (how much user participated) - turnRatio := float64(userMessageCount) / float64(len(messages)) - - // Length score (longer responses = more engaged) - avgUserLength := totalResponseLength / userMessageCount - lengthScore := float64(avgUserLength) / 200.0 - if lengthScore > 1.0 { - lengthScore = 1.0 - } - - // Recency bonus (recent conversations get boost) - recencyBonus := 1.0 // Assume all conversations being extracted are recent - - // Weighted engagement score - engagement := (0.5 * turnRatio) + (0.3 * lengthScore) + (0.2 * recencyBonus) - - log.Printf("📊 [MEMORY-EXTRACTION] Engagement: %.2f (turn: %.2f, length: %.2f)", engagement, turnRatio, lengthScore) - - return engagement -} - -// storeEngagement stores conversation engagement in database -func (s *MemoryExtractionService) storeEngagement( - ctx context.Context, - userID string, - conversationID string, - messages []map[string]interface{}, - engagementScore float64, -) error { - - userMessageCount := 0 - totalResponseLength := 0 - - for _, msg := range messages { - role, _ := msg["role"].(string) - content, _ := msg["content"].(string) - - if role == "user" { - userMessageCount++ - totalResponseLength += len(content) - } - } - - avgResponseLength := 0 - if userMessageCount > 0 { - avgResponseLength = totalResponseLength / userMessageCount - } - - engagement := &models.ConversationEngagement{ - ID: primitive.NewObjectID(), - UserID: userID, - ConversationID: conversationID, - MessageCount: len(messages), - UserMessageCount: userMessageCount, - AvgResponseLength: avgResponseLength, - EngagementScore: engagementScore, - CreatedAt: time.Now(), - UpdatedAt: time.Now(), - } - - // Upsert (update if exists, insert if not) - filter := bson.M{"userId": userID, "conversationId": conversationID} - update := bson.M{ - "$set": bson.M{ - "messageCount": engagement.MessageCount, - "userMessageCount": engagement.UserMessageCount, - "avgResponseLength": engagement.AvgResponseLength, - "engagementScore": engagement.EngagementScore, - "updatedAt": engagement.UpdatedAt, - }, - "$setOnInsert": bson.M{ - "_id": engagement.ID, - "createdAt": engagement.CreatedAt, - }, - } - - opts := options.Update().SetUpsert(true) - _, err := s.engagementCollection.UpdateOne(ctx, filter, update, opts) - - return err -} - -// buildConversationTranscript builds a human-readable transcript -func (s *MemoryExtractionService) buildConversationTranscript(messages []map[string]interface{}) string { - var builder strings.Builder - - for _, msg := range messages { - role, _ := msg["role"].(string) - content, _ := msg["content"].(string) - - // Skip system messages - if role == "system" { - continue - } - - // Format message - if role == "user" { - builder.WriteString(fmt.Sprintf("USER: %s\n\n", content)) - } else if role == "assistant" { - builder.WriteString(fmt.Sprintf("ASSISTANT: %s\n\n", content)) - } - } - - return builder.String() -} - -// buildExistingMemoriesContext formats existing memories for LLM context -func (s *MemoryExtractionService) buildExistingMemoriesContext( - ctx context.Context, - userID string, - memories []models.DecryptedMemory, -) string { - if len(memories) == 0 { - return "(No existing memories - this is the first extraction)" - } - - var builder strings.Builder - builder.WriteString(fmt.Sprintf("(%d existing memories):\n\n", len(memories))) - - for i, mem := range memories { - // Format: [category] content (tags: tag1, tag2) - tags := "" - if len(mem.Tags) > 0 { - tags = fmt.Sprintf(" (tags: %s)", strings.Join(mem.Tags, ", ")) - } - - builder.WriteString(fmt.Sprintf("%d. [%s] %s%s\n", i+1, mem.Category, mem.DecryptedContent, tags)) - } - - return builder.String() -} - -// getExtractorModelForUser gets user's preferred extractor model -func (s *MemoryExtractionService) getExtractorModelForUser(ctx context.Context, userID string) (string, error) { - // Query MongoDB for user preferences - usersCollection := s.mongodb.Collection(database.CollectionUsers) - - var user models.User - err := usersCollection.FindOne(ctx, bson.M{"supabaseUserId": userID}).Decode(&user) - if err != nil { - return "", nil // No user found, return empty (will use pool) - } - - // If user has a preference and it's not empty, use it - if user.Preferences.MemoryExtractorModelID != "" { - log.Printf("🎯 [MEMORY-EXTRACTION] Using user-preferred model: %s", user.Preferences.MemoryExtractorModelID) - return user.Preferences.MemoryExtractorModelID, nil - } - - // No preference set - return empty (will use pool) - return "", nil -} - -// getProviderAndModel resolves model ID to provider and actual model name -func (s *MemoryExtractionService) getProviderAndModel(modelID string) (*models.Provider, string, error) { - if modelID == "" { - return nil, "", fmt.Errorf("model ID is required") - } - - // Try to resolve through ChatService (handles aliases) - if s.chatService != nil { - if provider, actualModel, found := s.chatService.ResolveModelAlias(modelID); found { - return provider, actualModel, nil - } - } - - return nil, "", fmt.Errorf("model %s not found in providers", modelID) -} - -// updateJobStatus updates job status -func (s *MemoryExtractionService) updateJobStatus(ctx context.Context, jobID primitive.ObjectID, status string) { - update := bson.M{ - "$set": bson.M{ - "status": status, - }, - } - s.jobCollection.UpdateOne(ctx, bson.M{"_id": jobID}, update) -} - -// markJobCompleted marks job as completed -func (s *MemoryExtractionService) markJobCompleted(ctx context.Context, jobID primitive.ObjectID) { - now := time.Now() - update := bson.M{ - "$set": bson.M{ - "status": models.JobStatusCompleted, - "processedAt": now, - }, - } - s.jobCollection.UpdateOne(ctx, bson.M{"_id": jobID}, update) -} - -// markJobFailed marks job as failed with error message -func (s *MemoryExtractionService) markJobFailed(ctx context.Context, jobID primitive.ObjectID, errorMsg string) { - update := bson.M{ - "$set": bson.M{ - "status": models.JobStatusFailed, - "errorMessage": errorMsg, - }, - "$inc": bson.M{ - "attemptCount": 1, - }, - } - s.jobCollection.UpdateOne(ctx, bson.M{"_id": jobID}, update) -} - -// Helper function for pointer diff --git a/backend/internal/services/memory_integration_test.go b/backend/internal/services/memory_integration_test.go deleted file mode 100644 index bde92ed8..00000000 --- a/backend/internal/services/memory_integration_test.go +++ /dev/null @@ -1,465 +0,0 @@ -package services - -import ( - "testing" - "time" - - "claraverse/internal/models" -) - -// TestMemoryExtractionSchema tests the extraction schema structure -func TestMemoryExtractionSchema(t *testing.T) { - schema := memoryExtractionSchema - - // Verify schema structure - if schema["type"] != "object" { - t.Error("Schema should be object type") - } - - properties, ok := schema["properties"].(map[string]interface{}) - if !ok { - t.Fatal("Schema should have properties") - } - - // Verify required fields - requiredFields := []string{"memories"} - required, ok := schema["required"].([]string) - if !ok { - t.Fatal("Schema should have required fields") - } - - for _, field := range requiredFields { - found := false - for _, req := range required { - if req == field { - found = true - break - } - } - if !found { - t.Errorf("Required field %s not found in schema", field) - } - } - - // Verify memories array structure - memories, ok := properties["memories"].(map[string]interface{}) - if !ok { - t.Fatal("memories field should exist") - } - - if memories["type"] != "array" { - t.Error("memories should be array type") - } - - items, ok := memories["items"].(map[string]interface{}) - if !ok { - t.Fatal("memories should have items definition") - } - - itemProps, ok := items["properties"].(map[string]interface{}) - if !ok { - t.Fatal("memory items should have properties") - } - - // Verify memory item fields - requiredMemoryFields := []string{"content", "category", "tags"} - for _, field := range requiredMemoryFields { - if _, exists := itemProps[field]; !exists { - t.Errorf("Memory item should have field: %s", field) - } - } -} - -// TestMemorySelectionSchema tests the selection schema structure -func TestMemorySelectionSchema(t *testing.T) { - schema := memorySelectionSchema - - // Verify schema structure - if schema["type"] != "object" { - t.Error("Schema should be object type") - } - - properties, ok := schema["properties"].(map[string]interface{}) - if !ok { - t.Fatal("Schema should have properties") - } - - // Verify required fields - requiredFields := []string{"selected_memory_ids", "reasoning"} - required, ok := schema["required"].([]string) - if !ok { - t.Fatal("Schema should have required fields") - } - - for _, field := range requiredFields { - found := false - for _, req := range required { - if req == field { - found = true - break - } - } - if !found { - t.Errorf("Required field %s not found in schema", field) - } - } - - // Verify selected_memory_ids array structure - selectedIDs, ok := properties["selected_memory_ids"].(map[string]interface{}) - if !ok { - t.Fatal("selected_memory_ids field should exist") - } - - if selectedIDs["type"] != "array" { - t.Error("selected_memory_ids should be array type") - } -} - -// TestMemoryExtractionSystemPrompt tests the extraction prompt content -func TestMemoryExtractionSystemPrompt(t *testing.T) { - prompt := MemoryExtractionSystemPrompt - - // Verify prompt contains key instructions - requiredPhrases := []string{ - "memory extraction system", - "Personal Information", - "Preferences", - "Important Context", - "Facts", - "Instructions", - } - - for _, phrase := range requiredPhrases { - if !contains(prompt, phrase) { - t.Errorf("Extraction prompt should contain: %q", phrase) - } - } - - // Verify prompt is not empty - if len(prompt) < 100 { - t.Error("Extraction prompt seems too short") - } - - t.Logf("Extraction prompt length: %d characters", len(prompt)) -} - -// TestMemorySelectionSystemPrompt tests the selection prompt content -func TestMemorySelectionSystemPrompt(t *testing.T) { - prompt := MemorySelectionSystemPrompt - - // Verify prompt contains key instructions - requiredPhrases := []string{ - "memory selection system", - "MOST RELEVANT", - "Direct Relevance", - "Contextual Information", - "User Preferences", - "Instructions", - } - - for _, phrase := range requiredPhrases { - if !contains(prompt, phrase) { - t.Errorf("Selection prompt should contain: %q", phrase) - } - } - - // Verify prompt is not empty - if len(prompt) < 100 { - t.Error("Selection prompt seems too short") - } - - t.Logf("Selection prompt length: %d characters", len(prompt)) -} - -// TestConversationEngagementCalculation tests engagement score logic -func TestConversationEngagementCalculation(t *testing.T) { - tests := []struct { - name string - messageCount int - userMessageCount int - avgResponseLength int - withinWeek bool - minExpected float64 - maxExpected float64 - }{ - { - name: "High engagement (balanced turn-taking, long responses, recent)", - messageCount: 20, - userMessageCount: 10, - avgResponseLength: 250, - withinWeek: true, - minExpected: 0.70, - maxExpected: 0.80, - }, - { - name: "Medium engagement (moderate activity)", - messageCount: 10, - userMessageCount: 4, - avgResponseLength: 150, - withinWeek: true, - minExpected: 0.40, - maxExpected: 0.70, - }, - { - name: "Low engagement (few messages, short responses, old)", - messageCount: 4, - userMessageCount: 1, - avgResponseLength: 50, - withinWeek: false, - minExpected: 0.00, - maxExpected: 0.30, - }, - { - name: "User dominated (high user ratio)", - messageCount: 10, - userMessageCount: 8, - avgResponseLength: 200, - withinWeek: true, - minExpected: 0.70, - maxExpected: 1.00, - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - // Calculate engagement score components - turnRatio := float64(tt.userMessageCount) / float64(tt.messageCount) - lengthScore := minFloat(1.0, float64(tt.avgResponseLength)/200.0) - recencyBonus := 0.0 - if tt.withinWeek { - recencyBonus = 1.0 - } - - // Engagement formula: (0.5 × TurnRatio) + (0.3 × LengthScore) + (0.2 × RecencyBonus) - engagementScore := (0.5 * turnRatio) + (0.3 * lengthScore) + (0.2 * recencyBonus) - - if engagementScore < tt.minExpected || engagementScore > tt.maxExpected { - t.Errorf("Expected engagement between %.2f and %.2f, got %.2f\n"+ - " Turn Ratio: %.2f, Length Score: %.2f, Recency: %.2f", - tt.minExpected, tt.maxExpected, engagementScore, - turnRatio, lengthScore, recencyBonus) - } - - t.Logf("Engagement Score: %.2f (turn: %.2f, length: %.2f, recency: %.2f)", - engagementScore, turnRatio, lengthScore, recencyBonus) - }) - } -} - -// TestDecayConfigDefaults tests the default decay configuration -func TestDecayConfigDefaults(t *testing.T) { - config := DefaultDecayConfig() - - // Test default weights - if config.RecencyWeight != 0.4 { - t.Errorf("Expected RecencyWeight 0.4, got %.2f", config.RecencyWeight) - } - if config.FrequencyWeight != 0.3 { - t.Errorf("Expected FrequencyWeight 0.3, got %.2f", config.FrequencyWeight) - } - if config.EngagementWeight != 0.3 { - t.Errorf("Expected EngagementWeight 0.3, got %.2f", config.EngagementWeight) - } - - // Test weights sum to 1.0 - totalWeight := config.RecencyWeight + config.FrequencyWeight + config.EngagementWeight - if totalWeight != 1.0 { - t.Errorf("Weights should sum to 1.0, got %.2f", totalWeight) - } - - // Test other defaults - if config.RecencyDecayRate != 0.05 { - t.Errorf("Expected RecencyDecayRate 0.05, got %.2f", config.RecencyDecayRate) - } - if config.FrequencyMax != 20 { - t.Errorf("Expected FrequencyMax 20, got %d", config.FrequencyMax) - } - if config.ArchiveThreshold != 0.15 { - t.Errorf("Expected ArchiveThreshold 0.15, got %.2f", config.ArchiveThreshold) - } -} - -// TestMemoryModelDefaults tests default values in Memory model -func TestMemoryModelDefaults(t *testing.T) { - // This test documents expected default states - defaultAccessCount := int64(0) - defaultIsArchived := false - defaultVersion := int64(1) - - if defaultAccessCount != 0 { - t.Errorf("New memories should have 0 access count") - } - if defaultIsArchived != false { - t.Errorf("New memories should not be archived") - } - if defaultVersion != 1 { - t.Errorf("New memories should start at version 1") - } -} - -// TestExtractedMemoryStructure tests the structure of extracted memories -func TestExtractedMemoryStructure(t *testing.T) { - // Test valid categories - validCategories := map[string]bool{ - "personal_info": true, - "preferences": true, - "context": true, - "fact": true, - "instruction": true, - } - - // Simulate an extracted memory (correct structure) - testResult := models.ExtractedMemoryFromLLM{ - Memories: []struct { - Content string `json:"content"` - Category string `json:"category"` - Tags []string `json:"tags"` - }{ - { - Content: "User prefers dark mode", - Category: "preferences", - Tags: []string{"ui", "theme", "preferences"}, - }, - }, - } - - // Validate we have memories - if len(testResult.Memories) == 0 { - t.Error("Should have at least one memory") - } - - // Validate first memory - testMemory := testResult.Memories[0] - - // Validate category - if !validCategories[testMemory.Category] { - t.Errorf("Invalid category: %s", testMemory.Category) - } - - // Validate content not empty - if testMemory.Content == "" { - t.Error("Memory content should not be empty") - } - - // Validate tags - if len(testMemory.Tags) == 0 { - t.Error("Memory should have at least one tag") - } -} - -// TestSelectedMemoriesStructure tests the structure of selected memories -func TestSelectedMemoriesStructure(t *testing.T) { - // Simulate a selection result - selection := models.SelectedMemoriesFromLLM{ - SelectedMemoryIDs: []string{"id1", "id2", "id3"}, - Reasoning: "These memories are relevant because...", - } - - // Validate IDs - if len(selection.SelectedMemoryIDs) == 0 { - t.Error("Selection should have memory IDs") - } - - // Validate reasoning - if selection.Reasoning == "" { - t.Error("Selection should have reasoning") - } - - // Max 5 memories rule - maxMemories := 5 - if len(selection.SelectedMemoryIDs) > maxMemories { - t.Errorf("Should not select more than %d memories, got %d", maxMemories, len(selection.SelectedMemoryIDs)) - } -} - -// TestMemoryLifecycleStates tests valid state transitions -func TestMemoryLifecycleStates(t *testing.T) { - now := time.Now() - - // State 1: Newly created - memory := models.Memory{ - AccessCount: 0, - LastAccessedAt: nil, - IsArchived: false, - ArchivedAt: nil, - CreatedAt: now, - UpdatedAt: now, - Version: 1, - } - - if memory.IsArchived { - t.Error("New memory should not be archived") - } - if memory.AccessCount != 0 { - t.Error("New memory should have 0 access count") - } - - // State 2: Accessed - accessTime := now.Add(1 * time.Hour) - memory.AccessCount = 1 - memory.LastAccessedAt = &accessTime - - if memory.LastAccessedAt == nil { - t.Error("Accessed memory should have LastAccessedAt") - } - - // State 3: Archived - archiveTime := now.Add(90 * 24 * time.Hour) - memory.IsArchived = true - memory.ArchivedAt = &archiveTime - - if !memory.IsArchived { - t.Error("Archived memory should have IsArchived=true") - } - if memory.ArchivedAt == nil { - t.Error("Archived memory should have ArchivedAt timestamp") - } -} - -// TestMemorySystemPromptInjection tests the memory context formatting -func TestMemorySystemPromptInjection(t *testing.T) { - // Simulate building memory context - memories := []models.DecryptedMemory{ - {DecryptedContent: "User prefers dark mode"}, - {DecryptedContent: "User name is Clara"}, - {DecryptedContent: "User timezone is America/New_York"}, - } - - // Build context string (simplified version of buildMemoryContext) - var context string - context = "\n\n## Relevant Context from Previous Conversations\n\n" - for i, mem := range memories { - context += string(rune('1'+i)) + ". " + mem.DecryptedContent + "\n" - } - - // Verify format - if !contains(context, "## Relevant Context") { - t.Error("Context should have header") - } - - for _, mem := range memories { - if !contains(context, mem.DecryptedContent) { - t.Errorf("Context should include memory: %s", mem.DecryptedContent) - } - } - - // Verify numbered list - if !contains(context, "1. ") { - t.Error("Context should use numbered list") - } - - t.Logf("Generated context:\n%s", context) -} - -// Helper function to check if string contains substring -func contains(s, substr string) bool { - return len(s) >= len(substr) && (s == substr || len(s) > len(substr) && (s[:len(substr)] == substr || contains(s[1:], substr))) -} - -// Helper function for min (renamed to avoid conflict) -func minFloat(a, b float64) float64 { - if a < b { - return a - } - return b -} diff --git a/backend/internal/services/memory_model_pool.go b/backend/internal/services/memory_model_pool.go deleted file mode 100644 index dcca63ff..00000000 --- a/backend/internal/services/memory_model_pool.go +++ /dev/null @@ -1,430 +0,0 @@ -package services - -import ( - "database/sql" - "fmt" - "log" - "sync" - "time" - - "claraverse/internal/config" - "claraverse/internal/models" -) - -// MemoryModelPool manages multiple models for memory operations with health tracking and failover -type MemoryModelPool struct { - extractorModels []ModelCandidate - selectorModels []ModelCandidate - extractorIndex int - selectorIndex int - healthTracker map[string]*ModelHealth - mu sync.Mutex - chatService *ChatService - db *sql.DB // Database connection for querying model_aliases -} - -// ModelCandidate represents a model eligible for memory operations -type ModelCandidate struct { - ModelID string - ProviderName string - SpeedMs int - DisplayName string -} - -// ModelHealth tracks model health and failures -type ModelHealth struct { - FailureCount int - SuccessCount int - LastFailure time.Time - LastSuccess time.Time - IsHealthy bool - ConsecutiveFails int -} - -const ( - // Health thresholds - MaxConsecutiveFailures = 3 - HealthCheckCooldown = 5 * time.Minute - MinSuccessesToRecover = 2 -) - -// NewMemoryModelPool creates a new model pool by discovering eligible models from providers -func NewMemoryModelPool(chatService *ChatService, db *sql.DB) (*MemoryModelPool, error) { - pool := &MemoryModelPool{ - chatService: chatService, - db: db, - healthTracker: make(map[string]*ModelHealth), - } - - // Discover models from ChatService - if err := pool.discoverModels(); err != nil { - log.Printf("⚠️ [MODEL-POOL] Failed to discover models: %v", err) - log.Printf("⚠️ [MODEL-POOL] Memory services will be disabled until models with memory flags are added") - } - - if len(pool.extractorModels) == 0 { - log.Printf("⚠️ [MODEL-POOL] No extractor models found - memory extraction disabled") - } - - if len(pool.selectorModels) == 0 { - log.Printf("⚠️ [MODEL-POOL] No selector models found - memory selection disabled") - } - - if len(pool.extractorModels) > 0 || len(pool.selectorModels) > 0 { - log.Printf("🎯 [MODEL-POOL] Initialized with %d extractors, %d selectors", - len(pool.extractorModels), len(pool.selectorModels)) - } - - // Return pool even if empty - allows graceful degradation - return pool, nil -} - -// discoverModels scans database for models with memory flags -func (p *MemoryModelPool) discoverModels() error { - // First try loading from database (MySQL-first approach) - dbModels, err := p.discoverFromDatabase() - if err == nil && len(dbModels) > 0 { - log.Printf("✅ [MODEL-POOL] Discovered %d models from database", len(dbModels)) - return nil - } - - // Fallback: Load providers configuration from providers.json - providersConfig, err := config.LoadProviders("providers.json") - if err != nil { - // Gracefully handle missing providers.json (expected in admin UI workflow) - log.Printf("⚠️ [MODEL-POOL] providers.json not found or invalid: %v", err) - log.Printf("ℹ️ [MODEL-POOL] This is normal when starting with empty database") - return nil // Not a fatal error - just means no models configured yet - } - - for _, providerConfig := range providersConfig.Providers { - if !providerConfig.Enabled { - continue - } - - for alias, modelAlias := range providerConfig.ModelAliases { - // Get model configuration map (convert from ModelAlias) - modelConfig := modelAliasToMap(modelAlias) - - // Check if model supports memory extraction - if isExtractor, ok := modelConfig["memory_extractor"].(bool); ok && isExtractor { - candidate := ModelCandidate{ - ModelID: alias, - ProviderName: providerConfig.Name, - DisplayName: getDisplayName(modelConfig), - SpeedMs: getSpeedMs(modelConfig), - } - p.extractorModels = append(p.extractorModels, candidate) - p.healthTracker[alias] = &ModelHealth{IsHealthy: true} - - log.Printf("✅ [MODEL-POOL] Found extractor: %s (%s) - %dms", - alias, providerConfig.Name, candidate.SpeedMs) - } - - // Check if model supports memory selection - if isSelector, ok := modelConfig["memory_selector"].(bool); ok && isSelector { - // Avoid duplicates if model is both extractor and selector - if _, exists := p.healthTracker[alias]; !exists { - p.healthTracker[alias] = &ModelHealth{IsHealthy: true} - } - - candidate := ModelCandidate{ - ModelID: alias, - ProviderName: providerConfig.Name, - DisplayName: getDisplayName(modelConfig), - SpeedMs: getSpeedMs(modelConfig), - } - p.selectorModels = append(p.selectorModels, candidate) - - log.Printf("✅ [MODEL-POOL] Found selector: %s (%s) - %dms", - alias, providerConfig.Name, candidate.SpeedMs) - } - } - } - - // Sort by speed (fastest first) - p.sortModelsBySpeed(p.extractorModels) - p.sortModelsBySpeed(p.selectorModels) - - return nil -} - -// discoverFromDatabase loads memory models from database (model_aliases table) -func (p *MemoryModelPool) discoverFromDatabase() ([]ModelCandidate, error) { - if p.db == nil { - return nil, fmt.Errorf("database connection not available") - } - - // Query for models with memory_extractor or memory_selector flags - rows, err := p.db.Query(` - SELECT - a.alias_name, - pr.name as provider_name, - a.display_name, - COALESCE(a.structured_output_speed_ms, 999999) as speed_ms, - COALESCE(a.memory_extractor, 0) as memory_extractor, - COALESCE(a.memory_selector, 0) as memory_selector - FROM model_aliases a - JOIN providers pr ON a.provider_id = pr.id - WHERE a.memory_extractor = 1 OR a.memory_selector = 1 - `) - - if err != nil { - return nil, fmt.Errorf("failed to query model_aliases: %w", err) - } - defer rows.Close() - - var candidates []ModelCandidate - for rows.Next() { - var aliasName, providerName, displayName string - var speedMs int - var isExtractor, isSelector int - - if err := rows.Scan(&aliasName, &providerName, &displayName, &speedMs, &isExtractor, &isSelector); err != nil { - log.Printf("⚠️ [MODEL-POOL] Failed to scan row: %v", err) - continue - } - - candidate := ModelCandidate{ - ModelID: aliasName, - ProviderName: providerName, - DisplayName: displayName, - SpeedMs: speedMs, - } - - if isExtractor == 1 { - p.extractorModels = append(p.extractorModels, candidate) - p.healthTracker[aliasName] = &ModelHealth{IsHealthy: true} - log.Printf("✅ [MODEL-POOL] Found extractor from DB: %s (%s) - %dms", aliasName, providerName, speedMs) - } - - if isSelector == 1 { - // Avoid duplicates if model is both extractor and selector - if _, exists := p.healthTracker[aliasName]; !exists { - p.healthTracker[aliasName] = &ModelHealth{IsHealthy: true} - } - p.selectorModels = append(p.selectorModels, candidate) - log.Printf("✅ [MODEL-POOL] Found selector from DB: %s (%s) - %dms", aliasName, providerName, speedMs) - } - - candidates = append(candidates, candidate) - } - - // Sort by speed (fastest first) - p.sortModelsBySpeed(p.extractorModels) - p.sortModelsBySpeed(p.selectorModels) - - return candidates, nil -} - -// modelAliasToMap converts ModelAlias struct to map for easier access -func modelAliasToMap(alias models.ModelAlias) map[string]interface{} { - m := make(map[string]interface{}) - - // Set display_name - m["display_name"] = alias.DisplayName - - // Set structured_output_speed_ms if available - if alias.StructuredOutputSpeedMs != nil { - m["structured_output_speed_ms"] = *alias.StructuredOutputSpeedMs - } - - // Set memory flags if available - if alias.MemoryExtractor != nil { - m["memory_extractor"] = *alias.MemoryExtractor - } - if alias.MemorySelector != nil { - m["memory_selector"] = *alias.MemorySelector - } - - return m -} - -// GetNextExtractor returns the next healthy extractor model using round-robin -func (p *MemoryModelPool) GetNextExtractor() (string, error) { - p.mu.Lock() - defer p.mu.Unlock() - - if len(p.extractorModels) == 0 { - return "", fmt.Errorf("no extractor models available") - } - - // Try all models in round-robin fashion - attempts := 0 - maxAttempts := len(p.extractorModels) - - for attempts < maxAttempts { - candidate := p.extractorModels[p.extractorIndex] - p.extractorIndex = (p.extractorIndex + 1) % len(p.extractorModels) - attempts++ - - // Check if model is healthy - health := p.healthTracker[candidate.ModelID] - if health.IsHealthy { - log.Printf("🔄 [MODEL-POOL] Selected extractor: %s (healthy)", candidate.ModelID) - return candidate.ModelID, nil - } - - // Check if enough time has passed since last failure (cooldown) - if time.Since(health.LastFailure) > HealthCheckCooldown { - log.Printf("⚡ [MODEL-POOL] Retrying extractor after cooldown: %s", candidate.ModelID) - health.IsHealthy = true - health.ConsecutiveFails = 0 - return candidate.ModelID, nil - } - - log.Printf("⏭️ [MODEL-POOL] Skipping unhealthy extractor: %s (fails: %d, last: %s ago)", - candidate.ModelID, health.ConsecutiveFails, time.Since(health.LastFailure).Round(time.Second)) - } - - // All models unhealthy - return fastest anyway as last resort - log.Printf("⚠️ [MODEL-POOL] All extractors unhealthy, using fastest: %s", p.extractorModels[0].ModelID) - return p.extractorModels[0].ModelID, nil -} - -// GetNextSelector returns the next healthy selector model using round-robin -func (p *MemoryModelPool) GetNextSelector() (string, error) { - p.mu.Lock() - defer p.mu.Unlock() - - if len(p.selectorModels) == 0 { - return "", fmt.Errorf("no selector models available") - } - - // Try all models in round-robin fashion - attempts := 0 - maxAttempts := len(p.selectorModels) - - for attempts < maxAttempts { - candidate := p.selectorModels[p.selectorIndex] - p.selectorIndex = (p.selectorIndex + 1) % len(p.selectorModels) - attempts++ - - // Check if model is healthy - health := p.healthTracker[candidate.ModelID] - if health.IsHealthy { - log.Printf("🔄 [MODEL-POOL] Selected selector: %s (healthy)", candidate.ModelID) - return candidate.ModelID, nil - } - - // Check if enough time has passed since last failure (cooldown) - if time.Since(health.LastFailure) > HealthCheckCooldown { - log.Printf("⚡ [MODEL-POOL] Retrying selector after cooldown: %s", candidate.ModelID) - health.IsHealthy = true - health.ConsecutiveFails = 0 - return candidate.ModelID, nil - } - - log.Printf("⏭️ [MODEL-POOL] Skipping unhealthy selector: %s (fails: %d, last: %s ago)", - candidate.ModelID, health.ConsecutiveFails, time.Since(health.LastFailure).Round(time.Second)) - } - - // All models unhealthy - return fastest anyway as last resort - log.Printf("⚠️ [MODEL-POOL] All selectors unhealthy, using fastest: %s", p.selectorModels[0].ModelID) - return p.selectorModels[0].ModelID, nil -} - -// MarkSuccess records a successful model call -func (p *MemoryModelPool) MarkSuccess(modelID string) { - p.mu.Lock() - defer p.mu.Unlock() - - health, exists := p.healthTracker[modelID] - if !exists { - return - } - - health.SuccessCount++ - health.LastSuccess = time.Now() - health.ConsecutiveFails = 0 - - // Restore health after consecutive successes - if !health.IsHealthy && health.SuccessCount >= MinSuccessesToRecover { - health.IsHealthy = true - log.Printf("💚 [MODEL-POOL] Model recovered: %s (successes: %d)", modelID, health.SuccessCount) - } -} - -// MarkFailure records a failed model call -func (p *MemoryModelPool) MarkFailure(modelID string) { - p.mu.Lock() - defer p.mu.Unlock() - - health, exists := p.healthTracker[modelID] - if !exists { - return - } - - health.FailureCount++ - health.ConsecutiveFails++ - health.LastFailure = time.Now() - - // Mark unhealthy after consecutive failures - if health.ConsecutiveFails >= MaxConsecutiveFailures { - health.IsHealthy = false - log.Printf("💔 [MODEL-POOL] Model marked unhealthy: %s (consecutive fails: %d, total fails: %d)", - modelID, health.ConsecutiveFails, health.FailureCount) - } else { - log.Printf("⚠️ [MODEL-POOL] Model failure: %s (consecutive: %d/%d)", - modelID, health.ConsecutiveFails, MaxConsecutiveFailures) - } -} - -// GetStats returns current pool statistics -func (p *MemoryModelPool) GetStats() map[string]interface{} { - p.mu.Lock() - defer p.mu.Unlock() - - healthyExtractors := 0 - healthySelectors := 0 - - for _, model := range p.extractorModels { - if p.healthTracker[model.ModelID].IsHealthy { - healthyExtractors++ - } - } - - for _, model := range p.selectorModels { - if p.healthTracker[model.ModelID].IsHealthy { - healthySelectors++ - } - } - - return map[string]interface{}{ - "total_extractors": len(p.extractorModels), - "healthy_extractors": healthyExtractors, - "total_selectors": len(p.selectorModels), - "healthy_selectors": healthySelectors, - } -} - -// Helper functions - -func getDisplayName(modelConfig map[string]interface{}) string { - if name, ok := modelConfig["display_name"].(string); ok { - return name - } - return "Unknown" -} - -func getSpeedMs(modelConfig map[string]interface{}) int { - if speed, ok := modelConfig["structured_output_speed_ms"].(float64); ok { - return int(speed) - } - if speed, ok := modelConfig["structured_output_speed_ms"].(int); ok { - return speed - } - return 999999 // Default to slow if not specified -} - -func (p *MemoryModelPool) sortModelsBySpeed(models []ModelCandidate) { - // Simple bubble sort (fine for small arrays) - n := len(models) - for i := 0; i < n-1; i++ { - for j := 0; j < n-i-1; j++ { - if models[j].SpeedMs > models[j+1].SpeedMs { - models[j], models[j+1] = models[j+1], models[j] - } - } - } -} diff --git a/backend/internal/services/memory_selection_service.go b/backend/internal/services/memory_selection_service.go deleted file mode 100644 index 366732d8..00000000 --- a/backend/internal/services/memory_selection_service.go +++ /dev/null @@ -1,443 +0,0 @@ -package services - -import ( - "bytes" - "context" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "strings" - "time" - - "claraverse/internal/crypto" - "claraverse/internal/database" - "claraverse/internal/models" - "go.mongodb.org/mongo-driver/bson/primitive" -) - -// MemorySelectionService handles selection of relevant memories using LLMs -type MemorySelectionService struct { - mongodb *database.MongoDB - encryptionService *crypto.EncryptionService - providerService *ProviderService - memoryStorageService *MemoryStorageService - chatService *ChatService - modelPool *MemoryModelPool // Dynamic model pool with round-robin and failover -} - -// Memory selection system prompt -const MemorySelectionSystemPrompt = `You are a memory selection system for Clara AI. Given the user's recent conversation and their memory bank, select the MOST RELEVANT memories. - -SELECTION CRITERIA: -1. **Direct Relevance**: Memory directly relates to current conversation topic -2. **Contextual Information**: Memory provides important background context -3. **User Preferences**: Memory contains preferences that affect how to respond -4. **Instructions**: Memory contains guidelines the user wants followed - -RULES: -- Select up to 5 memories maximum (fewer is better if not all are relevant) -- Prioritize memories that prevent asking redundant questions -- Include memories that personalize the response -- Skip memories that are obvious or unrelated -- If no memories are relevant, return empty array - -Return JSON with selected memory IDs and brief reasoning.` - -// memorySelectionSchema defines structured output for memory selection -var memorySelectionSchema = map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "selected_memory_ids": map[string]interface{}{ - "type": "array", - "items": map[string]interface{}{ - "type": "string", - "description": "Memory ID from the provided list", - }, - "description": "IDs of memories relevant to current conversation (max 5)", - }, - "reasoning": map[string]interface{}{ - "type": "string", - "description": "Brief explanation of why these memories are relevant", - }, - }, - "required": []string{"selected_memory_ids", "reasoning"}, - "additionalProperties": false, -} - -// NewMemorySelectionService creates a new memory selection service -func NewMemorySelectionService( - mongodb *database.MongoDB, - encryptionService *crypto.EncryptionService, - providerService *ProviderService, - memoryStorageService *MemoryStorageService, - chatService *ChatService, - modelPool *MemoryModelPool, -) *MemorySelectionService { - return &MemorySelectionService{ - mongodb: mongodb, - encryptionService: encryptionService, - providerService: providerService, - memoryStorageService: memoryStorageService, - chatService: chatService, - modelPool: modelPool, - } -} - -// SelectRelevantMemories selects memories relevant to the current conversation -func (s *MemorySelectionService) SelectRelevantMemories( - ctx context.Context, - userID string, - recentMessages []map[string]interface{}, - maxMemories int, -) ([]models.DecryptedMemory, error) { - - // Get all active memories for user - activeMemories, err := s.memoryStorageService.GetActiveMemories(ctx, userID) - if err != nil { - return nil, fmt.Errorf("failed to get active memories: %w", err) - } - - // If no memories, return empty - if len(activeMemories) == 0 { - log.Printf("📭 [MEMORY-SELECTION] No active memories for user %s", userID) - return []models.DecryptedMemory{}, nil - } - - log.Printf("🔍 [MEMORY-SELECTION] Selecting from %d active memories for user %s", len(activeMemories), userID) - - // If we have fewer memories than max, just return all and update access - if len(activeMemories) <= maxMemories { - log.Printf("📚 [MEMORY-SELECTION] Returning all %d memories (below max %d)", len(activeMemories), maxMemories) - memoryIDs := make([]primitive.ObjectID, len(activeMemories)) - for i, mem := range activeMemories { - memoryIDs[i] = mem.ID - } - s.memoryStorageService.UpdateMemoryAccess(ctx, memoryIDs) - return activeMemories, nil - } - - // Use LLM to select relevant memories - selectedIDs, reasoning, err := s.selectMemoriesWithLLM(ctx, userID, activeMemories, recentMessages, maxMemories) - if err != nil { - log.Printf("⚠️ [MEMORY-SELECTION] LLM selection failed: %v, falling back to top %d by score", err, maxMemories) - // Fallback: return top N by score - selectedMemories := activeMemories - if len(selectedMemories) > maxMemories { - selectedMemories = selectedMemories[:maxMemories] - } - memoryIDs := make([]primitive.ObjectID, len(selectedMemories)) - for i, mem := range selectedMemories { - memoryIDs[i] = mem.ID - } - s.memoryStorageService.UpdateMemoryAccess(ctx, memoryIDs) - return selectedMemories, nil - } - - log.Printf("🎯 [MEMORY-SELECTION] LLM selected %d memories: %s", len(selectedIDs), reasoning) - - // Filter memories by selected IDs - selectedMemories := s.filterMemoriesByIDs(activeMemories, selectedIDs) - - // Update access counts and timestamps - memoryIDs := make([]primitive.ObjectID, len(selectedMemories)) - for i, mem := range selectedMemories { - memoryIDs[i] = mem.ID - } - if len(memoryIDs) > 0 { - s.memoryStorageService.UpdateMemoryAccess(ctx, memoryIDs) - } - - return selectedMemories, nil -} - -// selectMemoriesWithLLM uses LLM to select relevant memories with automatic failover -func (s *MemorySelectionService) selectMemoriesWithLLM( - ctx context.Context, - userID string, - memories []models.DecryptedMemory, - recentMessages []map[string]interface{}, - maxMemories int, -) ([]string, string, error) { - - // Check if user has a custom selector model preference - userPreferredModel, err := s.getSelectorModelForUser(ctx, userID) - var selectorModelID string - - if err == nil && userPreferredModel != "" { - // User has a preference, use it - selectorModelID = userPreferredModel - log.Printf("👤 [MEMORY-SELECTION] Using user-preferred model: %s", selectorModelID) - } else { - // No user preference, get from model pool - selectorModelID, err = s.modelPool.GetNextSelector() - if err != nil { - return nil, "", fmt.Errorf("no selector models available: %w", err) - } - } - - // Try selection with automatic failover (max 3 attempts) - maxAttempts := 3 - var lastError error - - for attempt := 1; attempt <= maxAttempts; attempt++ { - selectedIDs, reasoning, err := s.trySelection(ctx, selectorModelID, memories, recentMessages, maxMemories) - - if err == nil { - // Success! - s.modelPool.MarkSuccess(selectorModelID) - return selectedIDs, reasoning, nil - } - - // Selection failed - lastError = err - s.modelPool.MarkFailure(selectorModelID) - log.Printf("⚠️ [MEMORY-SELECTION] Attempt %d/%d failed with model %s: %v", - attempt, maxAttempts, selectorModelID, err) - - // If not last attempt, get next model from pool - if attempt < maxAttempts { - selectorModelID, err = s.modelPool.GetNextSelector() - if err != nil { - return nil, "", fmt.Errorf("no more selectors available after %d attempts: %w", attempt, err) - } - log.Printf("🔄 [MEMORY-SELECTION] Retrying with next model: %s", selectorModelID) - } - } - - return nil, "", fmt.Errorf("selection failed after %d attempts, last error: %w", maxAttempts, lastError) -} - -// trySelection attempts selection with a specific model (internal helper) -func (s *MemorySelectionService) trySelection( - ctx context.Context, - selectorModelID string, - memories []models.DecryptedMemory, - recentMessages []map[string]interface{}, - maxMemories int, -) ([]string, string, error) { - - // Get provider and model - provider, actualModel, err := s.getProviderAndModel(selectorModelID) - if err != nil { - return nil, "", fmt.Errorf("failed to get provider for selector: %w", err) - } - - log.Printf("🤖 [MEMORY-SELECTION] Using model: %s (%s)", selectorModelID, actualModel) - - // Build conversation context - conversationContext := s.buildConversationContext(recentMessages) - - // Build memory list for prompt - memoryList := s.buildMemoryListPrompt(memories) - - // Build user prompt - userPrompt := fmt.Sprintf(`RECENT CONVERSATION: -%s - -MEMORY BANK (%d memories): -%s - -Select up to %d memories that are DIRECTLY relevant to the current conversation. Return JSON with selected memory IDs and reasoning.`, - conversationContext, len(memories), memoryList, maxMemories) - - // Build messages - llmMessages := []map[string]interface{}{ - { - "role": "system", - "content": MemorySelectionSystemPrompt, - }, - { - "role": "user", - "content": userPrompt, - }, - } - - // Build request with structured output - requestBody := map[string]interface{}{ - "model": actualModel, - "messages": llmMessages, - "stream": false, - "temperature": 0.2, // Low temp for consistency - "response_format": map[string]interface{}{ - "type": "json_schema", - "json_schema": map[string]interface{}{ - "name": "memory_selection", - "strict": true, - "schema": memorySelectionSchema, - }, - }, - } - - reqBody, err := json.Marshal(requestBody) - if err != nil { - return nil, "", fmt.Errorf("failed to marshal request: %w", err) - } - - // Create HTTP request with timeout - httpReq, err := http.NewRequestWithContext(ctx, "POST", provider.BaseURL+"/chat/completions", bytes.NewBuffer(reqBody)) - if err != nil { - return nil, "", fmt.Errorf("failed to create request: %w", err) - } - - httpReq.Header.Set("Content-Type", "application/json") - httpReq.Header.Set("Authorization", "Bearer "+provider.APIKey) - - // Send request with 30s timeout - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(httpReq) - if err != nil { - return nil, "", fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - body, err := io.ReadAll(resp.Body) - if err != nil { - return nil, "", fmt.Errorf("failed to read response: %w", err) - } - - if resp.StatusCode != http.StatusOK { - log.Printf("⚠️ [MEMORY-SELECTION] API error: %s", string(body)) - return nil, "", fmt.Errorf("API error (status %d): %s", resp.StatusCode, string(body)) - } - - // Parse response - var apiResponse struct { - Choices []struct { - Message struct { - Content string `json:"content"` - } `json:"message"` - } `json:"choices"` - } - - if err := json.Unmarshal(body, &apiResponse); err != nil { - return nil, "", fmt.Errorf("failed to parse API response: %w", err) - } - - if len(apiResponse.Choices) == 0 { - return nil, "", fmt.Errorf("no response from selector model") - } - - // Parse the selection result - var result models.SelectedMemoriesFromLLM - content := apiResponse.Choices[0].Message.Content - - if err := json.Unmarshal([]byte(content), &result); err != nil { - // SECURITY: Don't log decrypted memory content - only log length - log.Printf("⚠️ [MEMORY-SELECTION] Failed to parse selection: %v (response length: %d bytes)", err, len(content)) - return nil, "", fmt.Errorf("failed to parse selection: %w", err) - } - - return result.SelectedMemoryIDs, result.Reasoning, nil -} - -// buildConversationContext builds a concise context from recent messages -func (s *MemorySelectionService) buildConversationContext(messages []map[string]interface{}) string { - var builder strings.Builder - - // Only include last 10 messages to keep prompt concise - startIdx := len(messages) - 10 - if startIdx < 0 { - startIdx = 0 - } - - for _, msg := range messages[startIdx:] { - role, _ := msg["role"].(string) - content, _ := msg["content"].(string) - - // Skip system messages - if role == "system" { - continue - } - - // Format message - if role == "user" { - builder.WriteString(fmt.Sprintf("USER: %s\n", content)) - } else if role == "assistant" { - // Truncate long assistant messages - if len(content) > 200 { - content = content[:200] + "..." - } - builder.WriteString(fmt.Sprintf("ASSISTANT: %s\n", content)) - } - } - - return builder.String() -} - -// buildMemoryListPrompt creates a numbered list of memories for the prompt -func (s *MemorySelectionService) buildMemoryListPrompt(memories []models.DecryptedMemory) string { - var builder strings.Builder - - for i, mem := range memories { - builder.WriteString(fmt.Sprintf("%d. [ID: %s] [Category: %s] %s\n", - i+1, - mem.ID.Hex(), - mem.Category, - mem.DecryptedContent, - )) - } - - return builder.String() -} - -// filterMemoriesByIDs filters memories to only include selected IDs -func (s *MemorySelectionService) filterMemoriesByIDs( - memories []models.DecryptedMemory, - selectedIDs []string, -) []models.DecryptedMemory { - - // Build set for O(1) lookup - idSet := make(map[string]bool) - for _, id := range selectedIDs { - idSet[id] = true - } - - filtered := make([]models.DecryptedMemory, 0, len(selectedIDs)) - - for _, mem := range memories { - if idSet[mem.ID.Hex()] { - filtered = append(filtered, mem) - } - } - - return filtered -} - -// getSelectorModelForUser gets user's preferred selector model -func (s *MemorySelectionService) getSelectorModelForUser(ctx context.Context, userID string) (string, error) { - // Query MongoDB for user preferences - usersCollection := s.mongodb.Collection(database.CollectionUsers) - - var user models.User - err := usersCollection.FindOne(ctx, map[string]interface{}{"supabaseUserId": userID}).Decode(&user) - if err != nil { - return "", nil // No user found, return empty (will use pool) - } - - // If user has a preference and it's not empty, use it - if user.Preferences.MemorySelectorModelID != "" { - log.Printf("🎯 [MEMORY-SELECTION] Using user-preferred model: %s", user.Preferences.MemorySelectorModelID) - return user.Preferences.MemorySelectorModelID, nil - } - - // No preference set - return empty (will use pool) - return "", nil -} - -// getProviderAndModel resolves model ID to provider and actual model name -func (s *MemorySelectionService) getProviderAndModel(modelID string) (*models.Provider, string, error) { - if modelID == "" { - return nil, "", fmt.Errorf("model ID is required") - } - - // Try to resolve through ChatService (handles aliases) - if s.chatService != nil { - if provider, actualModel, found := s.chatService.ResolveModelAlias(modelID); found { - return provider, actualModel, nil - } - } - - return nil, "", fmt.Errorf("model %s not found in providers", modelID) -} diff --git a/backend/internal/services/memory_storage_service.go b/backend/internal/services/memory_storage_service.go deleted file mode 100644 index 7153c761..00000000 --- a/backend/internal/services/memory_storage_service.go +++ /dev/null @@ -1,534 +0,0 @@ -package services - -import ( - "context" - "crypto/sha256" - "encoding/hex" - "fmt" - "log" - "strings" - "time" - - "claraverse/internal/crypto" - "claraverse/internal/database" - "claraverse/internal/models" - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/bson/primitive" - "go.mongodb.org/mongo-driver/mongo" - "go.mongodb.org/mongo-driver/mongo/options" -) - -// MemoryStorageService handles CRUD operations for memories with encryption and deduplication -type MemoryStorageService struct { - mongodb *database.MongoDB - collection *mongo.Collection - encryptionService *crypto.EncryptionService -} - -// NewMemoryStorageService creates a new memory storage service -func NewMemoryStorageService(mongodb *database.MongoDB, encryptionService *crypto.EncryptionService) *MemoryStorageService { - return &MemoryStorageService{ - mongodb: mongodb, - collection: mongodb.Collection(database.CollectionMemories), - encryptionService: encryptionService, - } -} - -// CreateMemory creates a new memory with encryption and deduplication -func (s *MemoryStorageService) CreateMemory(ctx context.Context, userID, content, category string, tags []string, sourceEngagement float64, conversationID string) (*models.Memory, error) { - if userID == "" { - return nil, fmt.Errorf("user ID is required") - } - if content == "" { - return nil, fmt.Errorf("memory content is required") - } - - // Normalize and hash content for deduplication - normalizedContent := s.normalizeContent(content) - contentHash := s.calculateHash(normalizedContent) - - // Check for duplicate - existingMemory, err := s.CheckDuplicate(ctx, userID, contentHash) - if err != nil && err != mongo.ErrNoDocuments { - return nil, fmt.Errorf("failed to check duplicate: %w", err) - } - - // If duplicate exists, update it instead - if existingMemory != nil { - log.Printf("🔄 [MEMORY-STORAGE] Duplicate memory found (ID: %s), updating instead", existingMemory.ID.Hex()) - return s.UpdateExistingMemory(ctx, existingMemory, tags, sourceEngagement) - } - - // Encrypt content - encryptedContent, err := s.encryptionService.Encrypt(userID, []byte(content)) - if err != nil { - return nil, fmt.Errorf("failed to encrypt memory content: %w", err) - } - - // Initial score is based solely on source engagement - initialScore := sourceEngagement - - // Create new memory - memory := &models.Memory{ - ID: primitive.NewObjectID(), - UserID: userID, - ConversationID: conversationID, - EncryptedContent: encryptedContent, - ContentHash: contentHash, - Category: category, - Tags: tags, - Score: initialScore, - AccessCount: 0, - LastAccessedAt: nil, - IsArchived: false, - ArchivedAt: nil, - SourceEngagement: sourceEngagement, - CreatedAt: time.Now(), - UpdatedAt: time.Now(), - Version: 1, - } - - // Insert into database - _, err = s.collection.InsertOne(ctx, memory) - if err != nil { - return nil, fmt.Errorf("failed to insert memory: %w", err) - } - - log.Printf("✅ [MEMORY-STORAGE] Created new memory (ID: %s, Category: %s, Score: %.2f)", memory.ID.Hex(), category, initialScore) - return memory, nil -} - -// UpdateExistingMemory updates an existing memory (for deduplication) -func (s *MemoryStorageService) UpdateExistingMemory(ctx context.Context, memory *models.Memory, newTags []string, sourceEngagement float64) (*models.Memory, error) { - // Merge tags (avoid duplicates) - tagMap := make(map[string]bool) - for _, tag := range memory.Tags { - tagMap[tag] = true - } - for _, tag := range newTags { - tagMap[tag] = true - } - mergedTags := make([]string, 0, len(tagMap)) - for tag := range tagMap { - mergedTags = append(mergedTags, tag) - } - - // Boost score slightly on re-mention (indicates importance) - newScore := memory.Score + 0.1 - if newScore > 1.0 { - newScore = 1.0 - } - - // Update engagement if higher - if sourceEngagement > memory.SourceEngagement { - memory.SourceEngagement = sourceEngagement - } - - // Update memory - update := bson.M{ - "$set": bson.M{ - "tags": mergedTags, - "score": newScore, - "sourceEngagement": memory.SourceEngagement, - "updatedAt": time.Now(), - }, - "$inc": bson.M{ - "version": 1, - }, - } - - result := s.collection.FindOneAndUpdate( - ctx, - bson.M{"_id": memory.ID}, - update, - options.FindOneAndUpdate().SetReturnDocument(options.After), - ) - - var updatedMemory models.Memory - if err := result.Decode(&updatedMemory); err != nil { - return nil, fmt.Errorf("failed to decode updated memory: %w", err) - } - - log.Printf("🔄 [MEMORY-STORAGE] Updated memory (ID: %s, New Score: %.2f, Version: %d)", updatedMemory.ID.Hex(), newScore, updatedMemory.Version) - return &updatedMemory, nil -} - -// UpdateMemoryInPlace atomically updates a memory (content, category, tags) -// SECURITY: Replaces delete-create pattern to prevent race conditions -func (s *MemoryStorageService) UpdateMemoryInPlace( - ctx context.Context, - userID string, - memoryID primitive.ObjectID, - content string, - category string, - tags []string, - sourceEngagement float64, - conversationID string, -) (*models.Memory, error) { - if userID == "" { - return nil, fmt.Errorf("user ID is required") - } - if content == "" { - return nil, fmt.Errorf("memory content is required") - } - - // Normalize and hash new content - normalizedContent := s.normalizeContent(content) - contentHash := s.calculateHash(normalizedContent) - - // Encrypt new content - encryptedContent, err := s.encryptionService.Encrypt(userID, []byte(content)) - if err != nil { - return nil, fmt.Errorf("failed to encrypt memory content: %w", err) - } - - now := time.Now() - - // Atomic update with user authorization check - update := bson.M{ - "$set": bson.M{ - "encryptedContent": encryptedContent, - "contentHash": contentHash, - "category": category, - "tags": tags, - "sourceEngagement": sourceEngagement, - "conversationId": conversationID, - "updatedAt": now, - }, - "$inc": bson.M{ - "version": 1, - }, - } - - // SECURITY: Filter includes userId to prevent unauthorized updates - filter := bson.M{ - "_id": memoryID, - "userId": userID, // Critical: ensures user can only update their own memories - } - - result := s.collection.FindOneAndUpdate( - ctx, - filter, - update, - options.FindOneAndUpdate().SetReturnDocument(options.After), - ) - - var updatedMemory models.Memory - if err := result.Decode(&updatedMemory); err != nil { - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("memory not found or access denied") - } - return nil, fmt.Errorf("failed to update memory: %w", err) - } - - log.Printf("✅ [MEMORY-STORAGE] Updated memory atomically (ID: %s, Version: %d)", updatedMemory.ID.Hex(), updatedMemory.Version) - return &updatedMemory, nil -} - -// GetMemory retrieves and decrypts a single memory -func (s *MemoryStorageService) GetMemory(ctx context.Context, userID string, memoryID primitive.ObjectID) (*models.DecryptedMemory, error) { - var memory models.Memory - err := s.collection.FindOne(ctx, bson.M{"_id": memoryID, "userId": userID}).Decode(&memory) - if err != nil { - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("memory not found") - } - return nil, fmt.Errorf("failed to get memory: %w", err) - } - - // Decrypt content - decryptedBytes, err := s.encryptionService.Decrypt(userID, memory.EncryptedContent) - if err != nil { - return nil, fmt.Errorf("failed to decrypt memory: %w", err) - } - - decryptedMemory := &models.DecryptedMemory{ - Memory: memory, - DecryptedContent: string(decryptedBytes), - } - - return decryptedMemory, nil -} - -// ListMemories retrieves memories with optional filters and pagination -func (s *MemoryStorageService) ListMemories(ctx context.Context, userID string, category string, tags []string, includeArchived bool, page, pageSize int) ([]models.DecryptedMemory, int64, error) { - // Build filter - filter := bson.M{"userId": userID} - - if !includeArchived { - filter["isArchived"] = false - } - - if category != "" { - filter["category"] = category - } - - if len(tags) > 0 { - filter["tags"] = bson.M{"$in": tags} - } - - // Count total - total, err := s.collection.CountDocuments(ctx, filter) - if err != nil { - return nil, 0, fmt.Errorf("failed to count memories: %w", err) - } - - // Calculate pagination - skip := (page - 1) * pageSize - findOptions := options.Find(). - SetSort(bson.D{{Key: "score", Value: -1}, {Key: "updatedAt", Value: -1}}). - SetSkip(int64(skip)). - SetLimit(int64(pageSize)) - - // Find memories - cursor, err := s.collection.Find(ctx, filter, findOptions) - if err != nil { - return nil, 0, fmt.Errorf("failed to find memories: %w", err) - } - defer cursor.Close(ctx) - - var memories []models.Memory - if err := cursor.All(ctx, &memories); err != nil { - return nil, 0, fmt.Errorf("failed to decode memories: %w", err) - } - - // Decrypt all memories - decryptedMemories := make([]models.DecryptedMemory, 0, len(memories)) - for _, memory := range memories { - decryptedBytes, err := s.encryptionService.Decrypt(userID, memory.EncryptedContent) - if err != nil { - log.Printf("⚠️ [MEMORY-STORAGE] Failed to decrypt memory %s: %v", memory.ID.Hex(), err) - continue - } - - decryptedMemories = append(decryptedMemories, models.DecryptedMemory{ - Memory: memory, - DecryptedContent: string(decryptedBytes), - }) - } - - return decryptedMemories, total, nil -} - -// GetActiveMemories retrieves all active (non-archived) memories for a user (decrypted) -func (s *MemoryStorageService) GetActiveMemories(ctx context.Context, userID string) ([]models.DecryptedMemory, error) { - filter := bson.M{ - "userId": userID, - "isArchived": false, - } - - findOptions := options.Find().SetSort(bson.D{{Key: "score", Value: -1}}) - - cursor, err := s.collection.Find(ctx, filter, findOptions) - if err != nil { - return nil, fmt.Errorf("failed to find active memories: %w", err) - } - defer cursor.Close(ctx) - - var memories []models.Memory - if err := cursor.All(ctx, &memories); err != nil { - return nil, fmt.Errorf("failed to decode memories: %w", err) - } - - // Decrypt all memories - decryptedMemories := make([]models.DecryptedMemory, 0, len(memories)) - for _, memory := range memories { - decryptedBytes, err := s.encryptionService.Decrypt(userID, memory.EncryptedContent) - if err != nil { - log.Printf("⚠️ [MEMORY-STORAGE] Failed to decrypt memory %s: %v", memory.ID.Hex(), err) - continue - } - - decryptedMemories = append(decryptedMemories, models.DecryptedMemory{ - Memory: memory, - DecryptedContent: string(decryptedBytes), - }) - } - - log.Printf("📚 [MEMORY-STORAGE] Retrieved %d active memories for user %s", len(decryptedMemories), userID) - return decryptedMemories, nil -} - -// UpdateMemoryAccess increments access count and updates last accessed timestamp -func (s *MemoryStorageService) UpdateMemoryAccess(ctx context.Context, memoryIDs []primitive.ObjectID) error { - if len(memoryIDs) == 0 { - return nil - } - - now := time.Now() - filter := bson.M{"_id": bson.M{"$in": memoryIDs}} - update := bson.M{ - "$inc": bson.M{"accessCount": 1}, - "$set": bson.M{"lastAccessedAt": now}, - } - - result, err := s.collection.UpdateMany(ctx, filter, update) - if err != nil { - return fmt.Errorf("failed to update memory access: %w", err) - } - - log.Printf("📊 [MEMORY-STORAGE] Updated access for %d memories", result.ModifiedCount) - return nil -} - -// ArchiveMemory marks a memory as archived -func (s *MemoryStorageService) ArchiveMemory(ctx context.Context, userID string, memoryID primitive.ObjectID) error { - now := time.Now() - update := bson.M{ - "$set": bson.M{ - "isArchived": true, - "archivedAt": now, - "updatedAt": now, - }, - } - - result, err := s.collection.UpdateOne(ctx, bson.M{"_id": memoryID, "userId": userID}, update) - if err != nil { - return fmt.Errorf("failed to archive memory: %w", err) - } - - if result.MatchedCount == 0 { - return fmt.Errorf("memory not found or access denied") - } - - log.Printf("📦 [MEMORY-STORAGE] Archived memory %s", memoryID.Hex()) - return nil -} - -// UnarchiveMemory restores an archived memory -func (s *MemoryStorageService) UnarchiveMemory(ctx context.Context, userID string, memoryID primitive.ObjectID) error { - update := bson.M{ - "$set": bson.M{ - "isArchived": false, - "archivedAt": nil, - "updatedAt": time.Now(), - }, - } - - result, err := s.collection.UpdateOne(ctx, bson.M{"_id": memoryID, "userId": userID}, update) - if err != nil { - return fmt.Errorf("failed to unarchive memory: %w", err) - } - - if result.MatchedCount == 0 { - return fmt.Errorf("memory not found or access denied") - } - - log.Printf("📤 [MEMORY-STORAGE] Unarchived memory %s", memoryID.Hex()) - return nil -} - -// DeleteMemory permanently deletes a memory -func (s *MemoryStorageService) DeleteMemory(ctx context.Context, userID string, memoryID primitive.ObjectID) error { - result, err := s.collection.DeleteOne(ctx, bson.M{"_id": memoryID, "userId": userID}) - if err != nil { - return fmt.Errorf("failed to delete memory: %w", err) - } - - if result.DeletedCount == 0 { - return fmt.Errorf("memory not found or access denied") - } - - log.Printf("🗑️ [MEMORY-STORAGE] Deleted memory %s", memoryID.Hex()) - return nil -} - -// CheckDuplicate checks if a memory with the same content hash exists -func (s *MemoryStorageService) CheckDuplicate(ctx context.Context, userID, contentHash string) (*models.Memory, error) { - var memory models.Memory - err := s.collection.FindOne(ctx, bson.M{"userId": userID, "contentHash": contentHash}).Decode(&memory) - if err != nil { - return nil, err - } - return &memory, nil -} - -// normalizeContent normalizes content for deduplication -func (s *MemoryStorageService) normalizeContent(content string) string { - // Convert to lowercase - normalized := strings.ToLower(content) - - // Replace word separators with spaces first (before removing other punctuation) - // This prevents words from merging when punctuation is removed - normalized = strings.ReplaceAll(normalized, "\n", " ") - normalized = strings.ReplaceAll(normalized, "\t", " ") - normalized = strings.ReplaceAll(normalized, "\r", " ") - normalized = strings.ReplaceAll(normalized, "-", " ") - normalized = strings.ReplaceAll(normalized, "_", " ") - - // Trim whitespace - normalized = strings.TrimSpace(normalized) - - // Remove punctuation (simple version) - normalized = strings.Map(func(r rune) rune { - if (r >= 'a' && r <= 'z') || (r >= '0' && r <= '9') || r == ' ' { - return r - } - return -1 - }, normalized) - - // Collapse multiple spaces - normalized = strings.Join(strings.Fields(normalized), " ") - return normalized -} - -// calculateHash calculates SHA-256 hash of content -func (s *MemoryStorageService) calculateHash(content string) string { - hash := sha256.Sum256([]byte(content)) - return hex.EncodeToString(hash[:]) -} - -// GetMemoryStats returns statistics about user's memories -func (s *MemoryStorageService) GetMemoryStats(ctx context.Context, userID string) (map[string]interface{}, error) { - // Count total memories - total, err := s.collection.CountDocuments(ctx, bson.M{"userId": userID}) - if err != nil { - return nil, fmt.Errorf("failed to count total memories: %w", err) - } - - // Count active memories - active, err := s.collection.CountDocuments(ctx, bson.M{"userId": userID, "isArchived": false}) - if err != nil { - return nil, fmt.Errorf("failed to count active memories: %w", err) - } - - // Count archived memories - archived, err := s.collection.CountDocuments(ctx, bson.M{"userId": userID, "isArchived": true}) - if err != nil { - return nil, fmt.Errorf("failed to count archived memories: %w", err) - } - - // Calculate average score - pipeline := mongo.Pipeline{ - {{Key: "$match", Value: bson.M{"userId": userID, "isArchived": false}}}, - {{Key: "$group", Value: bson.M{ - "_id": nil, - "avgScore": bson.M{"$avg": "$score"}, - }}}, - } - - cursor, err := s.collection.Aggregate(ctx, pipeline) - if err != nil { - return nil, fmt.Errorf("failed to aggregate scores: %w", err) - } - defer cursor.Close(ctx) - - var avgScoreResult struct { - AvgScore float64 `bson:"avgScore"` - } - avgScore := 0.0 - if cursor.Next(ctx) { - if err := cursor.Decode(&avgScoreResult); err == nil { - avgScore = avgScoreResult.AvgScore - } - } - - stats := map[string]interface{}{ - "total_memories": total, - "active_memories": active, - "archived_memories": archived, - "avg_score": avgScore, - } - - return stats, nil -} diff --git a/backend/internal/services/memory_storage_service_test.go b/backend/internal/services/memory_storage_service_test.go deleted file mode 100644 index 7b19e6e6..00000000 --- a/backend/internal/services/memory_storage_service_test.go +++ /dev/null @@ -1,346 +0,0 @@ -package services - -import ( - "testing" -) - -// TestNormalizeContent tests content normalization for deduplication -func TestNormalizeContent(t *testing.T) { - service := &MemoryStorageService{} - - tests := []struct { - name string - input string - expected string - }{ - { - name: "Basic normalization", - input: "User prefers dark mode", - expected: "user prefers dark mode", - }, - { - name: "Remove punctuation", - input: "User's name is John, and he likes coffee!", - expected: "users name is john and he likes coffee", - }, - { - name: "Collapse whitespace", - input: "User likes lots of spaces", - expected: "user likes lots of spaces", - }, - { - name: "Mixed case and punctuation", - input: "User PREFERS Dark-Mode!!!", - expected: "user prefers dark mode", - }, - { - name: "Trim whitespace", - input: " user prefers dark mode ", - expected: "user prefers dark mode", - }, - { - name: "Numbers preserved", - input: "User is 25 years old", - expected: "user is 25 years old", - }, - { - name: "Special characters removed", - input: "User's email: john@example.com", - expected: "users email johnexamplecom", - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - result := service.normalizeContent(tt.input) - if result != tt.expected { - t.Errorf("Expected: %q, got: %q", tt.expected, result) - } - }) - } -} - -// TestCalculateHash ensures consistent hashing -func TestCalculateHash(t *testing.T) { - service := &MemoryStorageService{} - - tests := []struct { - name string - input1 string - input2 string - shouldMatch bool - }{ - { - name: "Identical strings", - input1: "user prefers dark mode", - input2: "user prefers dark mode", - shouldMatch: true, - }, - { - name: "Different strings", - input1: "user prefers dark mode", - input2: "user prefers light mode", - shouldMatch: false, - }, - { - name: "Empty string", - input1: "", - input2: "", - shouldMatch: true, - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - hash1 := service.calculateHash(tt.input1) - hash2 := service.calculateHash(tt.input2) - - if tt.shouldMatch && hash1 != hash2 { - t.Errorf("Expected hashes to match: %s != %s", hash1, hash2) - } - - if !tt.shouldMatch && hash1 == hash2 { - t.Errorf("Expected hashes to differ, both got: %s", hash1) - } - - // Verify SHA-256 produces 64 character hex string - if len(hash1) != 64 { - t.Errorf("Expected 64 character hash, got %d", len(hash1)) - } - }) - } -} - -// TestDeduplicationLogic tests the deduplication flow -func TestDeduplicationLogic(t *testing.T) { - service := &MemoryStorageService{} - - // Test cases that should be considered duplicates after normalization - duplicates := []struct { - original string - variant string - }{ - { - original: "User prefers dark mode", - variant: "USER PREFERS DARK MODE", - }, - { - original: "User prefers dark mode", - variant: "User prefers dark-mode!", - }, - { - original: "User likes coffee", - variant: "User likes coffee!!!", - }, - { - original: "User name is John", - variant: " User name is John ", - }, - } - - for _, tt := range duplicates { - t.Run(tt.original, func(t *testing.T) { - normalized1 := service.normalizeContent(tt.original) - normalized2 := service.normalizeContent(tt.variant) - - hash1 := service.calculateHash(normalized1) - hash2 := service.calculateHash(normalized2) - - if hash1 != hash2 { - t.Errorf("Expected duplicates to have same hash:\n Original: %q -> %q -> %s\n Variant: %q -> %q -> %s", - tt.original, normalized1, hash1, - tt.variant, normalized2, hash2, - ) - } - }) - } - - // Test cases that should NOT be considered duplicates - nonDuplicates := []struct { - content1 string - content2 string - }{ - { - content1: "User prefers dark mode", - content2: "User prefers light mode", - }, - { - content1: "User likes coffee", - content2: "User likes tea", - }, - { - content1: "User is 25 years old", - content2: "User is 30 years old", - }, - } - - for i, tt := range nonDuplicates { - t.Run(string(rune('A'+i)), func(t *testing.T) { - normalized1 := service.normalizeContent(tt.content1) - normalized2 := service.normalizeContent(tt.content2) - - hash1 := service.calculateHash(normalized1) - hash2 := service.calculateHash(normalized2) - - if hash1 == hash2 { - t.Errorf("Expected different hashes:\n Content1: %q -> %q\n Content2: %q -> %q\n Both got: %s", - tt.content1, normalized1, - tt.content2, normalized2, - hash1, - ) - } - }) - } -} - -// TestContentHashCollisions checks for hash collision resistance -func TestContentHashCollisions(t *testing.T) { - service := &MemoryStorageService{} - - // Generate 1000 different normalized contents with guaranteed uniqueness - contents := make([]string, 1000) - for i := 0; i < 1000; i++ { - // Use index to guarantee uniqueness - contents[i] = service.normalizeContent(string(rune('a'+(i%26))) + " test content " + string(rune('0'+(i%10))) + string(rune('0'+((i/10)%10))) + string(rune('0'+((i/100)%10)))) - } - - // Calculate hashes and check for collisions - hashes := make(map[string]string) - collisionCount := 0 - for _, content := range contents { - hash := service.calculateHash(content) - - if existingContent, exists := hashes[hash]; exists { - // Only report as collision if content is actually different - if existingContent != content { - t.Errorf("Hash collision detected!\n Hash: %s\n Content1: %q\n Content2: %q", - hash, existingContent, content) - collisionCount++ - } - } else { - hashes[hash] = content - } - } - - if collisionCount > 0 { - t.Errorf("Found %d true hash collisions", collisionCount) - } - - t.Logf("Generated %d unique hashes without collisions", len(hashes)) -} - -// TestNormalizationEdgeCases tests edge cases in normalization -func TestNormalizationEdgeCases(t *testing.T) { - service := &MemoryStorageService{} - - tests := []struct { - name string - input string - expected string - }{ - { - name: "Only punctuation", - input: "!@#$%^&*()", - expected: "", - }, - { - name: "Only whitespace", - input: " ", - expected: "", - }, - { - name: "Unicode characters", - input: "User likes café ☕", - expected: "user likes caf", - }, - { - name: "Mixed alphanumeric", - input: "abc123DEF456", - expected: "abc123def456", - }, - { - name: "Newlines and tabs", - input: "User\nprefers\tdark\nmode", - expected: "user prefers dark mode", - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - result := service.normalizeContent(tt.input) - if result != tt.expected { - t.Errorf("Expected: %q, got: %q", tt.expected, result) - } - }) - } -} - -// TestHashConsistency ensures hashing is deterministic -func TestHashConsistency(t *testing.T) { - service := &MemoryStorageService{} - - content := "user prefers dark mode" - - // Calculate hash multiple times - hash1 := service.calculateHash(content) - hash2 := service.calculateHash(content) - hash3 := service.calculateHash(content) - - if hash1 != hash2 || hash2 != hash3 { - t.Errorf("Hash should be deterministic, got different values: %s, %s, %s", hash1, hash2, hash3) - } -} - -// TestMemoryCategories ensures category values are valid -func TestMemoryCategories(t *testing.T) { - validCategories := []string{ - "personal_info", - "preferences", - "context", - "fact", - "instruction", - } - - // This test documents the expected categories - t.Logf("Valid memory categories: %v", validCategories) - - for _, category := range validCategories { - if category == "" { - t.Errorf("Category should not be empty") - } - } -} - -// BenchmarkNormalizeContent benchmarks content normalization -func BenchmarkNormalizeContent(b *testing.B) { - service := &MemoryStorageService{} - content := "User prefers dark mode and likes to use the application at night!" - - b.ResetTimer() - for i := 0; i < b.N; i++ { - service.normalizeContent(content) - } -} - -// BenchmarkCalculateHash benchmarks hash calculation -func BenchmarkCalculateHash(b *testing.B) { - service := &MemoryStorageService{} - content := "user prefers dark mode and likes to use the application at night" - - b.ResetTimer() - for i := 0; i < b.N; i++ { - service.calculateHash(content) - } -} - -// BenchmarkDeduplicationPipeline benchmarks the full deduplication pipeline -func BenchmarkDeduplicationPipeline(b *testing.B) { - service := &MemoryStorageService{} - content := "User prefers dark mode and likes to use the application at night!" - - b.ResetTimer() - for i := 0; i < b.N; i++ { - normalized := service.normalizeContent(content) - service.calculateHash(normalized) - } -} diff --git a/backend/internal/services/metrics.go b/backend/internal/services/metrics.go deleted file mode 100644 index 8ca2a2d0..00000000 --- a/backend/internal/services/metrics.go +++ /dev/null @@ -1,114 +0,0 @@ -package services - -import ( - "github.com/prometheus/client_golang/prometheus" - "github.com/prometheus/client_golang/prometheus/promauto" -) - -// Metrics holds all custom Prometheus metrics for the application -type Metrics struct { - // WebSocket metrics - WebSocketConnections prometheus.Gauge - WebSocketMessages *prometheus.CounterVec - - // Chat metrics - ChatRequests prometheus.Counter - ChatRequestLatency prometheus.Histogram - ChatErrors *prometheus.CounterVec - - // Connection manager reference for dynamic metrics - connManager *ConnectionManager -} - -var globalMetrics *Metrics - -// InitMetrics initializes the Prometheus metrics -func InitMetrics(connManager *ConnectionManager) *Metrics { - metrics := &Metrics{ - connManager: connManager, - - // WebSocket active connections (gauge - can go up and down) - WebSocketConnections: promauto.NewGauge(prometheus.GaugeOpts{ - Name: "claraverse_websocket_connections_active", - Help: "Number of active WebSocket connections", - }), - - // WebSocket messages by type (counter - only goes up) - WebSocketMessages: promauto.NewCounterVec(prometheus.CounterOpts{ - Name: "claraverse_websocket_messages_total", - Help: "Total number of WebSocket messages by type", - }, []string{"type", "direction"}), // direction: "inbound" or "outbound" - - // Chat requests counter - ChatRequests: promauto.NewCounter(prometheus.CounterOpts{ - Name: "claraverse_chat_requests_total", - Help: "Total number of chat requests processed", - }), - - // Chat request latency histogram - ChatRequestLatency: promauto.NewHistogram(prometheus.HistogramOpts{ - Name: "claraverse_chat_request_duration_seconds", - Help: "Chat request latency in seconds", - Buckets: []float64{0.1, 0.5, 1, 2, 5, 10, 30, 60, 120}, // up to 2 minutes for LLM responses - }), - - // Chat errors by type - ChatErrors: promauto.NewCounterVec(prometheus.CounterOpts{ - Name: "claraverse_chat_errors_total", - Help: "Total number of chat errors by type", - }, []string{"error_type"}), - } - - // Register a collector that updates WebSocket connections from ConnectionManager - prometheus.MustRegister(prometheus.NewGaugeFunc( - prometheus.GaugeOpts{ - Name: "claraverse_websocket_connections_current", - Help: "Current number of active WebSocket connections (from connection manager)", - }, - func() float64 { - if connManager != nil { - return float64(connManager.Count()) - } - return 0 - }, - )) - - globalMetrics = metrics - return metrics -} - -// GetMetrics returns the global metrics instance -func GetMetrics() *Metrics { - return globalMetrics -} - -// RecordWebSocketConnect records a new WebSocket connection -func (m *Metrics) RecordWebSocketConnect() { - m.WebSocketConnections.Inc() -} - -// RecordWebSocketDisconnect records a WebSocket disconnection -func (m *Metrics) RecordWebSocketDisconnect() { - m.WebSocketConnections.Dec() -} - -// RecordWebSocketMessage records a WebSocket message -func (m *Metrics) RecordWebSocketMessage(msgType, direction string) { - m.WebSocketMessages.WithLabelValues(msgType, direction).Inc() -} - -// RecordChatRequest records a chat request -func (m *Metrics) RecordChatRequest() { - m.ChatRequests.Inc() -} - -// RecordChatLatency records chat request latency -func (m *Metrics) RecordChatLatency(seconds float64) { - m.ChatRequestLatency.Observe(seconds) -} - -// RecordChatError records a chat error -func (m *Metrics) RecordChatError(errorType string) { - m.ChatErrors.WithLabelValues(errorType).Inc() -} - diff --git a/backend/internal/services/model_management_service.go b/backend/internal/services/model_management_service.go deleted file mode 100644 index d39676ae..00000000 --- a/backend/internal/services/model_management_service.go +++ /dev/null @@ -1,1354 +0,0 @@ -package services - -import ( - "bytes" - "claraverse/internal/database" - "claraverse/internal/models" - "context" - "database/sql" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "os" - "strings" - "sync" - "time" -) - -// ModelManagementService handles model CRUD operations with dual-write to SQLite and providers.json -type ModelManagementService struct { - db *database.DB - providersFile string - fileMutex sync.Mutex // Protects providers.json file operations -} - -// NewModelManagementService creates a new model management service -// providersFile is optional - if empty, only database operations are performed -func NewModelManagementService(db *database.DB) *ModelManagementService { - return &ModelManagementService{ - db: db, - providersFile: "", // No longer using providers file - } -} - -// ================== DUAL-WRITE COORDINATOR ================== - -// CreateModel creates a new model in both database and providers.json -func (s *ModelManagementService) CreateModel(ctx context.Context, req *CreateModelRequest) (*models.Model, error) { - log.Printf("📝 [MODEL-MGMT] Creating model: %s (provider %d)", req.ModelID, req.ProviderID) - - // Step 1: Begin SQLite transaction - tx, err := s.db.Begin() - if err != nil { - return nil, fmt.Errorf("failed to begin transaction: %w", err) - } - defer tx.Rollback() - - // Step 2: Insert into database - _, err = tx.Exec(` - INSERT INTO models (id, provider_id, name, display_name, description, context_length, - supports_tools, supports_streaming, supports_vision, is_visible, system_prompt, fetched_at) - VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?) - `, req.ModelID, req.ProviderID, req.Name, req.DisplayName, req.Description, req.ContextLength, - req.SupportsTools, req.SupportsStreaming, req.SupportsVision, req.IsVisible, req.SystemPrompt, time.Now()) - - if err != nil { - return nil, fmt.Errorf("failed to insert model: %w", err) - } - - // Step 3: Commit transaction - if err := tx.Commit(); err != nil { - return nil, fmt.Errorf("failed to commit transaction: %w", err) - } - - log.Printf("✅ [MODEL-MGMT] Created model: %s", req.ModelID) - - // Fetch and return the created model - return s.GetModelByID(req.ModelID) -} - -// UpdateModel updates an existing model in both database and providers.json -func (s *ModelManagementService) UpdateModel(ctx context.Context, modelID string, req *UpdateModelRequest) (*models.Model, error) { - log.Printf("📝 [MODEL-MGMT] Updating model: %s", modelID) - - // Build dynamic update query - updateParts := []string{} - args := []interface{}{} - - if req.DisplayName != nil { - updateParts = append(updateParts, "display_name = ?") - args = append(args, *req.DisplayName) - } - if req.Description != nil { - updateParts = append(updateParts, "description = ?") - args = append(args, *req.Description) - } - if req.ContextLength != nil { - updateParts = append(updateParts, "context_length = ?") - args = append(args, *req.ContextLength) - } - if req.SupportsTools != nil { - updateParts = append(updateParts, "supports_tools = ?") - args = append(args, *req.SupportsTools) - } - if req.SupportsStreaming != nil { - updateParts = append(updateParts, "supports_streaming = ?") - args = append(args, *req.SupportsStreaming) - } - if req.SupportsVision != nil { - updateParts = append(updateParts, "supports_vision = ?") - args = append(args, *req.SupportsVision) - } - if req.IsVisible != nil { - updateParts = append(updateParts, "is_visible = ?") - args = append(args, *req.IsVisible) - log.Printf("[DEBUG] Adding is_visible to update: value=%v type=%T", *req.IsVisible, *req.IsVisible) - } else { - log.Printf("[DEBUG] is_visible field is nil, not updating") - } - if req.SystemPrompt != nil { - updateParts = append(updateParts, "system_prompt = ?") - args = append(args, *req.SystemPrompt) - } - if req.SmartToolRouter != nil { - updateParts = append(updateParts, "smart_tool_router = ?") - args = append(args, *req.SmartToolRouter) - } - if req.FreeTier != nil { - updateParts = append(updateParts, "free_tier = ?") - args = append(args, *req.FreeTier) - } - - if len(updateParts) == 0 { - return s.GetModelByID(modelID) - } - - // Add WHERE clause - args = append(args, modelID) - query := fmt.Sprintf("UPDATE models SET %s WHERE id = ?", joinStrings(updateParts, ", ")) - - log.Printf("[DEBUG] SQL Query: %s", query) - log.Printf("[DEBUG] SQL Args: %v", args) - - // Step 1: Begin transaction - tx, err := s.db.Begin() - if err != nil { - return nil, fmt.Errorf("failed to begin transaction: %w", err) - } - defer tx.Rollback() - - // Step 2: Execute update - result, err := tx.Exec(query, args...) - if err != nil { - return nil, fmt.Errorf("failed to update model: %w", err) - } - - rowsAffected, _ := result.RowsAffected() - log.Printf("[DEBUG] SQL execution successful, rows affected: %d", rowsAffected) - - // Step 3: Commit transaction - if err := tx.Commit(); err != nil { - return nil, fmt.Errorf("failed to commit transaction: %w", err) - } - - log.Printf("✅ [MODEL-MGMT] Updated model: %s", modelID) - - // Get fresh model state from database - updatedModel, err := s.GetModelByID(modelID) - if err != nil { - return nil, err - } - log.Printf("[DEBUG] Retrieved is_visible after update: %v", updatedModel.IsVisible) - return updatedModel, nil -} - -// DeleteModel deletes a model from both database and providers.json -func (s *ModelManagementService) DeleteModel(ctx context.Context, modelID string) error { - log.Printf("🗑️ [MODEL-MGMT] Deleting model: %s", modelID) - - // Step 1: Begin transaction - tx, err := s.db.Begin() - if err != nil { - return fmt.Errorf("failed to begin transaction: %w", err) - } - defer tx.Rollback() - - // Step 2: Delete from database (cascades to model_capabilities and model_aliases) - result, err := tx.Exec("DELETE FROM models WHERE id = ?", modelID) - if err != nil { - return fmt.Errorf("failed to delete model: %w", err) - } - - rowsAffected, _ := result.RowsAffected() - if rowsAffected == 0 { - return fmt.Errorf("model not found: %s", modelID) - } - - // Step 3: Commit transaction - if err := tx.Commit(); err != nil { - return fmt.Errorf("failed to commit transaction: %w", err) - } - - log.Printf("✅ [MODEL-MGMT] Deleted model: %s", modelID) - return nil -} - -// reloadConfigServiceCache reloads the config service cache from database -func (s *ModelManagementService) reloadConfigServiceCache() error { - log.Printf("🔄 [MODEL-MGMT] Reloading config service cache from database...") - - configService := GetConfigService() - - // Get all providers and their aliases from database - rows, err := s.db.Query(` - SELECT DISTINCT provider_id FROM model_aliases - `) - if err != nil { - return fmt.Errorf("failed to query provider IDs: %w", err) - } - defer rows.Close() - - var providerIDs []int - for rows.Next() { - var providerID int - if err := rows.Scan(&providerID); err != nil { - return fmt.Errorf("failed to scan provider ID: %w", err) - } - providerIDs = append(providerIDs, providerID) - } - - // Reload aliases for each provider - for _, providerID := range providerIDs { - aliases, err := s.getModelAliasesForProvider(providerID) - if err != nil { - log.Printf("⚠️ [MODEL-MGMT] Failed to load aliases for provider %d: %v", providerID, err) - continue - } - - // Update config service cache - configService.SetModelAliases(providerID, aliases) - log.Printf("✅ [MODEL-MGMT] Reloaded %d aliases for provider %d", len(aliases), providerID) - } - - log.Printf("✅ [MODEL-MGMT] Config service cache reloaded successfully") - return nil -} - -// getModelAliasesForProvider retrieves all model aliases for a provider from model_aliases table -func (s *ModelManagementService) getModelAliasesForProvider(providerID int) (map[string]models.ModelAlias, error) { - rows, err := s.db.Query(` - SELECT alias_name, model_id, display_name, description, supports_vision, - agents_enabled, smart_tool_router, free_tier, structured_output_support, - structured_output_compliance, structured_output_warning, structured_output_speed_ms, - structured_output_badge, memory_extractor, memory_selector - FROM model_aliases - WHERE provider_id = ? - `, providerID) - if err != nil { - return nil, err - } - defer rows.Close() - - aliases := make(map[string]models.ModelAlias) - - for rows.Next() { - var aliasName, modelID, displayName string - var description, structuredOutputSupport, structuredOutputWarning, structuredOutputBadge sql.NullString - var supportsVision, agentsEnabled, smartToolRouter, freeTier, memoryExtractor, memorySelector sql.NullBool - var structuredOutputCompliance, structuredOutputSpeedMs sql.NullInt64 - - err := rows.Scan(&aliasName, &modelID, &displayName, &description, &supportsVision, - &agentsEnabled, &smartToolRouter, &freeTier, &structuredOutputSupport, - &structuredOutputCompliance, &structuredOutputWarning, &structuredOutputSpeedMs, - &structuredOutputBadge, &memoryExtractor, &memorySelector) - if err != nil { - return nil, err - } - - alias := models.ModelAlias{ - ActualModel: modelID, - DisplayName: displayName, - } - - if description.Valid { - alias.Description = description.String - } - if supportsVision.Valid { - vision := supportsVision.Bool - alias.SupportsVision = &vision - } - if agentsEnabled.Valid { - agents := agentsEnabled.Bool - alias.Agents = &agents - } - if smartToolRouter.Valid { - router := smartToolRouter.Bool - alias.SmartToolRouter = &router - } - if freeTier.Valid { - free := freeTier.Bool - alias.FreeTier = &free - } - if structuredOutputSupport.Valid { - alias.StructuredOutputSupport = structuredOutputSupport.String - } - if structuredOutputCompliance.Valid { - compliance := int(structuredOutputCompliance.Int64) - alias.StructuredOutputCompliance = &compliance - } - if structuredOutputWarning.Valid { - alias.StructuredOutputWarning = structuredOutputWarning.String - } - if structuredOutputSpeedMs.Valid { - speed := int(structuredOutputSpeedMs.Int64) - alias.StructuredOutputSpeedMs = &speed - } - if structuredOutputBadge.Valid { - alias.StructuredOutputBadge = structuredOutputBadge.String - } - if memoryExtractor.Valid { - extractor := memoryExtractor.Bool - alias.MemoryExtractor = &extractor - } - if memorySelector.Valid { - selector := memorySelector.Bool - alias.MemorySelector = &selector - } - - aliases[aliasName] = alias - } - - return aliases, nil -} - -// ================== MODEL FETCHING ================== - -// FetchModelsFromProvider fetches models from a provider's API and stores them -func (s *ModelManagementService) FetchModelsFromProvider(ctx context.Context, providerID int) (int, error) { - log.Printf("🔄 [MODEL-MGMT] Fetching models from provider %d", providerID) - - // Get provider details - provider, err := s.getProviderByID(providerID) - if err != nil { - return 0, fmt.Errorf("failed to get provider: %w", err) - } - - // Create HTTP request to provider's /v1/models endpoint - req, err := http.NewRequest("GET", provider.BaseURL+"/models", nil) - if err != nil { - return 0, fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Authorization", "Bearer "+provider.APIKey) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return 0, fmt.Errorf("failed to fetch models: %w", err) - } - defer resp.Body.Close() - - if resp.StatusCode != http.StatusOK { - body, _ := io.ReadAll(resp.Body) - return 0, fmt.Errorf("API error (status %d): %s", resp.StatusCode, string(body)) - } - - // Parse response - body, err := io.ReadAll(resp.Body) - if err != nil { - return 0, fmt.Errorf("failed to read response: %w", err) - } - - var modelsResp models.OpenAIModelsResponse - if err := json.Unmarshal(body, &modelsResp); err != nil { - return 0, fmt.Errorf("failed to parse models response: %w", err) - } - - log.Printf("✅ [MODEL-MGMT] Fetched %d models from provider %d", len(modelsResp.Data), providerID) - - // Store models in database (all hidden by default - admin must manually toggle visibility) - count := 0 - for _, modelData := range modelsResp.Data { - _, err := s.db.Exec(` - INSERT INTO models (id, provider_id, name, display_name, is_visible, fetched_at) - VALUES (?, ?, ?, ?, 0, ?) - ON DUPLICATE KEY UPDATE - name = VALUES(name), - display_name = VALUES(display_name), - fetched_at = VALUES(fetched_at) - `, modelData.ID, providerID, modelData.ID, modelData.ID, time.Now()) - - if err != nil { - log.Printf("⚠️ [MODEL-MGMT] Failed to store model %s: %v", modelData.ID, err) - } else { - count++ - } - } - - log.Printf("✅ [MODEL-MGMT] Stored %d models for provider %d", count, providerID) - return count, nil -} - -// ================== MODEL TESTING ================== - -// TestModelConnection performs a basic connection test -func (s *ModelManagementService) TestModelConnection(ctx context.Context, modelID string) (*ConnectionTestResult, error) { - log.Printf("🔌 [MODEL-MGMT] Testing connection for model: %s", modelID) - - model, err := s.GetModelByID(modelID) - if err != nil { - return nil, err - } - - provider, err := s.getProviderByID(model.ProviderID) - if err != nil { - return nil, err - } - - start := time.Now() - - // Send test prompt - reqBody := map[string]interface{}{ - "model": modelID, - "messages": []map[string]string{ - {"role": "user", "content": "Hello! Respond with OK"}, - }, - "max_tokens": 10, - } - - jsonData, _ := json.Marshal(reqBody) - req, err := http.NewRequest("POST", provider.BaseURL+"/chat/completions", bytes.NewBuffer(jsonData)) - if err != nil { - return nil, err - } - - req.Header.Set("Authorization", "Bearer "+provider.APIKey) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return &ConnectionTestResult{ - ModelID: modelID, - Passed: false, - LatencyMs: int(time.Since(start).Milliseconds()), - Error: err.Error(), - }, nil - } - defer resp.Body.Close() - - latency := int(time.Since(start).Milliseconds()) - - if resp.StatusCode != http.StatusOK { - body, _ := io.ReadAll(resp.Body) - return &ConnectionTestResult{ - ModelID: modelID, - Passed: false, - LatencyMs: latency, - Error: fmt.Sprintf("API error (status %d): %s", resp.StatusCode, string(body)), - }, nil - } - - // Update database - _, err = s.db.Exec(` - REPLACE INTO model_capabilities (model_id, provider_id, connection_test_passed, last_tested) - VALUES (?, ?, 1, ?) - `, modelID, model.ProviderID, time.Now()) - - if err != nil { - log.Printf("⚠️ [MODEL-MGMT] Failed to update test result: %v", err) - } - - log.Printf("✅ [MODEL-MGMT] Connection test passed for %s (latency: %dms)", modelID, latency) - return &ConnectionTestResult{ - ModelID: modelID, - Passed: true, - LatencyMs: latency, - }, nil -} - -// RunBenchmark runs a comprehensive benchmark suite on a model -func (s *ModelManagementService) RunBenchmark(ctx context.Context, modelID string) (*BenchmarkResults, error) { - log.Printf("📊 [MODEL-MGMT] Starting benchmark suite for model: %s", modelID) - - model, err := s.GetModelByID(modelID) - if err != nil { - log.Printf("❌ [MODEL-MGMT] Failed to get model %s: %v", modelID, err) - return nil, fmt.Errorf("model not found: %w", err) - } - - provider, err := s.getProviderByID(model.ProviderID) - if err != nil { - log.Printf("❌ [MODEL-MGMT] Failed to get provider %d: %v", model.ProviderID, err) - return nil, fmt.Errorf("provider not found: %w", err) - } - - log.Printf(" Provider: %s (%s)", provider.Name, provider.BaseURL) - - results := &BenchmarkResults{ - LastTested: time.Now().Format(time.RFC3339), - } - - // 1. Run connection test - log.Printf(" [1/3] Running connection test...") - connResult, err := s.TestModelConnection(ctx, modelID) - if err == nil { - results.ConnectionTest = connResult - log.Printf(" ✓ Connection test complete") - } else { - log.Printf(" ✗ Connection test failed: %v", err) - } - - // 2. Run structured output test - log.Printf(" [2/3] Running structured output test (5 prompts)...") - structuredResult, err := s.testStructuredOutput(ctx, modelID, provider) - if err == nil { - results.StructuredOutput = structuredResult - log.Printf(" ✓ Structured output test complete") - } else { - log.Printf(" ✗ Structured output test failed: %v", err) - } - - // 3. Run performance test - log.Printf(" [3/3] Running performance test (3 prompts)...") - perfResult, err := s.testPerformance(ctx, modelID, provider) - if err == nil { - results.Performance = perfResult - log.Printf(" ✓ Performance test complete") - } else { - log.Printf(" ✗ Performance test failed: %v", err) - } - - // 4. Update database with benchmark results - if results.StructuredOutput != nil { - _, err = s.db.Exec(` - UPDATE model_capabilities - SET structured_output_compliance = ?, - structured_output_speed_ms = ?, - benchmark_date = ? - WHERE model_id = ? AND provider_id = ? - `, results.StructuredOutput.CompliancePercentage, - results.StructuredOutput.AverageSpeedMs, - time.Now(), - modelID, - model.ProviderID) - - if err != nil { - log.Printf("⚠️ [MODEL-MGMT] Failed to update benchmark results in DB: %v", err) - } - } - - log.Printf("✅ [MODEL-MGMT] Benchmark suite completed for %s", modelID) - return results, nil -} - -// testStructuredOutput tests JSON schema compliance -func (s *ModelManagementService) testStructuredOutput(ctx context.Context, modelID string, provider *models.Provider) (*StructuredOutputBenchmark, error) { - log.Printf("🧪 [MODEL-MGMT] Testing structured output for model: %s at %s", modelID, provider.BaseURL) - - testPrompts := []string{ - `Generate a JSON object with fields: name (string), age (number), active (boolean)`, - `Create a JSON array with 3 objects, each with id and title fields`, - `Output JSON with nested structure: user { profile { name, email } }`, - `Return JSON with array field "tags" containing 5 strings`, - `Generate JSON matching: { count: number, items: string[] }`, - } - - passedTests := 0 - totalLatency := 0 - totalTests := len(testPrompts) - failureReasons := []string{} - - client := &http.Client{Timeout: 60 * time.Second} - - for i, prompt := range testPrompts { - start := time.Now() - - reqBody := map[string]interface{}{ - "model": modelID, - "messages": []map[string]string{ - {"role": "user", "content": prompt}, - }, - "max_tokens": 200, - } - - jsonData, _ := json.Marshal(reqBody) - req, err := http.NewRequest("POST", provider.BaseURL+"/chat/completions", bytes.NewBuffer(jsonData)) - if err != nil { - failureReasons = append(failureReasons, fmt.Sprintf("Test %d: request creation failed - %v", i+1, err)) - continue - } - - req.Header.Set("Authorization", "Bearer "+provider.APIKey) - req.Header.Set("Content-Type", "application/json") - - resp, err := client.Do(req) - if err != nil { - failureReasons = append(failureReasons, fmt.Sprintf("Test %d: HTTP request failed - %v", i+1, err)) - continue - } - - latency := int(time.Since(start).Milliseconds()) - totalLatency += latency - - if resp.StatusCode == http.StatusOK { - // Check if response is valid JSON - body, _ := io.ReadAll(resp.Body) - var result map[string]interface{} - if json.Unmarshal(body, &result) == nil { - passedTests++ - log.Printf(" ✓ Test %d passed (latency: %dms)", i+1, latency) - } else { - failureReasons = append(failureReasons, fmt.Sprintf("Test %d: invalid JSON response", i+1)) - } - } else { - body, _ := io.ReadAll(resp.Body) - failureReasons = append(failureReasons, fmt.Sprintf("Test %d: HTTP %d - %s", i+1, resp.StatusCode, string(body))) - } - resp.Body.Close() - } - - if len(failureReasons) > 0 { - log.Printf("⚠️ [MODEL-MGMT] Structured output test failures: %v", failureReasons) - } - - compliancePercentage := (passedTests * 100) / totalTests - avgSpeedMs := 0 - if totalLatency > 0 && passedTests > 0 { - avgSpeedMs = totalLatency / passedTests - } - - qualityLevel := "poor" - if compliancePercentage >= 90 { - qualityLevel = "excellent" - } else if compliancePercentage >= 75 { - qualityLevel = "good" - } else if compliancePercentage >= 50 { - qualityLevel = "fair" - } - - log.Printf(" 📊 Results: %d/%d passed (%d%%), avg speed: %dms, quality: %s", - passedTests, totalTests, compliancePercentage, avgSpeedMs, qualityLevel) - - return &StructuredOutputBenchmark{ - CompliancePercentage: compliancePercentage, - AverageSpeedMs: avgSpeedMs, - QualityLevel: qualityLevel, - TestsPassed: passedTests, - TestsFailed: totalTests - passedTests, - }, nil -} - -// testPerformance tests model performance metrics -func (s *ModelManagementService) testPerformance(ctx context.Context, modelID string, provider *models.Provider) (*PerformanceBenchmark, error) { - log.Printf("⚡ [MODEL-MGMT] Testing performance for model: %s", modelID) - - testPrompt := "Write a detailed explanation of machine learning in 200 words." - numTests := 3 - totalLatency := 0 - totalTokens := 0 - - client := &http.Client{Timeout: 60 * time.Second} - - for i := 0; i < numTests; i++ { - start := time.Now() - - reqBody := map[string]interface{}{ - "model": modelID, - "messages": []map[string]string{ - {"role": "user", "content": testPrompt}, - }, - "max_tokens": 300, - } - - jsonData, _ := json.Marshal(reqBody) - req, err := http.NewRequest("POST", provider.BaseURL+"/chat/completions", bytes.NewBuffer(jsonData)) - if err != nil { - continue - } - - req.Header.Set("Authorization", "Bearer "+provider.APIKey) - req.Header.Set("Content-Type", "application/json") - - resp, err := client.Do(req) - if err != nil { - continue - } - - latency := int(time.Since(start).Milliseconds()) - totalLatency += latency - - if resp.StatusCode == http.StatusOK { - body, _ := io.ReadAll(resp.Body) - var result map[string]interface{} - if json.Unmarshal(body, &result) == nil { - if usage, ok := result["usage"].(map[string]interface{}); ok { - if completionTokens, ok := usage["completion_tokens"].(float64); ok { - totalTokens += int(completionTokens) - } - } - } - } - resp.Body.Close() - } - - avgLatencyMs := totalLatency / numTests - avgTokens := float64(totalTokens) / float64(numTests) - tokensPerSecond := (avgTokens / float64(avgLatencyMs)) * 1000 - - return &PerformanceBenchmark{ - TokensPerSecond: tokensPerSecond, - AvgLatencyMs: avgLatencyMs, - TestedAt: time.Now().Format(time.RFC3339), - }, nil -} - -// ================== ALIAS MANAGEMENT ================== - -// CreateAlias creates a new model alias -func (s *ModelManagementService) CreateAlias(ctx context.Context, req *CreateAliasRequest) error { - log.Printf("📝 [MODEL-MGMT] Creating alias: %s -> %s (provider %d)", req.AliasName, req.ModelID, req.ProviderID) - - // Begin transaction - tx, err := s.db.Begin() - if err != nil { - return fmt.Errorf("failed to begin transaction: %w", err) - } - defer tx.Rollback() - - // Convert empty string to NULL for ENUM fields - var structuredOutputSupport interface{} - if req.StructuredOutputSupport == "" { - structuredOutputSupport = nil - } else { - structuredOutputSupport = req.StructuredOutputSupport - } - - // Convert empty strings to NULL for optional text fields - var structuredOutputWarning interface{} - if req.StructuredOutputWarning == "" { - structuredOutputWarning = nil - } else { - structuredOutputWarning = req.StructuredOutputWarning - } - - var structuredOutputBadge interface{} - if req.StructuredOutputBadge == "" { - structuredOutputBadge = nil - } else { - structuredOutputBadge = req.StructuredOutputBadge - } - - var description interface{} - if req.Description == "" { - description = nil - } else { - description = req.Description - } - - // Insert alias - _, err = tx.Exec(` - INSERT INTO model_aliases (alias_name, model_id, provider_id, display_name, description, - supports_vision, agents_enabled, smart_tool_router, free_tier, - structured_output_support, structured_output_compliance, structured_output_warning, - structured_output_speed_ms, structured_output_badge, memory_extractor, memory_selector) - VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?) - `, req.AliasName, req.ModelID, req.ProviderID, req.DisplayName, description, - req.SupportsVision, req.AgentsEnabled, req.SmartToolRouter, req.FreeTier, - structuredOutputSupport, req.StructuredOutputCompliance, structuredOutputWarning, - req.StructuredOutputSpeedMs, structuredOutputBadge, req.MemoryExtractor, req.MemorySelector) - - if err != nil { - return fmt.Errorf("failed to insert alias: %w", err) - } - - if err := tx.Commit(); err != nil { - return fmt.Errorf("failed to commit transaction: %w", err) - } - - // Reload config service cache with updated aliases from database - if err := s.reloadConfigServiceCache(); err != nil { - log.Printf("⚠️ [MODEL-MGMT] Failed to reload config cache: %v", err) - } - - log.Printf("✅ [MODEL-MGMT] Created alias: %s", req.AliasName) - return nil -} - -// DeleteAlias deletes a model alias -func (s *ModelManagementService) DeleteAlias(ctx context.Context, aliasName string, providerID int) error { - log.Printf("🗑️ [MODEL-MGMT] Deleting alias: %s (provider %d)", aliasName, providerID) - - tx, err := s.db.Begin() - if err != nil { - return fmt.Errorf("failed to begin transaction: %w", err) - } - defer tx.Rollback() - - result, err := tx.Exec("DELETE FROM model_aliases WHERE alias_name = ? AND provider_id = ?", aliasName, providerID) - if err != nil { - return fmt.Errorf("failed to delete alias: %w", err) - } - - rowsAffected, _ := result.RowsAffected() - if rowsAffected == 0 { - return fmt.Errorf("alias not found: %s", aliasName) - } - - if err := tx.Commit(); err != nil { - return fmt.Errorf("failed to commit transaction: %w", err) - } - - // Reload config service cache with updated aliases from database - if err := s.reloadConfigServiceCache(); err != nil { - log.Printf("⚠️ [MODEL-MGMT] Failed to reload config cache: %v", err) - } - - log.Printf("✅ [MODEL-MGMT] Deleted alias: %s", aliasName) - return nil -} - -// GetAliases retrieves all aliases for a model -func (s *ModelManagementService) GetAliases(ctx context.Context, modelID string) ([]models.ModelAliasView, error) { - log.Printf("🔍 [MODEL-MGMT] Fetching aliases for model: %s", modelID) - - rows, err := s.db.Query(` - SELECT id, alias_name, model_id, provider_id, display_name, description, - supports_vision, agents_enabled, smart_tool_router, free_tier, - structured_output_support, structured_output_compliance, structured_output_warning, - structured_output_speed_ms, structured_output_badge, memory_extractor, memory_selector, - created_at, updated_at - FROM model_aliases - WHERE model_id = ? - ORDER BY created_at DESC - `, modelID) - - if err != nil { - return nil, fmt.Errorf("failed to query aliases: %w", err) - } - defer rows.Close() - - var aliases []models.ModelAliasView - for rows.Next() { - var alias models.ModelAliasView - var description, structuredOutputSupport, structuredOutputWarning, structuredOutputBadge sql.NullString - var structuredOutputCompliance, structuredOutputSpeedMs sql.NullInt64 - var supportsVision, agentsEnabled, smartToolRouter, freeTier, memoryExtractor, memorySelector sql.NullBool - - err := rows.Scan( - &alias.ID, &alias.AliasName, &alias.ModelID, &alias.ProviderID, &alias.DisplayName, &description, - &supportsVision, &agentsEnabled, &smartToolRouter, &freeTier, - &structuredOutputSupport, &structuredOutputCompliance, &structuredOutputWarning, - &structuredOutputSpeedMs, &structuredOutputBadge, &memoryExtractor, &memorySelector, - &alias.CreatedAt, &alias.UpdatedAt, - ) - - if err != nil { - return nil, fmt.Errorf("failed to scan alias: %w", err) - } - - // Handle nullable fields - if description.Valid { - alias.Description = &description.String - } - if supportsVision.Valid { - alias.SupportsVision = &supportsVision.Bool - } - if agentsEnabled.Valid { - alias.AgentsEnabled = &agentsEnabled.Bool - } - if smartToolRouter.Valid { - alias.SmartToolRouter = &smartToolRouter.Bool - } - if freeTier.Valid { - alias.FreeTier = &freeTier.Bool - } - if structuredOutputSupport.Valid { - alias.StructuredOutputSupport = &structuredOutputSupport.String - } - if structuredOutputCompliance.Valid { - compliance := int(structuredOutputCompliance.Int64) - alias.StructuredOutputCompliance = &compliance - } - if structuredOutputWarning.Valid { - alias.StructuredOutputWarning = &structuredOutputWarning.String - } - if structuredOutputSpeedMs.Valid { - speed := int(structuredOutputSpeedMs.Int64) - alias.StructuredOutputSpeedMs = &speed - } - if structuredOutputBadge.Valid { - alias.StructuredOutputBadge = &structuredOutputBadge.String - } - if memoryExtractor.Valid { - alias.MemoryExtractor = &memoryExtractor.Bool - } - if memorySelector.Valid { - alias.MemorySelector = &memorySelector.Bool - } - - aliases = append(aliases, alias) - } - - if err := rows.Err(); err != nil { - return nil, fmt.Errorf("error iterating aliases: %w", err) - } - - log.Printf("✅ [MODEL-MGMT] Found %d aliases for model %s", len(aliases), modelID) - return aliases, nil -} - -// ImportAliasesFromJSON imports all aliases from providers.json into the database -func (s *ModelManagementService) ImportAliasesFromJSON(ctx context.Context) error { - log.Printf("📥 [MODEL-MGMT] Starting import of aliases from providers.json to database...") - - // Read providers.json - data, err := os.ReadFile(s.providersFile) - if err != nil { - return fmt.Errorf("failed to read providers.json: %w", err) - } - - var cfg models.ProvidersConfig - if err := json.Unmarshal(data, &cfg); err != nil { - return fmt.Errorf("failed to parse providers.json: %w", err) - } - - totalImported := 0 - totalSkipped := 0 - - // Iterate through all providers - for _, provider := range cfg.Providers { - // Get provider ID from database - var providerID int - err := s.db.QueryRow(`SELECT id FROM providers WHERE name = ?`, provider.Name).Scan(&providerID) - if err != nil { - log.Printf("⚠️ [MODEL-MGMT] Provider %s not found in database, skipping aliases", provider.Name) - continue - } - - // Iterate through all model_aliases for this provider - for aliasName, aliasConfig := range provider.ModelAliases { - // Check if alias already exists - var existingID int - err := s.db.QueryRow(`SELECT id FROM model_aliases WHERE alias_name = ? AND provider_id = ?`, - aliasName, providerID).Scan(&existingID) - - if err == nil { - // Alias already exists, skip - totalSkipped++ - continue - } - - // Extract values from aliasConfig - modelID := aliasConfig.ActualModel - displayName := aliasConfig.DisplayName - description := aliasConfig.Description - supportsVision := aliasConfig.SupportsVision - agentsEnabled := aliasConfig.Agents - smartToolRouter := aliasConfig.SmartToolRouter - freeTier := aliasConfig.FreeTier - structuredOutputSupport := aliasConfig.StructuredOutputSupport - structuredOutputCompliance := aliasConfig.StructuredOutputCompliance - structuredOutputWarning := aliasConfig.StructuredOutputWarning - structuredOutputSpeedMs := aliasConfig.StructuredOutputSpeedMs - structuredOutputBadge := aliasConfig.StructuredOutputBadge - memoryExtractor := aliasConfig.MemoryExtractor - memorySelector := aliasConfig.MemorySelector - - // Insert alias into database - _, err = s.db.Exec(` - INSERT INTO model_aliases (alias_name, model_id, provider_id, display_name, description, - supports_vision, agents_enabled, smart_tool_router, free_tier, - structured_output_support, structured_output_compliance, structured_output_warning, - structured_output_speed_ms, structured_output_badge, memory_extractor, memory_selector) - VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?) - `, aliasName, modelID, providerID, displayName, description, - supportsVision, agentsEnabled, smartToolRouter, freeTier, - structuredOutputSupport, structuredOutputCompliance, structuredOutputWarning, - structuredOutputSpeedMs, structuredOutputBadge, memoryExtractor, memorySelector) - - if err != nil { - log.Printf("⚠️ [MODEL-MGMT] Failed to import alias %s: %v", aliasName, err) - continue - } - - totalImported++ - log.Printf(" ✓ Imported alias: %s -> %s (provider: %s)", aliasName, modelID, provider.Name) - } - } - - log.Printf("✅ [MODEL-MGMT] Import complete: %d aliases imported, %d skipped (already exist)", totalImported, totalSkipped) - return nil -} - -// ================== HELPER METHODS ================== - -// GetModelByID retrieves a model by ID -func (s *ModelManagementService) GetModelByID(modelID string) (*models.Model, error) { - var m models.Model - var displayName, description, systemPrompt, providerFavicon sql.NullString - var contextLength sql.NullInt64 - var fetchedAt sql.NullTime - - err := s.db.QueryRow(` - SELECT m.id, m.provider_id, p.name as provider_name, p.favicon as provider_favicon, - m.name, m.display_name, m.description, m.context_length, m.supports_tools, - m.supports_streaming, m.supports_vision, m.smart_tool_router, m.is_visible, m.system_prompt, m.fetched_at - FROM models m - JOIN providers p ON m.provider_id = p.id - WHERE m.id = ? - `, modelID).Scan(&m.ID, &m.ProviderID, &m.ProviderName, &providerFavicon, - &m.Name, &displayName, &description, &contextLength, &m.SupportsTools, - &m.SupportsStreaming, &m.SupportsVision, &m.SmartToolRouter, &m.IsVisible, &systemPrompt, &fetchedAt) - - if err == sql.ErrNoRows { - return nil, fmt.Errorf("model not found: %s", modelID) - } - if err != nil { - return nil, fmt.Errorf("failed to query model: %w", err) - } - - if displayName.Valid { - m.DisplayName = displayName.String - } - if description.Valid { - m.Description = description.String - } - if contextLength.Valid { - m.ContextLength = int(contextLength.Int64) - } - if systemPrompt.Valid { - m.SystemPrompt = systemPrompt.String - } - if providerFavicon.Valid { - m.ProviderFavicon = providerFavicon.String - } - if fetchedAt.Valid { - m.FetchedAt = fetchedAt.Time - } - - return &m, nil -} - -// getProviderByID retrieves a provider by ID -func (s *ModelManagementService) getProviderByID(id int) (*models.Provider, error) { - var p models.Provider - var systemPrompt, favicon sql.NullString - err := s.db.QueryRow(` - SELECT id, name, base_url, api_key, enabled, audio_only, system_prompt, favicon, created_at, updated_at - FROM providers - WHERE id = ? - `, id).Scan(&p.ID, &p.Name, &p.BaseURL, &p.APIKey, &p.Enabled, &p.AudioOnly, &systemPrompt, &favicon, &p.CreatedAt, &p.UpdatedAt) - - if err == sql.ErrNoRows { - return nil, fmt.Errorf("provider not found") - } - if err != nil { - return nil, fmt.Errorf("failed to query provider: %w", err) - } - - if systemPrompt.Valid { - p.SystemPrompt = systemPrompt.String - } - if favicon.Valid { - p.Favicon = favicon.String - } - - return &p, nil -} - -// ================== REQUEST/RESPONSE TYPES ================== - -// CreateModelRequest represents a request to create a new model -type CreateModelRequest struct { - ModelID string - ProviderID int - Name string - DisplayName string - Description string - ContextLength int - SupportsTools bool - SupportsStreaming bool - SupportsVision bool - IsVisible bool - SystemPrompt string -} - -// UpdateModelRequest represents a request to update a model -type UpdateModelRequest struct { - DisplayName *string - Description *string - ContextLength *int - SupportsTools *bool - SupportsStreaming *bool - SupportsVision *bool - IsVisible *bool - SystemPrompt *string - SmartToolRouter *bool - FreeTier *bool -} - -// CreateAliasRequest represents a request to create a model alias -type CreateAliasRequest struct { - AliasName string `json:"alias_name"` - ModelID string `json:"model_id"` - ProviderID int `json:"provider_id"` - DisplayName string `json:"display_name"` - Description string `json:"description"` - SupportsVision *bool `json:"supports_vision"` - AgentsEnabled *bool `json:"agents_enabled"` - SmartToolRouter *bool `json:"smart_tool_router"` - FreeTier *bool `json:"free_tier"` - StructuredOutputSupport string `json:"structured_output_support"` - StructuredOutputCompliance *int `json:"structured_output_compliance"` - StructuredOutputWarning string `json:"structured_output_warning"` - StructuredOutputSpeedMs *int `json:"structured_output_speed_ms"` - StructuredOutputBadge string `json:"structured_output_badge"` - MemoryExtractor *bool `json:"memory_extractor"` - MemorySelector *bool `json:"memory_selector"` -} - -// ConnectionTestResult represents the result of a connection test -type ConnectionTestResult struct { - ModelID string - Passed bool - LatencyMs int - Error string -} - -// StructuredOutputBenchmark represents structured output test results -type StructuredOutputBenchmark struct { - CompliancePercentage int `json:"compliance_percentage"` - AverageSpeedMs int `json:"average_speed_ms"` - QualityLevel string `json:"quality_level"` - TestsPassed int `json:"tests_passed"` - TestsFailed int `json:"tests_failed"` -} - -// PerformanceBenchmark represents performance test results -type PerformanceBenchmark struct { - TokensPerSecond float64 `json:"tokens_per_second"` - AvgLatencyMs int `json:"avg_latency_ms"` - TestedAt string `json:"tested_at"` -} - -// BenchmarkResults represents comprehensive benchmark test results -type BenchmarkResults struct { - ConnectionTest *ConnectionTestResult `json:"connection_test,omitempty"` - StructuredOutput *StructuredOutputBenchmark `json:"structured_output,omitempty"` - Performance *PerformanceBenchmark `json:"performance,omitempty"` - LastTested string `json:"last_tested,omitempty"` -} - -// ================== GLOBAL TIER MANAGEMENT ================== - -// TierAssignment represents a model assigned to a global tier -type TierAssignment struct { - ModelID string `json:"model_id"` - ProviderID int `json:"provider_id"` - DisplayName string `json:"display_name"` - Tier string `json:"tier"` -} - -// SetGlobalTier assigns a model to a global tier (tier1-tier5) -// Only one model can occupy each tier slot -func (s *ModelManagementService) SetGlobalTier(modelID string, providerID int, tier string) error { - // Validate tier value - validTiers := map[string]bool{ - "tier1": true, - "tier2": true, - "tier3": true, - "tier4": true, - "tier5": true, - } - - if !validTiers[tier] { - return fmt.Errorf("invalid tier: %s (must be tier1, tier2, tier3, tier4, or tier5)", tier) - } - - // Check if model exists - var displayName string - err := s.db.QueryRow("SELECT display_name FROM models WHERE id = ? AND provider_id = ?", modelID, providerID).Scan(&displayName) - if err == sql.ErrNoRows { - return fmt.Errorf("model not found: %s", modelID) - } - if err != nil { - return fmt.Errorf("failed to verify model: %w", err) - } - - // Use model ID as alias if display name is empty - alias := modelID - if displayName != "" { - alias = displayName - } - - // Try to insert (will fail if tier already occupied due to unique constraint) - _, err = s.db.Exec(` - INSERT INTO recommended_models (provider_id, tier, model_alias) - VALUES (?, ?, ?) - ON DUPLICATE KEY UPDATE - provider_id = VALUES(provider_id), - model_alias = VALUES(model_alias), - updated_at = CURRENT_TIMESTAMP - `, providerID, tier, alias) - - if err != nil { - return fmt.Errorf("failed to set tier: %w", err) - } - - log.Printf("✅ [TIER] Assigned %s to %s", alias, tier) - return nil -} - -// GetGlobalTiers retrieves all 5 tier assignments -func (s *ModelManagementService) GetGlobalTiers() (map[string]*TierAssignment, error) { - rows, err := s.db.Query(` - SELECT r.tier, r.provider_id, r.model_alias, m.id as model_id, m.display_name - FROM recommended_models r - LEFT JOIN models m ON r.provider_id = m.provider_id AND (m.display_name = r.model_alias OR m.id = r.model_alias) - ORDER BY r.tier - `) - if err != nil { - return nil, fmt.Errorf("failed to query tiers: %w", err) - } - defer rows.Close() - - tiers := make(map[string]*TierAssignment) - - for rows.Next() { - var tier, modelAlias, modelID string - var providerID int - var displayName sql.NullString - - err := rows.Scan(&tier, &providerID, &modelAlias, &modelID, &displayName) - if err != nil { - log.Printf("⚠️ Failed to scan tier: %v", err) - continue - } - - assignment := &TierAssignment{ - ModelID: modelID, - ProviderID: providerID, - DisplayName: modelAlias, - Tier: tier, - } - - if displayName.Valid && displayName.String != "" { - assignment.DisplayName = displayName.String - } - - tiers[tier] = assignment - } - - return tiers, nil -} - -// ClearTier removes a model from a tier -func (s *ModelManagementService) ClearTier(tier string) error { - // Validate tier - validTiers := map[string]bool{ - "tier1": true, - "tier2": true, - "tier3": true, - "tier4": true, - "tier5": true, - } - - if !validTiers[tier] { - return fmt.Errorf("invalid tier: %s", tier) - } - - result, err := s.db.Exec("DELETE FROM recommended_models WHERE tier = ?", tier) - if err != nil { - return fmt.Errorf("failed to clear tier: %w", err) - } - - rowsAffected, _ := result.RowsAffected() - if rowsAffected == 0 { - return fmt.Errorf("tier %s is already empty", tier) - } - - log.Printf("✅ [TIER] Cleared %s", tier) - return nil -} - -// BulkUpdateAgentsEnabled updates agents_enabled for multiple models -func (s *ModelManagementService) BulkUpdateAgentsEnabled(modelIDs []string, enabled bool) error { - if len(modelIDs) == 0 { - return fmt.Errorf("no model IDs provided") - } - - // Build placeholders for IN clause - placeholders := make([]string, len(modelIDs)) - args := make([]interface{}, len(modelIDs)+1) - args[0] = enabled - - for i, modelID := range modelIDs { - placeholders[i] = "?" - args[i+1] = modelID - } - - query := fmt.Sprintf(` - UPDATE models - SET agents_enabled = ? - WHERE id IN (%s) - `, strings.Join(placeholders, ",")) - - result, err := s.db.Exec(query, args...) - if err != nil { - return fmt.Errorf("failed to bulk update agents_enabled: %w", err) - } - - rowsAffected, _ := result.RowsAffected() - log.Printf("✅ [BULK] Updated agents_enabled=%v for %d models", enabled, rowsAffected) - return nil -} - -// BulkUpdateVisibility bulk shows/hides models from users -func (s *ModelManagementService) BulkUpdateVisibility(modelIDs []string, visible bool) error { - if len(modelIDs) == 0 { - return fmt.Errorf("no model IDs provided") - } - - // Build placeholders for IN clause - placeholders := make([]string, len(modelIDs)) - args := make([]interface{}, len(modelIDs)+1) - args[0] = visible - - for i, modelID := range modelIDs { - placeholders[i] = "?" - args[i+1] = modelID - } - - query := fmt.Sprintf(` - UPDATE models - SET is_visible = ? - WHERE id IN (%s) - `, strings.Join(placeholders, ",")) - - result, err := s.db.Exec(query, args...) - if err != nil { - return fmt.Errorf("failed to bulk update visibility: %w", err) - } - - rowsAffected, _ := result.RowsAffected() - log.Printf("✅ [BULK] Updated is_visible=%v for %d models", visible, rowsAffected) - return nil -} - -// ================== UTILITY FUNCTIONS ================== - -// joinStrings joins a slice of strings with a separator -func joinStrings(parts []string, sep string) string { - if len(parts) == 0 { - return "" - } - result := parts[0] - for i := 1; i < len(parts); i++ { - result += sep + parts[i] - } - return result -} diff --git a/backend/internal/services/model_service.go b/backend/internal/services/model_service.go deleted file mode 100644 index 283ca7af..00000000 --- a/backend/internal/services/model_service.go +++ /dev/null @@ -1,652 +0,0 @@ -package services - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "database/sql" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "strings" - "time" -) - -// ModelService handles model operations -type ModelService struct { - db *database.DB -} - -// NewModelService creates a new model service -func NewModelService(db *database.DB) *ModelService { - return &ModelService{db: db} -} - -// GetDB returns the underlying database connection -func (s *ModelService) GetDB() *database.DB { - return s.db -} - -// GetAll returns all models, optionally filtered by visibility -// Excludes models from audio-only providers (those are for transcription only) -func (s *ModelService) GetAll(visibleOnly bool) ([]models.Model, error) { - query := ` - SELECT m.id, m.provider_id, p.name as provider_name, p.favicon as provider_favicon, - m.name, m.display_name, m.description, m.context_length, m.supports_tools, - m.supports_streaming, m.supports_vision, m.smart_tool_router, m.is_visible, m.system_prompt, m.fetched_at - FROM models m - JOIN providers p ON m.provider_id = p.id - WHERE (p.audio_only = 0 OR p.audio_only IS NULL) - ` - if visibleOnly { - query += " AND m.is_visible = 1" - } - query += " ORDER BY p.name, m.name" - - rows, err := s.db.Query(query) - if err != nil { - return nil, fmt.Errorf("failed to query models: %w", err) - } - defer rows.Close() - - var modelsList []models.Model - for rows.Next() { - var m models.Model - var displayName, description, systemPrompt, providerFavicon sql.NullString - var contextLength sql.NullInt64 - var fetchedAt sql.NullTime - - err := rows.Scan(&m.ID, &m.ProviderID, &m.ProviderName, &providerFavicon, - &m.Name, &displayName, &description, &contextLength, &m.SupportsTools, - &m.SupportsStreaming, &m.SupportsVision, &m.SmartToolRouter, &m.IsVisible, &systemPrompt, &fetchedAt) - if err != nil { - return nil, fmt.Errorf("failed to scan model: %w", err) - } - - // Handle nullable fields - if displayName.Valid { - m.DisplayName = displayName.String - } - if description.Valid { - m.Description = description.String - } - if contextLength.Valid { - m.ContextLength = int(contextLength.Int64) - } - if systemPrompt.Valid { - m.SystemPrompt = systemPrompt.String - } - if providerFavicon.Valid { - m.ProviderFavicon = providerFavicon.String - } - if fetchedAt.Valid { - m.FetchedAt = fetchedAt.Time - } - - modelsList = append(modelsList, m) - } - - return modelsList, nil -} - -// GetByProvider returns models for a specific provider -func (s *ModelService) GetByProvider(providerID int, visibleOnly bool) ([]models.Model, error) { - query := ` - SELECT m.id, m.provider_id, p.name as provider_name, p.favicon as provider_favicon, - m.name, m.display_name, m.description, m.context_length, m.supports_tools, - m.supports_streaming, m.supports_vision, m.smart_tool_router, m.is_visible, m.system_prompt, m.fetched_at - FROM models m - JOIN providers p ON m.provider_id = p.id - WHERE m.provider_id = ? - ` - if visibleOnly { - query += " AND m.is_visible = 1" - } - query += " ORDER BY m.name" - - rows, err := s.db.Query(query, providerID) - if err != nil { - return nil, fmt.Errorf("failed to query models: %w", err) - } - defer rows.Close() - - var modelsList []models.Model - for rows.Next() { - var m models.Model - var displayName, description, systemPrompt, providerFavicon sql.NullString - var contextLength sql.NullInt64 - var fetchedAt sql.NullTime - - err := rows.Scan(&m.ID, &m.ProviderID, &m.ProviderName, &providerFavicon, - &m.Name, &displayName, &description, &contextLength, &m.SupportsTools, - &m.SupportsStreaming, &m.SupportsVision, &m.SmartToolRouter, &m.IsVisible, &systemPrompt, &fetchedAt) - if err != nil { - return nil, fmt.Errorf("failed to scan model: %w", err) - } - - // Handle nullable fields - if displayName.Valid { - m.DisplayName = displayName.String - } - if description.Valid { - m.Description = description.String - } - if contextLength.Valid { - m.ContextLength = int(contextLength.Int64) - } - if systemPrompt.Valid { - m.SystemPrompt = systemPrompt.String - } - if providerFavicon.Valid { - m.ProviderFavicon = providerFavicon.String - } - if fetchedAt.Valid { - m.FetchedAt = fetchedAt.Time - } - - modelsList = append(modelsList, m) - } - - return modelsList, nil -} - -// GetToolPredictorModels returns only models that can be used as tool predictors -// These are models with smart_tool_router = true and is_visible = true -func (s *ModelService) GetToolPredictorModels() ([]models.Model, error) { - query := ` - SELECT m.id, m.provider_id, p.name as provider_name, p.favicon as provider_favicon, - m.name, m.display_name, m.description, m.context_length, m.supports_tools, - m.supports_streaming, m.supports_vision, m.smart_tool_router, m.is_visible, m.system_prompt, m.fetched_at - FROM models m - JOIN providers p ON m.provider_id = p.id - WHERE m.smart_tool_router = 1 - AND m.is_visible = 1 - AND (p.audio_only = 0 OR p.audio_only IS NULL) - ORDER BY p.name, m.name - ` - - rows, err := s.db.Query(query) - if err != nil { - return nil, fmt.Errorf("failed to query tool predictor models: %w", err) - } - defer rows.Close() - - var modelsList []models.Model - for rows.Next() { - var m models.Model - var displayName, description, systemPrompt, providerFavicon sql.NullString - var contextLength sql.NullInt64 - var fetchedAt sql.NullTime - - err := rows.Scan(&m.ID, &m.ProviderID, &m.ProviderName, &providerFavicon, - &m.Name, &displayName, &description, &contextLength, &m.SupportsTools, - &m.SupportsStreaming, &m.SupportsVision, &m.SmartToolRouter, &m.IsVisible, &systemPrompt, &fetchedAt) - if err != nil { - return nil, fmt.Errorf("failed to scan tool predictor model: %w", err) - } - - // Handle nullable fields - if displayName.Valid { - m.DisplayName = displayName.String - } - if description.Valid { - m.Description = description.String - } - if providerFavicon.Valid { - m.ProviderFavicon = providerFavicon.String - } - if contextLength.Valid { - m.ContextLength = int(contextLength.Int64) - } - if systemPrompt.Valid { - m.SystemPrompt = systemPrompt.String - } - if fetchedAt.Valid { - m.FetchedAt = fetchedAt.Time - } - - modelsList = append(modelsList, m) - } - - return modelsList, nil -} - -// FetchFromProvider fetches models from a provider's API -func (s *ModelService) FetchFromProvider(provider *models.Provider) error { - log.Printf("🔄 Fetching models from provider: %s", provider.Name) - - // Create HTTP request to provider's /v1/models endpoint - req, err := http.NewRequest("GET", provider.BaseURL+"/models", nil) - if err != nil { - return fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Authorization", "Bearer "+provider.APIKey) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return fmt.Errorf("failed to fetch models: %w", err) - } - defer resp.Body.Close() - - if resp.StatusCode != http.StatusOK { - body, _ := io.ReadAll(resp.Body) - return fmt.Errorf("API error (status %d): %s", resp.StatusCode, string(body)) - } - - // Parse response - body, err := io.ReadAll(resp.Body) - if err != nil { - return fmt.Errorf("failed to read response: %w", err) - } - - var modelsResp models.OpenAIModelsResponse - if err := json.Unmarshal(body, &modelsResp); err != nil { - return fmt.Errorf("failed to parse models response: %w", err) - } - - log.Printf("✅ Fetched %d models from %s", len(modelsResp.Data), provider.Name) - - // Store models in database - for _, modelData := range modelsResp.Data { - _, err := s.db.Exec(` - INSERT INTO models (id, provider_id, name, display_name, fetched_at) - VALUES (?, ?, ?, ?, ?) - ON DUPLICATE KEY UPDATE - name = VALUES(name), - display_name = VALUES(display_name), - fetched_at = VALUES(fetched_at) - `, modelData.ID, provider.ID, modelData.ID, modelData.ID, time.Now()) - - if err != nil { - log.Printf("⚠️ Failed to store model %s: %v", modelData.ID, err) - } - } - - // Log refresh - _, err = s.db.Exec(` - INSERT INTO model_refresh_log (provider_id, models_fetched, refreshed_at) - VALUES (?, ?, ?) - `, provider.ID, len(modelsResp.Data), time.Now()) - - if err != nil { - log.Printf("⚠️ Failed to log refresh: %v", err) - } - - log.Printf("✅ Refreshed %d models for provider %s", len(modelsResp.Data), provider.Name) - return nil -} - -// SyncModelAliasMetadata syncs metadata from model aliases to the database -// This updates existing model records with flags like smart_tool_router, agents, supports_vision -func (s *ModelService) SyncModelAliasMetadata(providerID int, aliases map[string]models.ModelAlias) error { - if len(aliases) == 0 { - return nil - } - - log.Printf("🔄 [MODEL-SYNC] Syncing metadata for %d model aliases (provider %d)", len(aliases), providerID) - - for modelID, alias := range aliases { - // Build update statement for fields that are set in the alias - updateParts := []string{} - args := []interface{}{} - - // Smart tool router flag - if alias.SmartToolRouter != nil { - updateParts = append(updateParts, "smart_tool_router = ?") - if *alias.SmartToolRouter { - args = append(args, 1) - } else { - args = append(args, 0) - } - } - - // Free tier flag - if alias.FreeTier != nil { - updateParts = append(updateParts, "free_tier = ?") - if *alias.FreeTier { - args = append(args, 1) - } else { - args = append(args, 0) - } - } - - // Supports vision flag - if alias.SupportsVision != nil { - updateParts = append(updateParts, "supports_vision = ?") - if *alias.SupportsVision { - args = append(args, 1) - } else { - args = append(args, 0) - } - } - - // Display name - if alias.DisplayName != "" { - updateParts = append(updateParts, "display_name = ?") - args = append(args, alias.DisplayName) - } - - // Description - if alias.Description != "" { - updateParts = append(updateParts, "description = ?") - args = append(args, alias.Description) - } - - if len(updateParts) == 0 { - continue // No metadata to sync for this alias - } - - // Add WHERE clause arguments - args = append(args, modelID, providerID) - - query := fmt.Sprintf(` - UPDATE models - SET %s - WHERE id = ? AND provider_id = ? - `, strings.Join(updateParts, ", ")) - - result, err := s.db.Exec(query, args...) - if err != nil { - log.Printf("⚠️ [MODEL-SYNC] Failed to update model %s: %v", modelID, err) - continue - } - - rowsAffected, _ := result.RowsAffected() - if rowsAffected > 0 { - log.Printf(" ✅ Updated model %s: %s", modelID, strings.Join(updateParts, ", ")) - } - } - - log.Printf("✅ [MODEL-SYNC] Model alias metadata sync completed for provider %d", providerID) - return nil -} - -// LoadAllAliasesFromDB loads all model aliases from the database -// Returns map[providerID]map[aliasName]ModelAlias -func (s *ModelService) LoadAllAliasesFromDB() (map[int]map[string]models.ModelAlias, error) { - query := ` - SELECT provider_id, alias_name, model_id, display_name, description, - supports_vision, agents_enabled, smart_tool_router, free_tier, - structured_output_support, structured_output_compliance, - structured_output_warning, structured_output_speed_ms, - structured_output_badge, memory_extractor, memory_selector - FROM model_aliases - ORDER BY provider_id, alias_name - ` - - rows, err := s.db.Query(query) - if err != nil { - return nil, fmt.Errorf("failed to query model aliases: %w", err) - } - defer rows.Close() - - result := make(map[int]map[string]models.ModelAlias) - - for rows.Next() { - var providerID int - var aliasName, modelID, displayName string - var description, structuredOutputSupport, structuredOutputWarning, structuredOutputBadge sql.NullString - var supportsVision, agentsEnabled, smartToolRouter, freeTier, memoryExtractor, memorySelector sql.NullBool - var structuredOutputCompliance, structuredOutputSpeedMs sql.NullInt64 - - err := rows.Scan(&providerID, &aliasName, &modelID, &displayName, &description, - &supportsVision, &agentsEnabled, &smartToolRouter, &freeTier, - &structuredOutputSupport, &structuredOutputCompliance, - &structuredOutputWarning, &structuredOutputSpeedMs, - &structuredOutputBadge, &memoryExtractor, &memorySelector) - if err != nil { - log.Printf("⚠️ Failed to scan alias: %v", err) - continue - } - - // Initialize provider map if not exists - if result[providerID] == nil { - result[providerID] = make(map[string]models.ModelAlias) - } - - // Build ModelAlias struct - alias := models.ModelAlias{ - ActualModel: modelID, - DisplayName: displayName, - } - - if description.Valid { - alias.Description = description.String - } - if supportsVision.Valid { - val := supportsVision.Bool - alias.SupportsVision = &val - } - if agentsEnabled.Valid { - val := agentsEnabled.Bool - alias.Agents = &val - } - if smartToolRouter.Valid { - val := smartToolRouter.Bool - alias.SmartToolRouter = &val - } - if freeTier.Valid { - val := freeTier.Bool - alias.FreeTier = &val - } - if structuredOutputSupport.Valid { - alias.StructuredOutputSupport = structuredOutputSupport.String - } - if structuredOutputCompliance.Valid { - val := int(structuredOutputCompliance.Int64) - alias.StructuredOutputCompliance = &val - } - if structuredOutputWarning.Valid { - alias.StructuredOutputWarning = structuredOutputWarning.String - } - if structuredOutputSpeedMs.Valid { - val := int(structuredOutputSpeedMs.Int64) - alias.StructuredOutputSpeedMs = &val - } - if structuredOutputBadge.Valid { - alias.StructuredOutputBadge = structuredOutputBadge.String - } - if memoryExtractor.Valid { - val := memoryExtractor.Bool - alias.MemoryExtractor = &val - } - if memorySelector.Valid { - val := memorySelector.Bool - alias.MemorySelector = &val - } - - result[providerID][aliasName] = alias - } - - log.Printf("✅ Loaded %d provider alias sets from database", len(result)) - return result, nil -} - -// LoadAllRecommendedModelsFromDB loads all recommended models from the database -// Returns map[providerID]*RecommendedModels -func (s *ModelService) LoadAllRecommendedModelsFromDB() (map[int]*models.RecommendedModels, error) { - query := ` - SELECT provider_id, tier, model_alias - FROM recommended_models - ORDER BY provider_id, tier - ` - - rows, err := s.db.Query(query) - if err != nil { - return nil, fmt.Errorf("failed to query recommended models: %w", err) - } - defer rows.Close() - - result := make(map[int]*models.RecommendedModels) - - for rows.Next() { - var providerID int - var tier, modelAlias string - - err := rows.Scan(&providerID, &tier, &modelAlias) - if err != nil { - log.Printf("⚠️ Failed to scan recommended model: %v", err) - continue - } - - // Initialize provider recommendations if not exists - if result[providerID] == nil { - result[providerID] = &models.RecommendedModels{} - } - - // Set the appropriate tier - switch tier { - case "top": - result[providerID].Top = modelAlias - case "medium": - result[providerID].Medium = modelAlias - case "fastest": - result[providerID].Fastest = modelAlias - case "new": - result[providerID].New = modelAlias - } - } - - log.Printf("✅ Loaded recommended models for %d providers from database", len(result)) - return result, nil -} - -// SaveAliasesToDB saves model aliases to the database -func (s *ModelService) SaveAliasesToDB(providerID int, aliases map[string]models.ModelAlias) error { - if len(aliases) == 0 { - return nil - } - - log.Printf("💾 [MODEL-ALIAS] Saving %d aliases to database for provider %d", len(aliases), providerID) - - for aliasName, alias := range aliases { - _, err := s.db.Exec(` - INSERT INTO model_aliases ( - alias_name, model_id, provider_id, display_name, description, - supports_vision, agents_enabled, smart_tool_router, free_tier, - structured_output_support, structured_output_compliance, - structured_output_warning, structured_output_speed_ms, - structured_output_badge, memory_extractor, memory_selector - ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?) - ON DUPLICATE KEY UPDATE - model_id = VALUES(model_id), - display_name = VALUES(display_name), - description = VALUES(description), - supports_vision = VALUES(supports_vision), - agents_enabled = VALUES(agents_enabled), - smart_tool_router = VALUES(smart_tool_router), - free_tier = VALUES(free_tier), - structured_output_support = VALUES(structured_output_support), - structured_output_compliance = VALUES(structured_output_compliance), - structured_output_warning = VALUES(structured_output_warning), - structured_output_speed_ms = VALUES(structured_output_speed_ms), - structured_output_badge = VALUES(structured_output_badge), - memory_extractor = VALUES(memory_extractor), - memory_selector = VALUES(memory_selector) - `, - aliasName, - alias.ActualModel, - providerID, - alias.DisplayName, - nullString(alias.Description), - nullBool(alias.SupportsVision), - nullBool(alias.Agents), - nullBool(alias.SmartToolRouter), - nullBool(alias.FreeTier), - nullString(alias.StructuredOutputSupport), - nullInt(alias.StructuredOutputCompliance), - nullString(alias.StructuredOutputWarning), - nullInt(alias.StructuredOutputSpeedMs), - nullString(alias.StructuredOutputBadge), - nullBool(alias.MemoryExtractor), - nullBool(alias.MemorySelector), - ) - - if err != nil { - log.Printf("⚠️ [MODEL-ALIAS] Failed to save alias %s: %v", aliasName, err) - continue - } - } - - log.Printf("✅ [MODEL-ALIAS] Saved %d aliases to database for provider %d", len(aliases), providerID) - return nil -} - -// SaveRecommendedModelsToDB saves recommended models to the database -func (s *ModelService) SaveRecommendedModelsToDB(providerID int, recommended *models.RecommendedModels) error { - if recommended == nil { - return nil - } - - log.Printf("💾 [RECOMMENDED] Saving recommended models to database for provider %d", providerID) - - // Delete existing recommendations for this provider - _, err := s.db.Exec("DELETE FROM recommended_models WHERE provider_id = ?", providerID) - if err != nil { - return fmt.Errorf("failed to delete old recommendations: %w", err) - } - - // Insert new recommendations - tiers := map[string]string{ - "top": recommended.Top, - "medium": recommended.Medium, - "fastest": recommended.Fastest, - "new": recommended.New, - } - - for tier, modelAlias := range tiers { - if modelAlias == "" { - continue - } - - _, err := s.db.Exec(` - INSERT INTO recommended_models (provider_id, tier, model_alias) - VALUES (?, ?, ?) - `, providerID, tier, modelAlias) - - if err != nil { - log.Printf("⚠️ [RECOMMENDED] Failed to save %s tier: %v", tier, err) - } - } - - log.Printf("✅ [RECOMMENDED] Saved recommended models for provider %d", providerID) - return nil -} - -// Helper functions for nullable values -func nullString(s string) interface{} { - if s == "" { - return nil - } - return s -} - -func nullInt(i *int) interface{} { - if i == nil { - return nil - } - return *i -} - -func nullBool(b *bool) interface{} { - if b == nil { - return nil - } - return *b -} - -// IsFreeTier checks if a model is marked as free tier -func (s *ModelService) IsFreeTier(modelID string) bool { - var isFreeTier int - err := s.db.QueryRow(` - SELECT COALESCE(free_tier, 0) - FROM models - WHERE id = ? - `, modelID).Scan(&isFreeTier) - - return err == nil && isFreeTier == 1 -} diff --git a/backend/internal/services/model_service_test.go b/backend/internal/services/model_service_test.go deleted file mode 100644 index 4f81ac91..00000000 --- a/backend/internal/services/model_service_test.go +++ /dev/null @@ -1,466 +0,0 @@ -package services - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "encoding/json" - "net/http" - "net/http/httptest" - "os" - "testing" - "time" -) - -func setupTestDBForModels(t *testing.T) (*database.DB, func()) { - t.Skip("SQLite tests are deprecated - please use DATABASE_URL with MySQL DSN") - tmpFile := "test_model_service.db" - db, err := database.New(tmpFile) - if err != nil { - t.Fatalf("Failed to create test database: %v", err) - } - - if err := db.Initialize(); err != nil { - t.Fatalf("Failed to initialize test database: %v", err) - } - - cleanup := func() { - db.Close() - os.Remove(tmpFile) - } - - return db, cleanup -} - -func createTestProvider(t *testing.T, db *database.DB, name string) *models.Provider { - providerService := NewProviderService(db) - config := models.ProviderConfig{ - Name: name, - BaseURL: "https://api.test.com/v1", - APIKey: "test-key", - Enabled: true, - } - - provider, err := providerService.Create(config) - if err != nil { - t.Fatalf("Failed to create test provider: %v", err) - } - - return provider -} - -func insertTestModel(t *testing.T, db *database.DB, model *models.Model) { - _, err := db.Exec(` - INSERT OR REPLACE INTO models - (id, provider_id, name, display_name, description, context_length, - supports_tools, supports_streaming, supports_vision, is_visible, fetched_at) - VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?) - `, model.ID, model.ProviderID, model.Name, model.DisplayName, model.Description, - model.ContextLength, model.SupportsTools, model.SupportsStreaming, - model.SupportsVision, model.IsVisible, time.Now()) - - if err != nil { - t.Fatalf("Failed to insert test model: %v", err) - } -} - -func TestNewModelService(t *testing.T) { - db, cleanup := setupTestDBForModels(t) - defer cleanup() - - service := NewModelService(db) - if service == nil { - t.Fatal("Expected non-nil model service") - } -} - -func TestModelService_GetAll(t *testing.T) { - db, cleanup := setupTestDBForModels(t) - defer cleanup() - - service := NewModelService(db) - provider := createTestProvider(t, db, "Test Provider") - - // Create test models - testModels := []models.Model{ - { - ID: "model-1", - ProviderID: provider.ID, - Name: "Model 1", - IsVisible: true, - SupportsStreaming: true, - }, - { - ID: "model-2", - ProviderID: provider.ID, - Name: "Model 2", - IsVisible: false, - SupportsStreaming: true, - }, - { - ID: "model-3", - ProviderID: provider.ID, - Name: "Model 3", - IsVisible: true, - SupportsTools: true, - }, - } - - for i := range testModels { - insertTestModel(t, db, &testModels[i]) - } - - // Get all models (including hidden) - allModels, err := service.GetAll(false) - if err != nil { - t.Fatalf("Failed to get all models: %v", err) - } - - if len(allModels) != 3 { - t.Errorf("Expected 3 models, got %d", len(allModels)) - } - - // Get only visible models - visibleModels, err := service.GetAll(true) - if err != nil { - t.Fatalf("Failed to get visible models: %v", err) - } - - if len(visibleModels) != 2 { - t.Errorf("Expected 2 visible models, got %d", len(visibleModels)) - } -} - -func TestModelService_GetByProvider(t *testing.T) { - db, cleanup := setupTestDBForModels(t) - defer cleanup() - - service := NewModelService(db) - provider1 := createTestProvider(t, db, "Provider 1") - provider2 := createTestProvider(t, db, "Provider 2") - - // Create models for both providers - testModels := []models.Model{ - {ID: "model-1", ProviderID: provider1.ID, Name: "Model 1", IsVisible: true}, - {ID: "model-2", ProviderID: provider1.ID, Name: "Model 2", IsVisible: true}, - {ID: "model-3", ProviderID: provider2.ID, Name: "Model 3", IsVisible: true}, - } - - for i := range testModels { - insertTestModel(t, db, &testModels[i]) - } - - // Get models for provider 1 - provider1Models, err := service.GetByProvider(provider1.ID, false) - if err != nil { - t.Fatalf("Failed to get provider 1 models: %v", err) - } - - if len(provider1Models) != 2 { - t.Errorf("Expected 2 models for provider 1, got %d", len(provider1Models)) - } - - // Get models for provider 2 - provider2Models, err := service.GetByProvider(provider2.ID, false) - if err != nil { - t.Fatalf("Failed to get provider 2 models: %v", err) - } - - if len(provider2Models) != 1 { - t.Errorf("Expected 1 model for provider 2, got %d", len(provider2Models)) - } -} - -func TestModelService_GetByProvider_VisibleOnly(t *testing.T) { - db, cleanup := setupTestDBForModels(t) - defer cleanup() - - service := NewModelService(db) - provider := createTestProvider(t, db, "Test Provider") - - // Create test models with different visibility - testModels := []models.Model{ - {ID: "model-1", ProviderID: provider.ID, Name: "Model 1", IsVisible: true}, - {ID: "model-2", ProviderID: provider.ID, Name: "Model 2", IsVisible: false}, - {ID: "model-3", ProviderID: provider.ID, Name: "Model 3", IsVisible: true}, - } - - for i := range testModels { - insertTestModel(t, db, &testModels[i]) - } - - // Get only visible models - visibleModels, err := service.GetByProvider(provider.ID, true) - if err != nil { - t.Fatalf("Failed to get visible models: %v", err) - } - - if len(visibleModels) != 2 { - t.Errorf("Expected 2 visible models, got %d", len(visibleModels)) - } -} - -func TestModelService_FetchFromProvider(t *testing.T) { - db, cleanup := setupTestDBForModels(t) - defer cleanup() - - // Create mock server - mockResponse := models.OpenAIModelsResponse{ - Object: "list", - Data: []struct { - ID string `json:"id"` - Object string `json:"object"` - Created int64 `json:"created"` - OwnedBy string `json:"owned_by"` - }{ - {ID: "gpt-4", Object: "model", Created: 1234567890, OwnedBy: "openai"}, - {ID: "gpt-3.5-turbo", Object: "model", Created: 1234567891, OwnedBy: "openai"}, - }, - } - - server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { - // Verify request - if r.URL.Path != "/models" { - t.Errorf("Expected path /models, got %s", r.URL.Path) - } - - authHeader := r.Header.Get("Authorization") - if authHeader != "Bearer test-key" { - t.Errorf("Expected Authorization header 'Bearer test-key', got %s", authHeader) - } - - // Return mock response - w.Header().Set("Content-Type", "application/json") - json.NewEncoder(w).Encode(mockResponse) - })) - defer server.Close() - - service := NewModelService(db) - providerService := NewProviderService(db) - - // Create provider with mock server URL - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: server.URL, - APIKey: "test-key", - Enabled: true, - } - - provider, err := providerService.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - // Fetch models from provider - if err := service.FetchFromProvider(provider); err != nil { - t.Fatalf("Failed to fetch models from provider: %v", err) - } - - // Verify models were stored - models, err := service.GetByProvider(provider.ID, false) - if err != nil { - t.Fatalf("Failed to get models: %v", err) - } - - if len(models) != 2 { - t.Errorf("Expected 2 models, got %d", len(models)) - } - - // Verify model data - if models[0].ID != "gpt-3.5-turbo" && models[1].ID != "gpt-3.5-turbo" { - t.Error("Expected to find gpt-3.5-turbo model") - } - - if models[0].ID != "gpt-4" && models[1].ID != "gpt-4" { - t.Error("Expected to find gpt-4 model") - } -} - -func TestModelService_FetchFromProvider_InvalidAuth(t *testing.T) { - db, cleanup := setupTestDBForModels(t) - defer cleanup() - - // Create mock server that returns 401 - server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { - w.WriteHeader(http.StatusUnauthorized) - w.Write([]byte(`{"error":"invalid_api_key"}`)) - })) - defer server.Close() - - service := NewModelService(db) - providerService := NewProviderService(db) - - // Create provider with mock server URL - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: server.URL, - APIKey: "invalid-key", - Enabled: true, - } - - provider, err := providerService.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - // Fetch should fail - err = service.FetchFromProvider(provider) - if err == nil { - t.Error("Expected error for invalid API key, got nil") - } -} - -func TestModelService_FetchFromProvider_InvalidJSON(t *testing.T) { - db, cleanup := setupTestDBForModels(t) - defer cleanup() - - // Create mock server that returns invalid JSON - server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { - w.Header().Set("Content-Type", "application/json") - w.Write([]byte(`{invalid json}`)) - })) - defer server.Close() - - service := NewModelService(db) - providerService := NewProviderService(db) - - // Create provider with mock server URL - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: server.URL, - APIKey: "test-key", - Enabled: true, - } - - provider, err := providerService.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - // Fetch should fail - err = service.FetchFromProvider(provider) - if err == nil { - t.Error("Expected error for invalid JSON, got nil") - } -} - -func TestModelService_FetchFromProvider_EmptyResponse(t *testing.T) { - db, cleanup := setupTestDBForModels(t) - defer cleanup() - - // Create mock server that returns empty models list - mockResponse := models.OpenAIModelsResponse{ - Object: "list", - Data: []struct { - ID string `json:"id"` - Object string `json:"object"` - Created int64 `json:"created"` - OwnedBy string `json:"owned_by"` - }{}, - } - - server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { - w.Header().Set("Content-Type", "application/json") - json.NewEncoder(w).Encode(mockResponse) - })) - defer server.Close() - - service := NewModelService(db) - providerService := NewProviderService(db) - - // Create provider with mock server URL - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: server.URL, - APIKey: "test-key", - Enabled: true, - } - - provider, err := providerService.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - // Fetch should succeed but return no models - if err := service.FetchFromProvider(provider); err != nil { - t.Fatalf("Failed to fetch models: %v", err) - } - - // Verify no models were stored - models, err := service.GetByProvider(provider.ID, false) - if err != nil { - t.Fatalf("Failed to get models: %v", err) - } - - if len(models) != 0 { - t.Errorf("Expected 0 models, got %d", len(models)) - } -} - -func TestModelService_FetchFromProvider_UpdateExisting(t *testing.T) { - db, cleanup := setupTestDBForModels(t) - defer cleanup() - - // Create mock server - mockResponse := models.OpenAIModelsResponse{ - Object: "list", - Data: []struct { - ID string `json:"id"` - Object string `json:"object"` - Created int64 `json:"created"` - OwnedBy string `json:"owned_by"` - }{ - {ID: "gpt-4", Object: "model", Created: 1234567890, OwnedBy: "openai"}, - }, - } - - server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { - w.Header().Set("Content-Type", "application/json") - json.NewEncoder(w).Encode(mockResponse) - })) - defer server.Close() - - service := NewModelService(db) - providerService := NewProviderService(db) - - // Create provider - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: server.URL, - APIKey: "test-key", - Enabled: true, - } - - provider, err := providerService.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - // Insert existing model with different display name - existingModel := models.Model{ - ID: "gpt-4", - ProviderID: provider.ID, - Name: "gpt-4", - DisplayName: "Old GPT-4", - IsVisible: true, - } - insertTestModel(t, db, &existingModel) - - // Fetch models - should update existing - if err := service.FetchFromProvider(provider); err != nil { - t.Fatalf("Failed to fetch models: %v", err) - } - - // Verify model was updated (count should still be 1) - models, err := service.GetByProvider(provider.ID, false) - if err != nil { - t.Fatalf("Failed to get models: %v", err) - } - - if len(models) != 1 { - t.Errorf("Expected 1 model, got %d", len(models)) - } - - if models[0].ID != "gpt-4" { - t.Errorf("Expected model ID 'gpt-4', got %s", models[0].ID) - } -} diff --git a/backend/internal/services/payment_service.go b/backend/internal/services/payment_service.go deleted file mode 100644 index 9be570bc..00000000 --- a/backend/internal/services/payment_service.go +++ /dev/null @@ -1,1564 +0,0 @@ -package services - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "context" - "crypto/hmac" - "crypto/sha256" - "encoding/hex" - "encoding/json" - "fmt" - "log" - "net/http" - "os" - "strings" - "time" - - "github.com/dodopayments/dodopayments-go" - "github.com/dodopayments/dodopayments-go/option" - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/bson/primitive" - "go.mongodb.org/mongo-driver/mongo" - "go.mongodb.org/mongo-driver/mongo/options" -) - -// WebhookEvent represents a webhook event from DodoPayments -type WebhookEvent struct { - ID string `json:"id"` - Type string `json:"type"` - Data map[string]interface{} `json:"data"` -} - -// PaymentService handles payment and subscription operations -type PaymentService struct { - client *dodopayments.Client - webhookSecret string - mongoDB *database.MongoDB - userService *UserService - tierService *TierService - usageLimiter *UsageLimiterService - - subscriptions *mongo.Collection - events *mongo.Collection -} - -// NewPaymentService creates a new payment service -func NewPaymentService( - apiKey, webhookSecret, businessID string, - mongoDB *database.MongoDB, - userService *UserService, - tierService *TierService, - usageLimiter *UsageLimiterService, -) *PaymentService { - var client *dodopayments.Client - if apiKey != "" { - // Determine environment mode - env := os.Getenv("DODO_ENVIRONMENT") - var envOpt option.RequestOption - if env == "test" { - envOpt = option.WithEnvironmentTestMode() - } else { - envOpt = option.WithEnvironmentLiveMode() - } - - // Initialize DodoPayments client - client = dodopayments.NewClient( - option.WithBearerToken(apiKey), - envOpt, - ) - log.Println("✅ DodoPayments client initialized") - } else { - log.Println("⚠️ DodoPayments API key not provided, payment features disabled") - } - - var subscriptions *mongo.Collection - var events *mongo.Collection - if mongoDB != nil { - subscriptions = mongoDB.Database().Collection("subscriptions") - events = mongoDB.Database().Collection("subscription_events") - } - - return &PaymentService{ - client: client, - webhookSecret: webhookSecret, - mongoDB: mongoDB, - userService: userService, - tierService: tierService, - usageLimiter: usageLimiter, - subscriptions: subscriptions, - events: events, - } -} - -// CheckoutResponse represents the response for checkout creation -type CheckoutResponse struct { - CheckoutURL string `json:"checkout_url"` - SessionID string `json:"session_id"` -} - -// CreateCheckoutSession creates a checkout session for a subscription -func (s *PaymentService) CreateCheckoutSession(ctx context.Context, userID, userEmail, planID string) (*CheckoutResponse, error) { - plan := models.GetPlanByID(planID) - if plan == nil { - return nil, fmt.Errorf("invalid plan ID: %s", planID) - } - - if plan.Tier == models.TierFree { - return nil, fmt.Errorf("cannot create checkout for free plan") - } - - if plan.ContactSales { - return nil, fmt.Errorf("enterprise plan requires contact sales") - } - - if plan.DodoProductID == "" { - return nil, fmt.Errorf("plan %s does not have a DodoPayments product ID configured", planID) - } - - // Get or create user in MongoDB (sync from Supabase if new) - user, err := s.userService.GetUserBySupabaseID(ctx, userID) - if err != nil { - // User doesn't exist in MongoDB, sync them from Supabase - if userEmail == "" { - return nil, fmt.Errorf("failed to get user and no email provided for sync") - } - log.Printf("📝 Syncing new user %s (%s) to MongoDB", userID, userEmail) - user, err = s.userService.SyncUserFromSupabase(ctx, userID, userEmail) - if err != nil { - return nil, fmt.Errorf("failed to sync user: %w", err) - } - } - - customerID := user.DodoCustomerID - if customerID == "" { - // Create customer in DodoPayments - if s.client == nil { - return nil, fmt.Errorf("DodoPayments client not initialized") - } - - // Generate a customer name from email (DodoPayments requires a name) - // Use the part before @ as the name, or full email if no @ - customerName := user.Email - if atIndex := strings.Index(user.Email, "@"); atIndex > 0 { - customerName = user.Email[:atIndex] - } - - customer, err := s.client.Customers.New(ctx, dodopayments.CustomerNewParams{ - Email: dodopayments.F(user.Email), - Name: dodopayments.F(customerName), - Metadata: dodopayments.F(map[string]string{ - "supabase_user_id": userID, - }), - }) - if err != nil { - return nil, fmt.Errorf("failed to create customer: %w", err) - } - - customerID = customer.CustomerID - if err := s.updateUserDodoCustomer(ctx, userID, customerID); err != nil { - return nil, fmt.Errorf("failed to update customer ID: %w", err) - } - } - - if s.client == nil { - return nil, fmt.Errorf("DodoPayments client not initialized") - } - - // Create checkout session using the SDK - Link to our existing customer! - session, err := s.client.CheckoutSessions.New(ctx, dodopayments.CheckoutSessionNewParams{ - CheckoutSessionRequest: dodopayments.CheckoutSessionRequestParam{ - ProductCart: dodopayments.F([]dodopayments.CheckoutSessionRequestProductCartParam{{ - ProductID: dodopayments.F(plan.DodoProductID), - Quantity: dodopayments.F(int64(1)), - }}), - ReturnURL: dodopayments.F(fmt.Sprintf("%s/settings?tab=billing&checkout=success", getBaseURL())), - // Attach the checkout to our existing customer - Customer: dodopayments.F[dodopayments.CustomerRequestUnionParam](dodopayments.AttachExistingCustomerParam{ - CustomerID: dodopayments.F(customerID), - }), - }, - }) - if err != nil { - return nil, fmt.Errorf("failed to create checkout session: %w", err) - } - - return &CheckoutResponse{ - CheckoutURL: session.CheckoutURL, - SessionID: session.SessionID, - }, nil -} - -// GetCurrentSubscription gets the user's current subscription -func (s *PaymentService) GetCurrentSubscription(ctx context.Context, userID string) (*models.Subscription, error) { - // First, check if user has a promo tier in the users collection - if s.userService != nil { - user, err := s.userService.GetUserBySupabaseID(ctx, userID) - if err == nil && user != nil && user.SubscriptionTier != "" { - // User has a tier set (either from promo or previous subscription) - sub := &models.Subscription{ - UserID: userID, - Tier: user.SubscriptionTier, - Status: user.SubscriptionStatus, - CancelAtPeriodEnd: false, - } - if user.SubscriptionExpiresAt != nil { - sub.CurrentPeriodEnd = *user.SubscriptionExpiresAt - } - return sub, nil - } - } - - if s.subscriptions == nil { - // Return default free tier subscription - return &models.Subscription{ - UserID: userID, - Tier: models.TierFree, - Status: models.SubStatusActive, - }, nil - } - - var sub models.Subscription - err := s.subscriptions.FindOne(ctx, bson.M{"userId": userID}).Decode(&sub) - if err == mongo.ErrNoDocuments { - // No subscription found, return free tier - return &models.Subscription{ - UserID: userID, - Tier: models.TierFree, - Status: models.SubStatusActive, - }, nil - } - if err != nil { - return nil, fmt.Errorf("failed to get subscription: %w", err) - } - - return &sub, nil -} - -// PlanChangeResult represents the result of a plan change -type PlanChangeResult struct { - Type string `json:"type"` // "upgrade" or "downgrade" - Immediate bool `json:"immediate"` // true for upgrades, false for downgrades - NewTier string `json:"new_tier"` - EffectiveAt time.Time `json:"effective_at,omitempty"` -} - -// ChangePlan handles both upgrades and downgrades -func (s *PaymentService) ChangePlan(ctx context.Context, userID, newPlanID string) (*PlanChangeResult, error) { - current, err := s.GetCurrentSubscription(ctx, userID) - if err != nil { - return nil, err - } - - newPlan := models.GetPlanByID(newPlanID) - if newPlan == nil { - return nil, fmt.Errorf("invalid plan ID: %s", newPlanID) - } - - currentPlan := models.GetPlanByTier(current.Tier) - if currentPlan == nil { - currentPlan = models.GetPlanByTier(models.TierFree) - } - - // Determine if upgrade or downgrade - comparison := models.CompareTiers(current.Tier, newPlan.Tier) - isUpgrade := comparison < 0 - - if comparison == 0 { - return nil, fmt.Errorf("user is already on %s plan", newPlan.Tier) - } - - if isUpgrade { - // UPGRADE: Immediate with proration - if current.DodoSubscriptionID == "" { - return nil, fmt.Errorf("no active subscription to upgrade") - } - - if s.client == nil { - return nil, fmt.Errorf("DodoPayments client not initialized") - } - - // Change plan using DodoPayments SDK (handles proration automatically) - err = s.client.Subscriptions.ChangePlan(ctx, current.DodoSubscriptionID, dodopayments.SubscriptionChangePlanParams{ - ProductID: dodopayments.F(newPlan.DodoProductID), - ProrationBillingMode: dodopayments.F(dodopayments.SubscriptionChangePlanParamsProrationBillingModeProratedImmediately), - Quantity: dodopayments.F(int64(1)), - }) - if err != nil { - return nil, fmt.Errorf("failed to change plan: %w", err) - } - - // Log the change for audit - log.Printf("✅ Plan changed to %s for subscription %s", newPlan.DodoProductID, current.DodoSubscriptionID) - - // Update subscription in our DB (webhook will confirm, but optimistic update) - now := time.Now() - update := bson.M{ - "$set": bson.M{ - "tier": newPlan.Tier, - "status": models.SubStatusActive, - "scheduledTier": "", - "scheduledChangeAt": nil, - "cancelAtPeriodEnd": false, - "updatedAt": now, - }, - } - - if s.subscriptions != nil { - _, err = s.subscriptions.UpdateOne(ctx, bson.M{"userId": userID}, update) - if err != nil { - log.Printf("⚠️ Failed to update subscription optimistically: %v", err) - } - } - - // Invalidate tier cache - if s.tierService != nil { - s.tierService.InvalidateCache(userID) - } - - return &PlanChangeResult{ - Type: "upgrade", - Immediate: true, - NewTier: newPlan.Tier, - }, nil - } else { - // DOWNGRADE: Schedule for end of period - if current.DodoSubscriptionID == "" { - // No active subscription, just update tier - now := time.Now() - if s.subscriptions != nil { - _, err = s.subscriptions.UpdateOne(ctx, bson.M{"userId": userID}, bson.M{ - "$set": bson.M{ - "tier": newPlan.Tier, - "status": models.SubStatusActive, - "updatedAt": now, - }, - }) - } - if s.tierService != nil { - s.tierService.InvalidateCache(userID) - } - return &PlanChangeResult{ - Type: "downgrade", - Immediate: true, - NewTier: newPlan.Tier, - }, nil - } - - // Schedule downgrade for end of period - periodEnd := current.CurrentPeriodEnd - if periodEnd.IsZero() { - periodEnd = time.Now().Add(30 * 24 * time.Hour) // Default to 30 days - } - - update := bson.M{ - "$set": bson.M{ - "scheduledTier": newPlan.Tier, - "scheduledChangeAt": periodEnd, - "updatedAt": time.Now(), - }, - } - - if s.subscriptions != nil { - _, err = s.subscriptions.UpdateOne(ctx, bson.M{"userId": userID}, update) - if err != nil { - return nil, fmt.Errorf("failed to schedule downgrade: %w", err) - } - } - - return &PlanChangeResult{ - Type: "downgrade", - Immediate: false, - NewTier: newPlan.Tier, - EffectiveAt: periodEnd, - }, nil - } -} - -// PreviewPlanChange shows what will happen before confirming -type PlanChangePreview struct { - ChangeType string `json:"change_type"` // "upgrade" or "downgrade" - Immediate bool `json:"immediate"` - CurrentTier string `json:"current_tier"` - NewTier string `json:"new_tier"` - ProratedAmount int64 `json:"prorated_amount,omitempty"` // cents - EffectiveAt time.Time `json:"effective_at,omitempty"` -} - -// PreviewPlanChange previews a plan change -func (s *PaymentService) PreviewPlanChange(ctx context.Context, userID, newPlanID string) (*PlanChangePreview, error) { - current, err := s.GetCurrentSubscription(ctx, userID) - if err != nil { - return nil, err - } - - newPlan := models.GetPlanByID(newPlanID) - if newPlan == nil { - return nil, fmt.Errorf("invalid plan ID: %s", newPlanID) - } - - currentPlan := models.GetPlanByTier(current.Tier) - if currentPlan == nil { - currentPlan = models.GetPlanByTier(models.TierFree) - } - - comparison := models.CompareTiers(current.Tier, newPlan.Tier) - isUpgrade := comparison < 0 - - if comparison == 0 { - return nil, fmt.Errorf("user is already on %s plan", newPlan.Tier) - } - - preview := &PlanChangePreview{ - CurrentTier: current.Tier, - NewTier: newPlan.Tier, - } - - if isUpgrade { - preview.ChangeType = "upgrade" - preview.Immediate = true - preview.EffectiveAt = time.Now() - - // Calculate proration if we have period info - if !current.CurrentPeriodEnd.IsZero() && !current.CurrentPeriodStart.IsZero() { - daysRemaining := int(time.Until(current.CurrentPeriodEnd).Hours() / 24) - totalDays := int(current.CurrentPeriodEnd.Sub(current.CurrentPeriodStart).Hours() / 24) - if daysRemaining > 0 && totalDays > 0 { - preview.ProratedAmount = s.CalculateProration( - currentPlan.PriceMonthly, - newPlan.PriceMonthly, - daysRemaining, - totalDays, - ) - } - } - } else { - preview.ChangeType = "downgrade" - preview.Immediate = false - preview.EffectiveAt = current.CurrentPeriodEnd - if preview.EffectiveAt.IsZero() { - preview.EffectiveAt = time.Now().Add(30 * 24 * time.Hour) - } - } - - return preview, nil -} - -// CalculateProration calculates prorated charge for plan change -func (s *PaymentService) CalculateProration(fromPrice, toPrice int64, daysRemaining, totalDays int) int64 { - if daysRemaining <= 0 || totalDays <= 0 { - return 0 - } - - // Calculate daily rates - fromDaily := float64(fromPrice) / float64(totalDays) - toDaily := float64(toPrice) / float64(totalDays) - - // Calculate difference for remaining days - difference := (toDaily - fromDaily) * float64(daysRemaining) - - return int64(difference) -} - -// CancelSubscription schedules cancellation at period end -func (s *PaymentService) CancelSubscription(ctx context.Context, userID string) error { - current, err := s.GetCurrentSubscription(ctx, userID) - if err != nil { - return err - } - - if current.Tier == models.TierFree { - return fmt.Errorf("no active subscription to cancel") - } - - if current.CancelAtPeriodEnd { - return fmt.Errorf("subscription is already scheduled for cancellation") - } - - if current.DodoSubscriptionID != "" && s.client != nil { - // Cancel in DodoPayments (cancel at next billing date) - _, err = s.client.Subscriptions.Update(ctx, current.DodoSubscriptionID, dodopayments.SubscriptionUpdateParams{ - CancelAtNextBillingDate: dodopayments.F(true), - }) - if err != nil { - return fmt.Errorf("failed to cancel subscription: %w", err) - } - } - - // Update in our DB - update := bson.M{ - "$set": bson.M{ - "cancelAtPeriodEnd": true, - "status": models.SubStatusPendingCancel, - "updatedAt": time.Now(), - }, - } - - if s.subscriptions != nil { - _, err = s.subscriptions.UpdateOne(ctx, bson.M{"userId": userID}, update) - if err != nil { - return fmt.Errorf("failed to update subscription: %w", err) - } - } - - return nil -} - -// ReactivateSubscription undoes cancellation if still in period -func (s *PaymentService) ReactivateSubscription(ctx context.Context, userID string) error { - current, err := s.GetCurrentSubscription(ctx, userID) - if err != nil { - return err - } - - if !current.CancelAtPeriodEnd { - return fmt.Errorf("subscription is not scheduled for cancellation") - } - - if current.DodoSubscriptionID != "" && s.client != nil { - // Reactivate in DodoPayments (clear cancel_at_next_billing_date) - _, err = s.client.Subscriptions.Update(ctx, current.DodoSubscriptionID, dodopayments.SubscriptionUpdateParams{ - CancelAtNextBillingDate: dodopayments.F(false), - }) - if err != nil { - return fmt.Errorf("failed to reactivate subscription: %w", err) - } - } - - // Update in our DB - update := bson.M{ - "$set": bson.M{ - "cancelAtPeriodEnd": false, - "status": models.SubStatusActive, - "updatedAt": time.Now(), - }, - } - - if s.subscriptions != nil { - _, err = s.subscriptions.UpdateOne(ctx, bson.M{"userId": userID}, update) - if err != nil { - return fmt.Errorf("failed to update subscription: %w", err) - } - } - - return nil -} - -// GetCustomerPortalURL gets the DodoPayments customer portal URL -func (s *PaymentService) GetCustomerPortalURL(ctx context.Context, userID string) (string, error) { - user, err := s.userService.GetUserBySupabaseID(ctx, userID) - if err != nil { - return "", fmt.Errorf("failed to get user: %w", err) - } - - if user.DodoCustomerID == "" { - return "", fmt.Errorf("user does not have a DodoPayments customer ID") - } - - if s.client == nil { - return "", fmt.Errorf("DodoPayments client not initialized") - } - - // Create a customer portal session using DodoPayments SDK - portalSession, err := s.client.Customers.CustomerPortal.New(ctx, user.DodoCustomerID, dodopayments.CustomerCustomerPortalNewParams{}) - if err != nil { - return "", fmt.Errorf("failed to create customer portal session: %w", err) - } - - return portalSession.Link, nil -} - -// GetAvailablePlans returns all available plans -func (s *PaymentService) GetAvailablePlans() []models.Plan { - return models.GetAvailablePlans() -} - -// DetermineChangeType determines if a plan change is an upgrade or downgrade -func (s *PaymentService) DetermineChangeType(fromTier, toTier string) (isUpgrade, isDowngrade bool) { - comparison := models.CompareTiers(fromTier, toTier) - isUpgrade = comparison < 0 - isDowngrade = comparison > 0 - return -} - -// VerifyWebhook verifies webhook signature (legacy method for tests) -func (s *PaymentService) VerifyWebhook(payload []byte, signature string) error { - if s.webhookSecret == "" { - return fmt.Errorf("webhook secret not configured") - } - - mac := hmac.New(sha256.New, []byte(s.webhookSecret)) - mac.Write(payload) - expectedSig := hex.EncodeToString(mac.Sum(nil)) - - if signature != expectedSig { - return fmt.Errorf("invalid webhook signature") - } - - return nil -} - -// VerifyAndParseWebhook verifies and parses webhook using DodoPayments SDK -// DodoPayments uses Standard Webhooks format with headers: -// - webhook-id: unique message ID -// - webhook-signature: v1, -// - webhook-timestamp: unix timestamp -func (s *PaymentService) VerifyAndParseWebhook(payload []byte, headers http.Header) (*WebhookEvent, error) { - // If SDK client is available, use it for verification - if s.client != nil && s.webhookSecret != "" { - event, err := s.client.Webhooks.Unwrap(payload, headers, option.WithWebhookKey(s.webhookSecret)) - if err != nil { - return nil, fmt.Errorf("webhook verification failed: %w", err) - } - - // Successfully verified with SDK - convert and return - return s.convertSDKEventToWebhookEvent(event) - } - - // Fallback: Use legacy HMAC verification for tests or when SDK is not available - signature := headers.Get("Webhook-Signature") - if signature == "" { - signature = headers.Get("Dodo-Signature") - } - if signature == "" { - return nil, fmt.Errorf("missing webhook signature header") - } - - if err := s.VerifyWebhook(payload, signature); err != nil { - return nil, err - } - - // Parse the payload - var event WebhookEvent - if err := json.Unmarshal(payload, &event); err != nil { - return nil, fmt.Errorf("failed to parse webhook payload: %w", err) - } - - return &event, nil -} - -// convertSDKEventToWebhookEvent converts SDK event to internal WebhookEvent -func (s *PaymentService) convertSDKEventToWebhookEvent(event *dodopayments.UnwrapWebhookEvent) (*WebhookEvent, error) { - // Convert SDK event to our internal WebhookEvent format - webhookEvent := &WebhookEvent{ - Type: string(event.Type), - } - - // Extract event ID and data based on event type - // The Data field embeds the subscription/payment struct directly - switch e := event.AsUnion().(type) { - case dodopayments.SubscriptionActiveWebhookEvent: - webhookEvent.ID = e.Data.SubscriptionID - webhookEvent.Data = map[string]interface{}{ - "subscription_id": e.Data.SubscriptionID, - "customer_id": e.Data.Customer.CustomerID, - "product_id": e.Data.ProductID, - "current_period_start": e.Data.PreviousBillingDate.Format(time.RFC3339), - "current_period_end": e.Data.NextBillingDate.Format(time.RFC3339), - } - case dodopayments.SubscriptionUpdatedWebhookEvent: - webhookEvent.ID = e.Data.SubscriptionID - webhookEvent.Data = map[string]interface{}{ - "subscription_id": e.Data.SubscriptionID, - "customer_id": e.Data.Customer.CustomerID, - "product_id": e.Data.ProductID, - "current_period_start": e.Data.PreviousBillingDate.Format(time.RFC3339), - "current_period_end": e.Data.NextBillingDate.Format(time.RFC3339), - } - case dodopayments.SubscriptionCancelledWebhookEvent: - webhookEvent.ID = e.Data.SubscriptionID - webhookEvent.Data = map[string]interface{}{ - "subscription_id": e.Data.SubscriptionID, - "customer_id": e.Data.Customer.CustomerID, - } - case dodopayments.SubscriptionRenewedWebhookEvent: - webhookEvent.ID = e.Data.SubscriptionID - webhookEvent.Data = map[string]interface{}{ - "subscription_id": e.Data.SubscriptionID, - "customer_id": e.Data.Customer.CustomerID, - "current_period_start": e.Data.PreviousBillingDate.Format(time.RFC3339), - "current_period_end": e.Data.NextBillingDate.Format(time.RFC3339), - } - case dodopayments.SubscriptionOnHoldWebhookEvent: - webhookEvent.ID = e.Data.SubscriptionID - webhookEvent.Data = map[string]interface{}{ - "subscription_id": e.Data.SubscriptionID, - "customer_id": e.Data.Customer.CustomerID, - } - case dodopayments.PaymentSucceededWebhookEvent: - webhookEvent.ID = e.Data.PaymentID - webhookEvent.Data = map[string]interface{}{ - "payment_id": e.Data.PaymentID, - "subscription_id": e.Data.SubscriptionID, - } - case dodopayments.PaymentFailedWebhookEvent: - webhookEvent.ID = e.Data.PaymentID - webhookEvent.Data = map[string]interface{}{ - "payment_id": e.Data.PaymentID, - "subscription_id": e.Data.SubscriptionID, - } - default: - // For unknown event types, try to extract basic info - webhookEvent.ID = fmt.Sprintf("evt_%d", time.Now().UnixNano()) - webhookEvent.Data = make(map[string]interface{}) - } - - return webhookEvent, nil -} - -// IsEventProcessed checks if a webhook event has already been processed -func (s *PaymentService) IsEventProcessed(ctx context.Context, eventID string) bool { - if s.events == nil { - return false - } - - count, err := s.events.CountDocuments(ctx, bson.M{"dodoEventId": eventID}) - if err != nil { - return false - } - - return count > 0 -} - -// HandleWebhookEvent processes a verified webhook event -func (s *PaymentService) HandleWebhookEvent(ctx context.Context, event *WebhookEvent) error { - // Check idempotency - if s.IsEventProcessed(ctx, event.ID) { - log.Printf("⚠️ Webhook event %s already processed, skipping", event.ID) - return fmt.Errorf("webhook event already processed (idempotent)") - } - - // Log event - eventDoc := models.SubscriptionEvent{ - ID: primitive.NewObjectID(), - DodoEventID: event.ID, - EventType: event.Type, - Metadata: event.Data, - CreatedAt: time.Now(), - } - - if s.events != nil { - _, err := s.events.InsertOne(ctx, eventDoc) - if err != nil { - log.Printf("⚠️ Failed to log webhook event: %v", err) - } - } - - // Handle event based on type - switch event.Type { - case "subscription.active": - return s.handleSubscriptionActive(ctx, event) - case "subscription.updated": - return s.handleSubscriptionUpdated(ctx, event) - case "subscription.on_hold": - return s.handleSubscriptionOnHold(ctx, event) - case "subscription.renewed": - return s.handleSubscriptionRenewed(ctx, event) - case "subscription.cancelled": - return s.handleSubscriptionCancelled(ctx, event) - case "payment.succeeded": - return s.handlePaymentSucceeded(ctx, event) - case "payment.failed": - return s.handlePaymentFailed(ctx, event) - default: - log.Printf("⚠️ Unhandled webhook event type: %s", event.Type) - return nil - } -} - -func (s *PaymentService) handleSubscriptionActive(ctx context.Context, event *WebhookEvent) error { - subID, _ := event.Data["subscription_id"].(string) - customerID, _ := event.Data["customer_id"].(string) - productID, _ := event.Data["product_id"].(string) - - if subID == "" || customerID == "" { - return fmt.Errorf("missing required fields in subscription.active event") - } - - // Find plan by product ID - var plan *models.Plan - for i := range models.AvailablePlans { - if models.AvailablePlans[i].DodoProductID == productID { - plan = &models.AvailablePlans[i] - break - } - } - - if plan == nil { - return fmt.Errorf("unknown product ID: %s", productID) - } - - // Find user by customer ID first - var user models.User - if s.mongoDB == nil { - return fmt.Errorf("MongoDB not available") - } - - err := s.mongoDB.Database().Collection("users").FindOne(ctx, bson.M{"dodoCustomerId": customerID}).Decode(&user) - if err != nil { - log.Printf("⚠️ User not found by customer ID %s, trying to fetch from DodoPayments...", customerID) - - // Fallback: Fetch customer from DodoPayments to get email - if s.client != nil { - customer, fetchErr := s.client.Customers.Get(ctx, customerID) - if fetchErr != nil { - return fmt.Errorf("failed to find user by customer ID and failed to fetch customer: %w", fetchErr) - } - - // Try to find user by email - err = s.mongoDB.Database().Collection("users").FindOne(ctx, bson.M{"email": customer.Email}).Decode(&user) - if err != nil { - return fmt.Errorf("failed to find user by customer ID or email (%s): %w", customer.Email, err) - } - - // Update the user's dodoCustomerId with the new customer ID - log.Printf("✅ Found user by email %s, updating dodoCustomerId to %s", customer.Email, customerID) - _, updateErr := s.mongoDB.Database().Collection("users").UpdateOne( - ctx, - bson.M{"_id": user.ID}, - bson.M{"$set": bson.M{"dodoCustomerId": customerID}}, - ) - if updateErr != nil { - log.Printf("⚠️ Failed to update dodoCustomerId: %v", updateErr) - } - } else { - return fmt.Errorf("failed to find user by customer ID: %w", err) - } - } - - // Parse period dates - var periodStart, periodEnd time.Time - if startStr, ok := event.Data["current_period_start"].(string); ok { - periodStart, _ = time.Parse(time.RFC3339, startStr) - } - if endStr, ok := event.Data["current_period_end"].(string); ok { - periodEnd, _ = time.Parse(time.RFC3339, endStr) - } - - // Upsert subscription - now := time.Now() - - if s.subscriptions != nil { - filter := bson.M{"userId": user.SupabaseUserID} - update := bson.M{ - "$set": bson.M{ - "userId": user.SupabaseUserID, - "dodoSubscriptionId": subID, - "dodoCustomerId": customerID, - "tier": plan.Tier, - "status": models.SubStatusActive, - "currentPeriodStart": periodStart, - "currentPeriodEnd": periodEnd, - "updatedAt": now, - }, - "$setOnInsert": bson.M{ - "createdAt": now, - }, - } - opts := options.Update().SetUpsert(true) - _, err := s.subscriptions.UpdateOne(ctx, filter, update, opts) - if err != nil { - return fmt.Errorf("failed to upsert subscription: %w", err) - } - } - - // Update user subscription tier - if s.userService != nil { - err := s.userService.UpdateSubscriptionWithStatus(ctx, user.SupabaseUserID, plan.Tier, models.SubStatusActive, &periodEnd) - if err != nil { - log.Printf("⚠️ Failed to update user subscription: %v", err) - } - } - - // Invalidate tier cache - if s.tierService != nil { - s.tierService.InvalidateCache(user.SupabaseUserID) - } - - // Reset usage counters on new subscription activation - if s.usageLimiter != nil { - if err := s.usageLimiter.ResetAllCounters(ctx, user.SupabaseUserID); err != nil { - log.Printf("⚠️ Failed to reset usage counters for user %s: %v", user.SupabaseUserID, err) - } else { - log.Printf("✅ [WEBHOOK] Reset usage counters for new subscriber %s", user.SupabaseUserID) - } - } - - log.Printf("✅ Subscription activated for user %s: %s", user.SupabaseUserID, plan.Tier) - return nil -} - -func (s *PaymentService) handleSubscriptionUpdated(ctx context.Context, event *WebhookEvent) error { - subID, _ := event.Data["subscription_id"].(string) - productID, _ := event.Data["product_id"].(string) - if subID == "" { - return fmt.Errorf("missing subscription_id in event") - } - - // Parse period dates from webhook - var periodStart, periodEnd time.Time - if startStr, ok := event.Data["current_period_start"].(string); ok && startStr != "" { - periodStart, _ = time.Parse(time.RFC3339, startStr) - } - if endStr, ok := event.Data["current_period_end"].(string); ok && endStr != "" { - periodEnd, _ = time.Parse(time.RFC3339, endStr) - } - - // Get current subscription - var sub models.Subscription - if s.subscriptions != nil { - err := s.subscriptions.FindOne(ctx, bson.M{"dodoSubscriptionId": subID}).Decode(&sub) - if err != nil { - if err == mongo.ErrNoDocuments { - // Subscription doesn't exist yet - this can happen due to webhook race conditions - // The subscription.active event will create it, so we can safely skip this update - log.Printf("⚠️ Subscription %s not found yet (race condition), skipping update", subID) - return nil - } - return fmt.Errorf("subscription not found: %w", err) - } - } else { - return fmt.Errorf("MongoDB not available") - } - - // Find new plan by product ID (for upgrades/downgrades) - var newPlan *models.Plan - if productID != "" { - for i := range models.AvailablePlans { - if models.AvailablePlans[i].DodoProductID == productID { - newPlan = &models.AvailablePlans[i] - break - } - } - } - - // Check if this is a scheduled downgrade being applied - if sub.HasScheduledChange() && time.Now().After(*sub.ScheduledChangeAt) { - // Apply scheduled downgrade - update := bson.M{ - "$set": bson.M{ - "tier": sub.ScheduledTier, - "scheduledTier": "", - "scheduledChangeAt": nil, - "updatedAt": time.Now(), - }, - } - if !periodEnd.IsZero() { - update["$set"].(bson.M)["currentPeriodEnd"] = periodEnd - } - if !periodStart.IsZero() { - update["$set"].(bson.M)["currentPeriodStart"] = periodStart - } - - if s.subscriptions != nil { - _, err := s.subscriptions.UpdateOne(ctx, bson.M{"_id": sub.ID}, update) - if err != nil { - return fmt.Errorf("failed to apply scheduled downgrade: %w", err) - } - } - - // Update user tier - if s.userService != nil { - err := s.userService.UpdateSubscriptionWithStatus(ctx, sub.UserID, sub.ScheduledTier, models.SubStatusActive, &periodEnd) - if err != nil { - log.Printf("⚠️ Failed to update user tier: %v", err) - } - } - - if s.tierService != nil { - s.tierService.InvalidateCache(sub.UserID) - } - - log.Printf("✅ Scheduled downgrade applied for subscription %s: %s -> %s", subID, sub.Tier, sub.ScheduledTier) - return nil - } - - // Handle tier change from plan upgrade/downgrade - if newPlan != nil && newPlan.Tier != sub.Tier { - updateFields := bson.M{ - "tier": newPlan.Tier, - "updatedAt": time.Now(), - } - if !periodEnd.IsZero() { - updateFields["currentPeriodEnd"] = periodEnd - } - if !periodStart.IsZero() { - updateFields["currentPeriodStart"] = periodStart - } - - update := bson.M{"$set": updateFields} - - if s.subscriptions != nil { - _, err := s.subscriptions.UpdateOne(ctx, bson.M{"_id": sub.ID}, update) - if err != nil { - return fmt.Errorf("failed to update subscription tier: %w", err) - } - } - - // Update user tier - if s.userService != nil { - err := s.userService.UpdateSubscriptionWithStatus(ctx, sub.UserID, newPlan.Tier, models.SubStatusActive, &periodEnd) - if err != nil { - log.Printf("⚠️ Failed to update user tier: %v", err) - } - } - - // Invalidate tier cache - if s.tierService != nil { - s.tierService.InvalidateCache(sub.UserID) - } - - // If upgrade, reset counters to give immediate access to new limits - if isUpgrade(sub.Tier, newPlan.Tier) { - if s.usageLimiter != nil { - if err := s.usageLimiter.ResetAllCounters(ctx, sub.UserID); err != nil { - log.Printf("⚠️ Failed to reset usage counters on upgrade: %v", err) - } else { - log.Printf("✅ [WEBHOOK] Reset usage counters for upgraded user %s (%s -> %s)", sub.UserID, sub.Tier, newPlan.Tier) - } - } - } - - log.Printf("✅ Subscription updated for %s: %s -> %s", subID, sub.Tier, newPlan.Tier) - return nil - } - - // Just update period dates if no tier change - if !periodEnd.IsZero() || !periodStart.IsZero() { - updateFields := bson.M{"updatedAt": time.Now()} - if !periodEnd.IsZero() { - updateFields["currentPeriodEnd"] = periodEnd - } - if !periodStart.IsZero() { - updateFields["currentPeriodStart"] = periodStart - } - - if s.subscriptions != nil { - _, err := s.subscriptions.UpdateOne(ctx, bson.M{"_id": sub.ID}, bson.M{"$set": updateFields}) - if err != nil { - log.Printf("⚠️ Failed to update subscription period dates: %v", err) - } - } - } - - log.Printf("✅ Subscription updated event processed for %s", subID) - return nil -} - -func (s *PaymentService) handleSubscriptionOnHold(ctx context.Context, event *WebhookEvent) error { - subID, _ := event.Data["subscription_id"].(string) - if subID == "" { - return fmt.Errorf("missing subscription_id in event") - } - - update := bson.M{ - "$set": bson.M{ - "status": models.SubStatusOnHold, - "updatedAt": time.Now(), - }, - } - - if s.subscriptions != nil { - _, err := s.subscriptions.UpdateOne(ctx, bson.M{"dodoSubscriptionId": subID}, update) - if err != nil { - return fmt.Errorf("failed to update subscription status: %w", err) - } - } - - log.Printf("⚠️ Subscription %s put on hold", subID) - return nil -} - -func (s *PaymentService) handleSubscriptionRenewed(ctx context.Context, event *WebhookEvent) error { - subID, _ := event.Data["subscription_id"].(string) - if subID == "" { - return fmt.Errorf("missing subscription_id in event") - } - - // Parse period dates - var periodStart, periodEnd time.Time - if startStr, ok := event.Data["current_period_start"].(string); ok { - periodStart, _ = time.Parse(time.RFC3339, startStr) - } - if endStr, ok := event.Data["current_period_end"].(string); ok { - periodEnd, _ = time.Parse(time.RFC3339, endStr) - } - - // Get subscription - var sub models.Subscription - if s.subscriptions != nil { - err := s.subscriptions.FindOne(ctx, bson.M{"dodoSubscriptionId": subID}).Decode(&sub) - if err != nil { - if err == mongo.ErrNoDocuments { - // Subscription doesn't exist yet - this can happen due to webhook race conditions - // The subscription.active event will create it, so we can safely skip this renewal - log.Printf("⚠️ Subscription %s not found yet (race condition), skipping renewal", subID) - return nil - } - return fmt.Errorf("subscription not found: %w", err) - } - - // Check if cancellation was scheduled - if sub.CancelAtPeriodEnd { - // Revert to free tier - update := bson.M{ - "$set": bson.M{ - "tier": models.TierFree, - "status": models.SubStatusCancelled, - "cancelAtPeriodEnd": false, - "cancelledAt": time.Now(), - "updatedAt": time.Now(), - }, - } - _, err = s.subscriptions.UpdateOne(ctx, bson.M{"_id": sub.ID}, update) - if err != nil { - return fmt.Errorf("failed to cancel subscription: %w", err) - } - - // Update user tier - if s.userService != nil { - err = s.userService.UpdateSubscriptionWithStatus(ctx, sub.UserID, models.TierFree, models.SubStatusCancelled, nil) - if err != nil { - log.Printf("⚠️ Failed to update user tier: %v", err) - } - } - - if s.tierService != nil { - s.tierService.InvalidateCache(sub.UserID) - } - - log.Printf("✅ Subscription %s cancelled and reverted to free", subID) - return nil - } - - // Check if a downgrade was scheduled (should be applied on renewal) - if sub.HasScheduledChange() { - // Apply scheduled downgrade - update := bson.M{ - "$set": bson.M{ - "tier": sub.ScheduledTier, - "scheduledTier": "", - "scheduledChangeAt": nil, - "currentPeriodStart": periodStart, - "currentPeriodEnd": periodEnd, - "status": models.SubStatusActive, - "updatedAt": time.Now(), - }, - } - _, err = s.subscriptions.UpdateOne(ctx, bson.M{"_id": sub.ID}, update) - if err != nil { - return fmt.Errorf("failed to apply scheduled downgrade: %w", err) - } - - // Update user tier - if s.userService != nil { - err = s.userService.UpdateSubscriptionWithStatus(ctx, sub.UserID, sub.ScheduledTier, models.SubStatusActive, &periodEnd) - if err != nil { - log.Printf("⚠️ Failed to update user tier: %v", err) - } - } - - if s.tierService != nil { - s.tierService.InvalidateCache(sub.UserID) - } - - log.Printf("✅ Subscription %s renewed and scheduled downgrade to %s applied", subID, sub.ScheduledTier) - return nil - } - - // Normal renewal - just update period dates - update := bson.M{ - "$set": bson.M{ - "currentPeriodStart": periodStart, - "currentPeriodEnd": periodEnd, - "status": models.SubStatusActive, - "updatedAt": time.Now(), - }, - } - _, err = s.subscriptions.UpdateOne(ctx, bson.M{"_id": sub.ID}, update) - if err != nil { - return fmt.Errorf("failed to update subscription: %w", err) - } - } - - log.Printf("✅ Subscription %s renewed", subID) - return nil -} - -func (s *PaymentService) handleSubscriptionCancelled(ctx context.Context, event *WebhookEvent) error { - subID, _ := event.Data["subscription_id"].(string) - if subID == "" { - return fmt.Errorf("missing subscription_id in event") - } - - // Get subscription - var sub models.Subscription - if s.subscriptions != nil { - err := s.subscriptions.FindOne(ctx, bson.M{"dodoSubscriptionId": subID}).Decode(&sub) - if err != nil { - return fmt.Errorf("subscription not found: %w", err) - } - - // Revert to free tier - update := bson.M{ - "$set": bson.M{ - "tier": models.TierFree, - "status": models.SubStatusCancelled, - "cancelledAt": time.Now(), - "updatedAt": time.Now(), - }, - } - _, err = s.subscriptions.UpdateOne(ctx, bson.M{"_id": sub.ID}, update) - if err != nil { - return fmt.Errorf("failed to cancel subscription: %w", err) - } - - // Update user tier - if s.userService != nil { - err = s.userService.UpdateSubscription(ctx, sub.UserID, models.TierFree, nil) - if err != nil { - log.Printf("⚠️ Failed to update user tier: %v", err) - } - } - - if s.tierService != nil { - s.tierService.InvalidateCache(sub.UserID) - } - } - - log.Printf("✅ Subscription %s cancelled", subID) - return nil -} - -func (s *PaymentService) handlePaymentSucceeded(ctx context.Context, event *WebhookEvent) error { - // Payment succeeded - subscription should already be active - // Just log for audit - log.Printf("✅ Payment succeeded for subscription") - return nil -} - -func (s *PaymentService) handlePaymentFailed(ctx context.Context, event *WebhookEvent) error { - subID, _ := event.Data["subscription_id"].(string) - if subID == "" { - return fmt.Errorf("missing subscription_id in event") - } - - // Update subscription status to on_hold - update := bson.M{ - "$set": bson.M{ - "status": models.SubStatusOnHold, - "updatedAt": time.Now(), - }, - } - - if s.subscriptions != nil { - _, err := s.subscriptions.UpdateOne(ctx, bson.M{"dodoSubscriptionId": subID}, update) - if err != nil { - return fmt.Errorf("failed to update subscription status: %w", err) - } - } - - log.Printf("⚠️ Payment failed for subscription %s", subID) - return nil -} - -// SyncSubscriptionFromDodo syncs subscription data from DodoPayments for a user -func (s *PaymentService) SyncSubscriptionFromDodo(ctx context.Context, userID string) (map[string]interface{}, error) { - if s.client == nil { - return nil, fmt.Errorf("DodoPayments client not initialized") - } - - // Get user - user, err := s.userService.GetUserBySupabaseID(ctx, userID) - if err != nil { - return nil, fmt.Errorf("failed to get user: %w", err) - } - - // If user has no dodoCustomerId, try to find by email - customerID := user.DodoCustomerID - if customerID == "" { - // Search for customer by email using the customers list API - log.Printf("⚠️ User %s has no dodoCustomerId, trying to find customer by email...", userID) - - // For now, we'll need the user to do a new checkout to create the customer link - return map[string]interface{}{ - "status": "no_customer", - "message": "No DodoPayments customer linked. Please initiate a new checkout to link your account.", - }, nil - } - - // Get subscriptions from DodoPayments for this customer - subscriptionsPage, err := s.client.Subscriptions.List(ctx, dodopayments.SubscriptionListParams{ - CustomerID: dodopayments.F(customerID), - }) - if err != nil { - return nil, fmt.Errorf("failed to list subscriptions from DodoPayments: %w", err) - } - - // Find active subscription (SubscriptionListResponse type) - var activeSub *dodopayments.SubscriptionListResponse - for i := range subscriptionsPage.Items { - sub := &subscriptionsPage.Items[i] - if sub.Status == dodopayments.SubscriptionStatusActive { - activeSub = sub - break - } - } - - if activeSub == nil { - // No active subscription found - return map[string]interface{}{ - "status": "no_subscription", - "message": "No active subscription found in DodoPayments", - "tier": models.TierFree, - }, nil - } - - // Find plan by product ID - var plan *models.Plan - if activeSub.ProductID != "" { - for i := range models.AvailablePlans { - if models.AvailablePlans[i].DodoProductID == activeSub.ProductID { - plan = &models.AvailablePlans[i] - break - } - } - } - - tier := models.TierFree - if plan != nil { - tier = plan.Tier - } - - // DodoPayments uses NextBillingDate (end of period) and PreviousBillingDate (start of period) - periodStart := activeSub.PreviousBillingDate - periodEnd := activeSub.NextBillingDate - - // Update local subscription - now := time.Now() - if s.subscriptions != nil { - filter := bson.M{"userId": userID} - update := bson.M{ - "$set": bson.M{ - "userId": userID, - "dodoSubscriptionId": activeSub.SubscriptionID, - "dodoCustomerId": customerID, - "tier": tier, - "status": models.SubStatusActive, - "currentPeriodStart": periodStart, - "currentPeriodEnd": periodEnd, - "updatedAt": now, - }, - "$setOnInsert": bson.M{ - "createdAt": now, - }, - } - opts := options.Update().SetUpsert(true) - _, err = s.subscriptions.UpdateOne(ctx, filter, update, opts) - if err != nil { - return nil, fmt.Errorf("failed to update local subscription: %w", err) - } - } - - // Update user tier - if s.userService != nil { - err = s.userService.UpdateSubscriptionWithStatus(ctx, userID, tier, models.SubStatusActive, &periodEnd) - if err != nil { - log.Printf("⚠️ Failed to update user subscription: %v", err) - } - } - - // Invalidate cache - if s.tierService != nil { - s.tierService.InvalidateCache(userID) - } - - log.Printf("✅ Synced subscription for user %s: tier=%s, sub_id=%s", userID, tier, activeSub.SubscriptionID) - - return map[string]interface{}{ - "status": "synced", - "tier": tier, - "subscription_id": activeSub.SubscriptionID, - "period_end": periodEnd, - }, nil -} - -// Helper functions - -// isUpgrade determines if a tier change is an upgrade -func isUpgrade(oldTier, newTier string) bool { - oldRank := models.TierOrder[oldTier] - newRank := models.TierOrder[newTier] - return newRank > oldRank -} - -func (s *PaymentService) updateUserDodoCustomer(ctx context.Context, userID, customerID string) error { - if s.mongoDB == nil { - return fmt.Errorf("MongoDB not available") - } - - _, err := s.mongoDB.Database().Collection("users").UpdateOne( - ctx, - bson.M{"supabaseUserId": userID}, - bson.M{"$set": bson.M{"dodoCustomerId": customerID}}, - ) - return err -} - -func getBaseURL() string { - if url := os.Getenv("FRONTEND_URL"); url != "" { - return strings.TrimSuffix(url, "/") - } - return "http://localhost:5173" -} - -// UsageStats represents the current usage statistics for a user -type UsageStats struct { - Schedules UsageStat `json:"schedules"` - APIKeys UsageStat `json:"api_keys"` - ExecutionsToday UsageStat `json:"executions_today"` - RequestsPerMin UsageStat `json:"requests_per_min"` - Messages UsageStatWithTime `json:"messages"` - FileUploads UsageStatWithTime `json:"file_uploads"` - ImageGenerations UsageStatWithTime `json:"image_generations"` - MemoryExtractions UsageStatWithTime `json:"memory_extractions"` // Daily memory extraction count -} - -// UsageStat represents a single usage statistic -type UsageStat struct { - Current int64 `json:"current"` - Max int64 `json:"max"` -} - -// UsageStatWithTime represents a usage statistic with reset time -type UsageStatWithTime struct { - Current int64 `json:"current"` - Max int64 `json:"max"` - ResetAt time.Time `json:"reset_at"` -} - -// GetUsageStats returns the current usage statistics for a user -func (s *PaymentService) GetUsageStats(ctx context.Context, userID string) (*UsageStats, error) { - if s.mongoDB == nil || s.tierService == nil { - return &UsageStats{}, nil - } - - // Get user's tier limits - limits := s.tierService.GetLimits(ctx, userID) - - // Count schedules - scheduleCount, err := s.mongoDB.Database().Collection("schedules").CountDocuments(ctx, bson.M{"userId": userID}) - if err != nil { - log.Printf("⚠️ Failed to count schedules for user %s: %v", userID, err) - scheduleCount = 0 - } - - // Count API keys - apiKeyCount, err := s.mongoDB.Database().Collection("api_keys").CountDocuments(ctx, bson.M{"userId": userID}) - if err != nil { - log.Printf("⚠️ Failed to count API keys for user %s: %v", userID, err) - apiKeyCount = 0 - } - - // Count executions today (from Redis if available, otherwise from MongoDB) - executionsToday := int64(0) - today := time.Now().UTC().Format("2006-01-02") - startOfDay, _ := time.Parse("2006-01-02", today) - - execCount, err := s.mongoDB.Database().Collection("executions").CountDocuments(ctx, bson.M{ - "userId": userID, - "createdAt": bson.M{ - "$gte": startOfDay, - }, - }) - if err != nil { - log.Printf("⚠️ Failed to count executions for user %s: %v", userID, err) - } else { - executionsToday = execCount - } - - // Get usage counts and reset times from UsageLimiterService - var msgCount, fileCount, imageCount int64 - var msgResetAt, fileResetAt, imageResetAt time.Time - - if s.usageLimiter != nil { - limiterStats, err := s.usageLimiter.GetUsageStats(ctx, userID) - if err == nil { - msgCount = limiterStats.MessagesUsed - fileCount = limiterStats.FileUploadsUsed - imageCount = limiterStats.ImageGensUsed - msgResetAt = limiterStats.MessageResetAt - fileResetAt = limiterStats.FileUploadResetAt - imageResetAt = limiterStats.ImageGenResetAt - } else { - log.Printf("⚠️ Failed to get usage limiter stats for user %s: %v", userID, err) - // Set default reset times - msgResetAt = time.Now().UTC().AddDate(0, 1, 0) - fileResetAt = time.Now().UTC().AddDate(0, 0, 1) - imageResetAt = time.Now().UTC().AddDate(0, 0, 1) - } - } else { - // No usageLimiter available, use default reset times - msgResetAt = time.Now().UTC().AddDate(0, 1, 0) - fileResetAt = time.Now().UTC().AddDate(0, 0, 1) - imageResetAt = time.Now().UTC().AddDate(0, 0, 1) - } - - // Count memory extractions today (completed jobs) - memoryExtractionCount := int64(0) - memoryExtractCount, err := s.mongoDB.Database().Collection("memory_extraction_jobs").CountDocuments(ctx, bson.M{ - "userId": userID, - "status": "completed", - "processedAt": bson.M{ - "$gte": startOfDay, - }, - }) - if err != nil { - log.Printf("⚠️ Failed to count memory extractions for user %s: %v", userID, err) - } else { - memoryExtractionCount = memoryExtractCount - } - - // Calculate next reset time for memory extractions (midnight UTC) - now := time.Now().UTC() - nextMidnight := time.Date(now.Year(), now.Month(), now.Day()+1, 0, 0, 0, 0, time.UTC) - - return &UsageStats{ - Schedules: UsageStat{ - Current: scheduleCount, - Max: int64(limits.MaxSchedules), - }, - APIKeys: UsageStat{ - Current: apiKeyCount, - Max: int64(limits.MaxAPIKeys), - }, - ExecutionsToday: UsageStat{ - Current: executionsToday, - Max: limits.MaxExecutionsPerDay, - }, - RequestsPerMin: UsageStat{ - Current: 0, // This would need real-time rate limiting data - Max: limits.RequestsPerMinute, - }, - Messages: UsageStatWithTime{ - Current: msgCount, - Max: limits.MaxMessagesPerMonth, - ResetAt: msgResetAt, - }, - FileUploads: UsageStatWithTime{ - Current: fileCount, - Max: limits.MaxFileUploadsPerDay, - ResetAt: fileResetAt, - }, - ImageGenerations: UsageStatWithTime{ - Current: imageCount, - Max: limits.MaxImageGensPerDay, - ResetAt: imageResetAt, - }, - MemoryExtractions: UsageStatWithTime{ - Current: memoryExtractionCount, - Max: limits.MaxMemoryExtractionsPerDay, - ResetAt: nextMidnight, - }, - }, nil -} diff --git a/backend/internal/services/payment_service_test.go b/backend/internal/services/payment_service_test.go deleted file mode 100644 index fa21f3f5..00000000 --- a/backend/internal/services/payment_service_test.go +++ /dev/null @@ -1,264 +0,0 @@ -package services - -import ( - "claraverse/internal/models" - "context" - "crypto/hmac" - "crypto/sha256" - "encoding/hex" - "testing" -) - -func TestNewPaymentService(t *testing.T) { - service := NewPaymentService("test_key", "webhook_secret", "business_id", nil, nil, nil, nil) - if service == nil { - t.Fatal("Expected non-nil payment service") - } -} - -func TestPaymentService_DetermineChangeType(t *testing.T) { - service := NewPaymentService("", "", "", nil, nil, nil, nil) - - tests := []struct { - name string - fromTier string - toTier string - isUpgrade bool - isDowngrade bool - }{ - {"free to pro", models.TierFree, models.TierPro, true, false}, - {"pro to max", models.TierPro, models.TierMax, true, false}, - {"max to pro", models.TierMax, models.TierPro, false, true}, - {"pro to free", models.TierPro, models.TierFree, false, true}, - {"same tier", models.TierPro, models.TierPro, false, false}, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - isUpgrade, isDowngrade := service.DetermineChangeType(tt.fromTier, tt.toTier) - if isUpgrade != tt.isUpgrade { - t.Errorf("isUpgrade = %v, want %v", isUpgrade, tt.isUpgrade) - } - if isDowngrade != tt.isDowngrade { - t.Errorf("isDowngrade = %v, want %v", isDowngrade, tt.isDowngrade) - } - }) - } -} - -func TestPaymentService_VerifyWebhookSignature(t *testing.T) { - secret := "test_webhook_secret" - service := NewPaymentService("", secret, "", nil, nil, nil, nil) - - payload := []byte(`{"type":"subscription.active","data":{}}`) - - // Generate valid signature - mac := hmac.New(sha256.New, []byte(secret)) - mac.Write(payload) - validSig := hex.EncodeToString(mac.Sum(nil)) - - tests := []struct { - name string - signature string - expectErr bool - }{ - {"valid signature", validSig, false}, - {"invalid signature", "invalid_sig", true}, - {"empty signature", "", true}, - {"wrong signature", "abcd1234", true}, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - err := service.VerifyWebhook(payload, tt.signature) - if tt.expectErr && err == nil { - t.Error("Expected error but got nil") - } - if !tt.expectErr && err != nil { - t.Errorf("Expected no error but got: %v", err) - } - }) - } -} - -func TestPaymentService_VerifyWebhook_NoSecret(t *testing.T) { - service := NewPaymentService("", "", "", nil, nil, nil, nil) - payload := []byte(`{"type":"subscription.active"}`) - - err := service.VerifyWebhook(payload, "signature") - if err == nil { - t.Error("Expected error when webhook secret is not configured") - } -} - -func TestPaymentService_CalculateProration(t *testing.T) { - service := NewPaymentService("", "", "", nil, nil, nil, nil) - - tests := []struct { - name string - fromPrice int64 // cents - toPrice int64 // cents - daysRemaining int - totalDays int - expectedCharge int64 - }{ - { - name: "upgrade mid-month", - fromPrice: 1000, // $10/month - toPrice: 2000, // $20/month - daysRemaining: 15, - totalDays: 30, - expectedCharge: 500, // $5 (half month difference) - }, - { - name: "upgrade near end", - fromPrice: 1000, - toPrice: 2000, - daysRemaining: 3, - totalDays: 30, - expectedCharge: 100, // ~$1 - }, - { - name: "downgrade mid-month", - fromPrice: 2000, - toPrice: 1000, - daysRemaining: 15, - totalDays: 30, - expectedCharge: -500, // Credit - }, - { - name: "zero days remaining", - fromPrice: 1000, - toPrice: 2000, - daysRemaining: 0, - totalDays: 30, - expectedCharge: 0, - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - charge := service.CalculateProration( - tt.fromPrice, tt.toPrice, - tt.daysRemaining, tt.totalDays, - ) - // Allow 10% variance for rounding - variance := float64(tt.expectedCharge) * 0.1 - if tt.expectedCharge < 0 { - variance = -variance - } - if float64(charge) < float64(tt.expectedCharge)-variance || - float64(charge) > float64(tt.expectedCharge)+variance { - t.Errorf("Proration = %d, want ~%d", charge, tt.expectedCharge) - } - }) - } -} - -func TestPaymentService_GetAvailablePlans(t *testing.T) { - service := NewPaymentService("", "", "", nil, nil, nil, nil) - - plans := service.GetAvailablePlans() - - // Should have free, pro, max, enterprise - if len(plans) < 4 { - t.Errorf("Expected at least 4 plans, got %d", len(plans)) - } - - // Verify enterprise has contact_sales flag - var enterprisePlan *models.Plan - for i := range plans { - if plans[i].Tier == models.TierEnterprise { - enterprisePlan = &plans[i] - break - } - } - - if enterprisePlan == nil { - t.Fatal("Enterprise plan not found") - } - - if !enterprisePlan.ContactSales { - t.Error("Enterprise plan should have ContactSales=true") - } - if enterprisePlan.PriceMonthly != 0 { - t.Error("Enterprise plan should have 0 price (contact sales)") - } - - // Verify pricing order: free < pro < max - var freePlan, proPlan, maxPlan *models.Plan - for i := range plans { - switch plans[i].Tier { - case models.TierFree: - freePlan = &plans[i] - case models.TierPro: - proPlan = &plans[i] - case models.TierMax: - maxPlan = &plans[i] - } - } - - if freePlan == nil || proPlan == nil || maxPlan == nil { - t.Fatal("Missing required plans") - } - - if freePlan.PriceMonthly != 0 { - t.Error("Free plan should be $0") - } - if proPlan.PriceMonthly >= maxPlan.PriceMonthly { - t.Error("Pro+ should be more expensive than Pro") - } -} - -func TestPaymentService_GetCurrentSubscription_NoMongoDB(t *testing.T) { - service := NewPaymentService("", "", "", nil, nil, nil, nil) - ctx := context.Background() - - sub, err := service.GetCurrentSubscription(ctx, "user-123") - if err != nil { - t.Fatalf("Expected no error, got: %v", err) - } - - if sub == nil { - t.Fatal("Expected subscription, got nil") - } - - if sub.Tier != models.TierFree { - t.Errorf("Expected free tier, got %s", sub.Tier) - } - - if sub.Status != models.SubStatusActive { - t.Errorf("Expected active status, got %s", sub.Status) - } -} - -func TestPaymentService_PreviewPlanChange(t *testing.T) { - service := NewPaymentService("", "", "", nil, nil, nil, nil) - ctx := context.Background() - - tests := []struct { - name string - currentTier string - newPlanID string - expectError bool - }{ - {"free to pro", models.TierFree, "pro", false}, - {"pro to max", models.TierPro, "max", false}, - {"max to pro", models.TierMax, "pro", false}, - {"invalid plan", models.TierFree, "invalid", true}, - {"same tier", models.TierFree, "free", true}, // Default tier is free without MongoDB - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - // This test would need MongoDB to set up current subscription - // For now, just test the error cases - if tt.expectError { - _, err := service.PreviewPlanChange(ctx, "user-123", tt.newPlanID) - if err == nil { - t.Error("Expected error but got nil") - } - } - }) - } -} diff --git a/backend/internal/services/provider_service.go b/backend/internal/services/provider_service.go deleted file mode 100644 index 71d6c5f1..00000000 --- a/backend/internal/services/provider_service.go +++ /dev/null @@ -1,379 +0,0 @@ -package services - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "database/sql" - "fmt" - "log" - "path/filepath" - "strings" -) - -// ProviderService handles provider operations -type ProviderService struct { - db *database.DB -} - -// NewProviderService creates a new provider service -func NewProviderService(db *database.DB) *ProviderService { - return &ProviderService{db: db} -} - -// GetAll returns all enabled providers -func (s *ProviderService) GetAll() ([]models.Provider, error) { - rows, err := s.db.Query(` - SELECT id, name, base_url, api_key, enabled, audio_only, image_only, image_edit_only, secure, default_model, system_prompt, favicon, created_at, updated_at - FROM providers - WHERE enabled = 1 - ORDER BY name - `) - if err != nil { - return nil, fmt.Errorf("failed to query providers: %w", err) - } - defer rows.Close() - - var providers []models.Provider - for rows.Next() { - var p models.Provider - var systemPrompt, favicon, defaultModel sql.NullString - if err := rows.Scan(&p.ID, &p.Name, &p.BaseURL, &p.APIKey, &p.Enabled, &p.AudioOnly, &p.ImageOnly, &p.ImageEditOnly, &p.Secure, &defaultModel, &systemPrompt, &favicon, &p.CreatedAt, &p.UpdatedAt); err != nil { - return nil, fmt.Errorf("failed to scan provider: %w", err) - } - if systemPrompt.Valid { - p.SystemPrompt = systemPrompt.String - } - if favicon.Valid { - p.Favicon = favicon.String - } - if defaultModel.Valid { - p.DefaultModel = defaultModel.String - } - providers = append(providers, p) - } - - return providers, nil -} - -// GetAllForModels returns all enabled providers that are NOT audio-only (for model selection) -func (s *ProviderService) GetAllForModels() ([]models.Provider, error) { - rows, err := s.db.Query(` - SELECT id, name, base_url, api_key, enabled, audio_only, image_only, image_edit_only, secure, default_model, system_prompt, favicon, created_at, updated_at - FROM providers - WHERE enabled = 1 AND (audio_only = 0 OR audio_only IS NULL) AND (image_only = 0 OR image_only IS NULL) AND (image_edit_only = 0 OR image_edit_only IS NULL) - ORDER BY name - `) - if err != nil { - return nil, fmt.Errorf("failed to query providers: %w", err) - } - defer rows.Close() - - var providers []models.Provider - for rows.Next() { - var p models.Provider - var systemPrompt, favicon, defaultModel sql.NullString - if err := rows.Scan(&p.ID, &p.Name, &p.BaseURL, &p.APIKey, &p.Enabled, &p.AudioOnly, &p.ImageOnly, &p.ImageEditOnly, &p.Secure, &defaultModel, &systemPrompt, &favicon, &p.CreatedAt, &p.UpdatedAt); err != nil { - return nil, fmt.Errorf("failed to scan provider: %w", err) - } - if systemPrompt.Valid { - p.SystemPrompt = systemPrompt.String - } - if favicon.Valid { - p.Favicon = favicon.String - } - if defaultModel.Valid { - p.DefaultModel = defaultModel.String - } - providers = append(providers, p) - } - - return providers, nil -} - -// GetByID returns a provider by ID -func (s *ProviderService) GetByID(id int) (*models.Provider, error) { - var p models.Provider - var systemPrompt, favicon, defaultModel sql.NullString - err := s.db.QueryRow(` - SELECT id, name, base_url, api_key, enabled, audio_only, image_only, image_edit_only, secure, default_model, system_prompt, favicon, created_at, updated_at - FROM providers - WHERE id = ? - `, id).Scan(&p.ID, &p.Name, &p.BaseURL, &p.APIKey, &p.Enabled, &p.AudioOnly, &p.ImageOnly, &p.ImageEditOnly, &p.Secure, &defaultModel, &systemPrompt, &favicon, &p.CreatedAt, &p.UpdatedAt) - - if err == sql.ErrNoRows { - return nil, fmt.Errorf("provider not found") - } - if err != nil { - return nil, fmt.Errorf("failed to query provider: %w", err) - } - - if systemPrompt.Valid { - p.SystemPrompt = systemPrompt.String - } - if favicon.Valid { - p.Favicon = favicon.String - } - if defaultModel.Valid { - p.DefaultModel = defaultModel.String - } - - return &p, nil -} - -// GetByName returns a provider by name -func (s *ProviderService) GetByName(name string) (*models.Provider, error) { - var p models.Provider - var systemPrompt, favicon, defaultModel sql.NullString - err := s.db.QueryRow(` - SELECT id, name, base_url, api_key, enabled, audio_only, image_only, image_edit_only, secure, default_model, system_prompt, favicon, created_at, updated_at - FROM providers - WHERE name = ? - `, name).Scan(&p.ID, &p.Name, &p.BaseURL, &p.APIKey, &p.Enabled, &p.AudioOnly, &p.ImageOnly, &p.ImageEditOnly, &p.Secure, &defaultModel, &systemPrompt, &favicon, &p.CreatedAt, &p.UpdatedAt) - - if err == sql.ErrNoRows { - return nil, nil // Not found, not an error - } - if err != nil { - return nil, fmt.Errorf("failed to query provider: %w", err) - } - - if systemPrompt.Valid { - p.SystemPrompt = systemPrompt.String - } - if favicon.Valid { - p.Favicon = favicon.String - } - if defaultModel.Valid { - p.DefaultModel = defaultModel.String - } - - return &p, nil -} - -// Create creates a new provider -func (s *ProviderService) Create(config models.ProviderConfig) (*models.Provider, error) { - result, err := s.db.Exec(` - INSERT INTO providers (name, base_url, api_key, enabled, audio_only, image_only, image_edit_only, default_model, system_prompt, favicon) - VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?) - `, config.Name, config.BaseURL, config.APIKey, config.Enabled, config.AudioOnly, config.ImageOnly, config.ImageEditOnly, config.DefaultModel, config.SystemPrompt, config.Favicon) - - if err != nil { - return nil, fmt.Errorf("failed to create provider: %w", err) - } - - id, err := result.LastInsertId() - if err != nil { - return nil, fmt.Errorf("failed to get inserted ID: %w", err) - } - - log.Printf(" ✅ Created provider %s with ID %d", config.Name, id) - return s.GetByID(int(id)) -} - -// Update updates an existing provider -func (s *ProviderService) Update(id int, config models.ProviderConfig) error { - _, err := s.db.Exec(` - UPDATE providers - SET base_url = ?, api_key = ?, enabled = ?, audio_only = ?, image_only = ?, image_edit_only = ?, - default_model = ?, system_prompt = ?, favicon = ?, updated_at = CURRENT_TIMESTAMP - WHERE id = ? - `, config.BaseURL, config.APIKey, config.Enabled, config.AudioOnly, config.ImageOnly, config.ImageEditOnly, - config.DefaultModel, config.SystemPrompt, config.Favicon, id) - - if err != nil { - return fmt.Errorf("failed to update provider: %w", err) - } - - log.Printf(" ✅ Updated provider %s (ID %d)", config.Name, id) - return nil -} - -// SyncFilters syncs filter configuration for a provider -func (s *ProviderService) SyncFilters(providerID int, filters []models.FilterConfig) error { - // Delete old filters - if _, err := s.db.Exec("DELETE FROM provider_model_filters WHERE provider_id = ?", providerID); err != nil { - return fmt.Errorf("failed to delete old filters: %w", err) - } - - // Insert new filters - for _, filter := range filters { - _, err := s.db.Exec(` - INSERT INTO provider_model_filters (provider_id, model_pattern, action, priority) - VALUES (?, ?, ?, ?) - `, providerID, filter.Pattern, filter.Action, filter.Priority) - - if err != nil { - return fmt.Errorf("failed to insert filter: %w", err) - } - - log.Printf(" ✓ Added filter: %s (%s)", filter.Pattern, filter.Action) - } - - return nil -} - -// ApplyFilters applies filter rules to models for a provider -func (s *ProviderService) ApplyFilters(providerID int) error { - // Get filters for this provider ordered by priority (higher first) - rows, err := s.db.Query(` - SELECT model_pattern, action - FROM provider_model_filters - WHERE provider_id = ? - ORDER BY priority DESC, id ASC - `, providerID) - if err != nil { - return fmt.Errorf("failed to query filters: %w", err) - } - defer rows.Close() - - var filters []struct { - Pattern string - Action string - } - - for rows.Next() { - var f struct { - Pattern string - Action string - } - if err := rows.Scan(&f.Pattern, &f.Action); err != nil { - return fmt.Errorf("failed to scan filter: %w", err) - } - filters = append(filters, f) - } - - if len(filters) == 0 { - // No filters, show all models - _, err := s.db.Exec(` - UPDATE models - SET is_visible = 1 - WHERE provider_id = ? - `, providerID) - return err - } - - // Reset visibility - if _, err := s.db.Exec("UPDATE models SET is_visible = 0 WHERE provider_id = ?", providerID); err != nil { - return fmt.Errorf("failed to reset visibility: %w", err) - } - - // Apply filters - for _, filter := range filters { - if filter.Action == "include" { - // Match pattern using SQL LIKE (convert * to %) - pattern := strings.ReplaceAll(filter.Pattern, "*", "%") - _, err := s.db.Exec(` - UPDATE models - SET is_visible = 1 - WHERE provider_id = ? AND (name LIKE ? OR id LIKE ?) - `, providerID, pattern, pattern) - if err != nil { - return fmt.Errorf("failed to apply include filter: %w", err) - } - } else if filter.Action == "exclude" { - pattern := strings.ReplaceAll(filter.Pattern, "*", "%") - _, err := s.db.Exec(` - UPDATE models - SET is_visible = 0 - WHERE provider_id = ? AND (name LIKE ? OR id LIKE ?) - `, providerID, pattern, pattern) - if err != nil { - return fmt.Errorf("failed to apply exclude filter: %w", err) - } - } - } - - return nil -} - -// matchesPattern checks if a model name matches a wildcard pattern -func matchesPattern(name, pattern string) bool { - matched, _ := filepath.Match(pattern, name) - return matched -} - -// GetByModelID returns the provider associated with a given model ID -func (s *ProviderService) GetByModelID(modelID string) (*models.Provider, error) { - var providerID int - err := s.db.QueryRow(` - SELECT provider_id FROM models WHERE id = ? - `, modelID).Scan(&providerID) - - if err == sql.ErrNoRows { - return nil, fmt.Errorf("model not found: %s", modelID) - } - if err != nil { - return nil, fmt.Errorf("failed to query model: %w", err) - } - - return s.GetByID(providerID) -} - -// GetAllIncludingDisabled returns all providers including disabled ones -func (s *ProviderService) GetAllIncludingDisabled() ([]models.Provider, error) { - rows, err := s.db.Query(` - SELECT id, name, base_url, api_key, enabled, audio_only, image_only, image_edit_only, secure, default_model, system_prompt, favicon, created_at, updated_at - FROM providers - ORDER BY name - `) - if err != nil { - return nil, fmt.Errorf("failed to query providers: %w", err) - } - defer rows.Close() - - var providers []models.Provider - for rows.Next() { - var p models.Provider - var systemPrompt, favicon, defaultModel sql.NullString - if err := rows.Scan(&p.ID, &p.Name, &p.BaseURL, &p.APIKey, &p.Enabled, &p.AudioOnly, &p.ImageOnly, &p.ImageEditOnly, &p.Secure, &defaultModel, &systemPrompt, &favicon, &p.CreatedAt, &p.UpdatedAt); err != nil { - return nil, fmt.Errorf("failed to scan provider: %w", err) - } - if systemPrompt.Valid { - p.SystemPrompt = systemPrompt.String - } - if favicon.Valid { - p.Favicon = favicon.String - } - if defaultModel.Valid { - p.DefaultModel = defaultModel.String - } - providers = append(providers, p) - } - - return providers, nil -} - -// Delete removes a provider and all its associated models from the database -func (s *ProviderService) Delete(id int) error { - // Models are deleted automatically via ON DELETE CASCADE - _, err := s.db.Exec(`DELETE FROM providers WHERE id = ?`, id) - if err != nil { - return fmt.Errorf("failed to delete provider: %w", err) - } - return nil -} - -// GetFilters retrieves all filters for a provider -func (s *ProviderService) GetFilters(providerID int) ([]models.FilterConfig, error) { - rows, err := s.db.Query(` - SELECT model_pattern, action, priority - FROM provider_model_filters - WHERE provider_id = ? - ORDER BY priority DESC, id ASC - `, providerID) - if err != nil { - return nil, fmt.Errorf("failed to query filters: %w", err) - } - defer rows.Close() - - var filters []models.FilterConfig - for rows.Next() { - var f models.FilterConfig - if err := rows.Scan(&f.Pattern, &f.Action, &f.Priority); err != nil { - return nil, fmt.Errorf("failed to scan filter: %w", err) - } - filters = append(filters, f) - } - - return filters, nil -} diff --git a/backend/internal/services/provider_service_test.go b/backend/internal/services/provider_service_test.go deleted file mode 100644 index 4ea27e80..00000000 --- a/backend/internal/services/provider_service_test.go +++ /dev/null @@ -1,470 +0,0 @@ -package services - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "os" - "testing" -) - -func setupTestDB(t *testing.T) (*database.DB, func()) { - t.Skip("SQLite tests are deprecated - please use DATABASE_URL with MySQL DSN") - tmpFile := "test_provider_service.db" - db, err := database.New(tmpFile) - if err != nil { - t.Fatalf("Failed to create test database: %v", err) - } - - if err := db.Initialize(); err != nil { - t.Fatalf("Failed to initialize test database: %v", err) - } - - cleanup := func() { - db.Close() - os.Remove(tmpFile) - } - - return db, cleanup -} - -func TestNewProviderService(t *testing.T) { - db, cleanup := setupTestDB(t) - defer cleanup() - - service := NewProviderService(db) - if service == nil { - t.Fatal("Expected non-nil provider service") - } -} - -func TestProviderService_Create(t *testing.T) { - db, cleanup := setupTestDB(t) - defer cleanup() - - service := NewProviderService(db) - - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: "https://api.test.com/v1", - APIKey: "test-key-123", - Enabled: true, - } - - provider, err := service.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - if provider.Name != config.Name { - t.Errorf("Expected name %s, got %s", config.Name, provider.Name) - } - - if provider.BaseURL != config.BaseURL { - t.Errorf("Expected base URL %s, got %s", config.BaseURL, provider.BaseURL) - } - - if provider.APIKey != config.APIKey { - t.Errorf("Expected API key %s, got %s", config.APIKey, provider.APIKey) - } - - if !provider.Enabled { - t.Error("Expected provider to be enabled") - } -} - -func TestProviderService_GetAll(t *testing.T) { - db, cleanup := setupTestDB(t) - defer cleanup() - - service := NewProviderService(db) - - // Create test providers - configs := []models.ProviderConfig{ - {Name: "Provider A", BaseURL: "https://a.com", APIKey: "key-a", Enabled: true}, - {Name: "Provider B", BaseURL: "https://b.com", APIKey: "key-b", Enabled: true}, - {Name: "Provider C", BaseURL: "https://c.com", APIKey: "key-c", Enabled: false}, - } - - for _, config := range configs { - if _, err := service.Create(config); err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - } - - providers, err := service.GetAll() - if err != nil { - t.Fatalf("Failed to get all providers: %v", err) - } - - // Should only return enabled providers - if len(providers) != 2 { - t.Errorf("Expected 2 enabled providers, got %d", len(providers)) - } - - // Verify order (alphabetical) - if providers[0].Name != "Provider A" { - t.Errorf("Expected first provider to be 'Provider A', got %s", providers[0].Name) - } -} - -func TestProviderService_GetByID(t *testing.T) { - db, cleanup := setupTestDB(t) - defer cleanup() - - service := NewProviderService(db) - - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: "https://api.test.com/v1", - APIKey: "test-key", - Enabled: true, - } - - created, err := service.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - // Get by ID - provider, err := service.GetByID(created.ID) - if err != nil { - t.Fatalf("Failed to get provider by ID: %v", err) - } - - if provider.ID != created.ID { - t.Errorf("Expected ID %d, got %d", created.ID, provider.ID) - } - - if provider.Name != config.Name { - t.Errorf("Expected name %s, got %s", config.Name, provider.Name) - } -} - -func TestProviderService_GetByID_NotFound(t *testing.T) { - db, cleanup := setupTestDB(t) - defer cleanup() - - service := NewProviderService(db) - - _, err := service.GetByID(999) - if err == nil { - t.Error("Expected error for non-existent provider, got nil") - } -} - -func TestProviderService_GetByName(t *testing.T) { - db, cleanup := setupTestDB(t) - defer cleanup() - - service := NewProviderService(db) - - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: "https://api.test.com/v1", - APIKey: "test-key", - Enabled: true, - } - - created, err := service.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - // Get by name - provider, err := service.GetByName("Test Provider") - if err != nil { - t.Fatalf("Failed to get provider by name: %v", err) - } - - if provider == nil { - t.Fatal("Expected provider, got nil") - } - - if provider.ID != created.ID { - t.Errorf("Expected ID %d, got %d", created.ID, provider.ID) - } -} - -func TestProviderService_GetByName_NotFound(t *testing.T) { - db, cleanup := setupTestDB(t) - defer cleanup() - - service := NewProviderService(db) - - provider, err := service.GetByName("Non-existent Provider") - if err != nil { - t.Fatalf("Expected no error for non-existent provider, got: %v", err) - } - - if provider != nil { - t.Error("Expected nil provider for non-existent name") - } -} - -func TestProviderService_Update(t *testing.T) { - db, cleanup := setupTestDB(t) - defer cleanup() - - service := NewProviderService(db) - - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: "https://api.test.com/v1", - APIKey: "test-key", - Enabled: true, - } - - created, err := service.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - // Update provider - updateConfig := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: "https://api.updated.com/v2", - APIKey: "updated-key", - Enabled: false, - } - - if err := service.Update(created.ID, updateConfig); err != nil { - t.Fatalf("Failed to update provider: %v", err) - } - - // Verify update - updated, err := service.GetByID(created.ID) - if err != nil { - t.Fatalf("Failed to get updated provider: %v", err) - } - - if updated.BaseURL != updateConfig.BaseURL { - t.Errorf("Expected base URL %s, got %s", updateConfig.BaseURL, updated.BaseURL) - } - - if updated.APIKey != updateConfig.APIKey { - t.Errorf("Expected API key %s, got %s", updateConfig.APIKey, updated.APIKey) - } - - if updated.Enabled != updateConfig.Enabled { - t.Errorf("Expected enabled %v, got %v", updateConfig.Enabled, updated.Enabled) - } -} - -func TestProviderService_SyncFilters(t *testing.T) { - db, cleanup := setupTestDB(t) - defer cleanup() - - service := NewProviderService(db) - - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: "https://api.test.com/v1", - APIKey: "test-key", - Enabled: true, - } - - created, err := service.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - filters := []models.FilterConfig{ - {Pattern: "gpt-4*", Action: "include", Priority: 10}, - {Pattern: "gpt-3.5*", Action: "include", Priority: 5}, - {Pattern: "*preview*", Action: "exclude", Priority: 1}, - } - - if err := service.SyncFilters(created.ID, filters); err != nil { - t.Fatalf("Failed to sync filters: %v", err) - } - - // Verify filters were inserted - var count int - err = db.QueryRow("SELECT COUNT(*) FROM provider_model_filters WHERE provider_id = ?", created.ID).Scan(&count) - if err != nil { - t.Fatalf("Failed to count filters: %v", err) - } - - if count != len(filters) { - t.Errorf("Expected %d filters, got %d", len(filters), count) - } -} - -func TestProviderService_SyncFilters_Replace(t *testing.T) { - db, cleanup := setupTestDB(t) - defer cleanup() - - service := NewProviderService(db) - - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: "https://api.test.com/v1", - APIKey: "test-key", - Enabled: true, - } - - created, err := service.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - // Add initial filters - initialFilters := []models.FilterConfig{ - {Pattern: "gpt-4*", Action: "include", Priority: 10}, - } - - if err := service.SyncFilters(created.ID, initialFilters); err != nil { - t.Fatalf("Failed to sync initial filters: %v", err) - } - - // Update with new filters - newFilters := []models.FilterConfig{ - {Pattern: "claude*", Action: "include", Priority: 15}, - {Pattern: "opus*", Action: "include", Priority: 20}, - } - - if err := service.SyncFilters(created.ID, newFilters); err != nil { - t.Fatalf("Failed to sync new filters: %v", err) - } - - // Verify only new filters exist - var count int - err = db.QueryRow("SELECT COUNT(*) FROM provider_model_filters WHERE provider_id = ?", created.ID).Scan(&count) - if err != nil { - t.Fatalf("Failed to count filters: %v", err) - } - - if count != len(newFilters) { - t.Errorf("Expected %d filters, got %d", len(newFilters), count) - } -} - -func TestProviderService_ApplyFilters(t *testing.T) { - db, cleanup := setupTestDB(t) - defer cleanup() - - service := NewProviderService(db) - - // Create provider - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: "https://api.test.com/v1", - APIKey: "test-key", - Enabled: true, - } - - provider, err := service.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - // Create test models - testModels := []models.Model{ - {ID: "gpt-4-turbo", ProviderID: provider.ID, Name: "gpt-4-turbo", IsVisible: false}, - {ID: "gpt-4-preview", ProviderID: provider.ID, Name: "gpt-4-preview", IsVisible: false}, - {ID: "gpt-3.5-turbo", ProviderID: provider.ID, Name: "gpt-3.5-turbo", IsVisible: false}, - {ID: "claude-3", ProviderID: provider.ID, Name: "claude-3", IsVisible: false}, - } - - for _, model := range testModels { - _, err := db.Exec(` - INSERT INTO models (id, provider_id, name, is_visible) - VALUES (?, ?, ?, ?) - `, model.ID, model.ProviderID, model.Name, model.IsVisible) - if err != nil { - t.Fatalf("Failed to create test model: %v", err) - } - } - - // Set up filters - filters := []models.FilterConfig{ - {Pattern: "gpt-4*", Action: "include", Priority: 10}, - {Pattern: "*preview*", Action: "exclude", Priority: 5}, - } - - if err := service.SyncFilters(provider.ID, filters); err != nil { - t.Fatalf("Failed to sync filters: %v", err) - } - - // Apply filters - if err := service.ApplyFilters(provider.ID); err != nil { - t.Fatalf("Failed to apply filters: %v", err) - } - - // Verify visibility - // gpt-4-turbo should be visible (matches include, not excluded) - // gpt-4-preview should be hidden (matches exclude) - // gpt-3.5-turbo should be hidden (doesn't match any include) - // claude-3 should be hidden (doesn't match any include) - - var visibleCount int - err = db.QueryRow(` - SELECT COUNT(*) FROM models - WHERE provider_id = ? AND is_visible = 1 - `, provider.ID).Scan(&visibleCount) - if err != nil { - t.Fatalf("Failed to count visible models: %v", err) - } - - if visibleCount != 1 { - t.Errorf("Expected 1 visible model, got %d", visibleCount) - } - - // Check specific model - var isVisible bool - err = db.QueryRow(` - SELECT is_visible FROM models WHERE id = ? - `, "gpt-4-turbo").Scan(&isVisible) - if err != nil { - t.Fatalf("Failed to get model visibility: %v", err) - } - - if !isVisible { - t.Error("Expected gpt-4-turbo to be visible") - } -} - -func TestProviderService_ApplyFilters_NoFilters(t *testing.T) { - db, cleanup := setupTestDB(t) - defer cleanup() - - service := NewProviderService(db) - - // Create provider - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: "https://api.test.com/v1", - APIKey: "test-key", - Enabled: true, - } - - provider, err := service.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - // Create test model - _, err = db.Exec(` - INSERT INTO models (id, provider_id, name, is_visible) - VALUES (?, ?, ?, ?) - `, "test-model", provider.ID, "test-model", false) - if err != nil { - t.Fatalf("Failed to create test model: %v", err) - } - - // Apply filters (no filters configured) - if err := service.ApplyFilters(provider.ID); err != nil { - t.Fatalf("Failed to apply filters: %v", err) - } - - // All models should be visible when no filters exist - var isVisible bool - err = db.QueryRow("SELECT is_visible FROM models WHERE id = ?", "test-model").Scan(&isVisible) - if err != nil { - t.Fatalf("Failed to get model visibility: %v", err) - } - - if !isVisible { - t.Error("Expected model to be visible when no filters configured") - } -} diff --git a/backend/internal/services/pubsub_service.go b/backend/internal/services/pubsub_service.go deleted file mode 100644 index 7cd48310..00000000 --- a/backend/internal/services/pubsub_service.go +++ /dev/null @@ -1,242 +0,0 @@ -package services - -import ( - "context" - "encoding/json" - "log" - "sync" - - "github.com/redis/go-redis/v9" -) - -// PubSubService manages Redis pub/sub for cross-instance communication -type PubSubService struct { - redis *RedisService - pubsub *redis.PubSub - handlers map[string][]MessageHandler - mu sync.RWMutex - instanceID string - ctx context.Context - cancel context.CancelFunc -} - -// MessageHandler is a callback for handling pub/sub messages -type MessageHandler func(channel string, message *PubSubMessage) - -// PubSubMessage represents a message sent via pub/sub -type PubSubMessage struct { - Type string `json:"type"` // Message type (e.g., "execution_update", "agent_status") - UserID string `json:"userId"` // Target user ID - AgentID string `json:"agentId,omitempty"` - InstanceID string `json:"instanceId"` // Source instance ID - Payload map[string]interface{} `json:"payload"` // Message payload -} - -// NewPubSubService creates a new pub/sub service -func NewPubSubService(redisService *RedisService, instanceID string) *PubSubService { - ctx, cancel := context.WithCancel(context.Background()) - return &PubSubService{ - redis: redisService, - handlers: make(map[string][]MessageHandler), - instanceID: instanceID, - ctx: ctx, - cancel: cancel, - } -} - -// Subscribe subscribes to a channel pattern -func (s *PubSubService) Subscribe(pattern string, handler MessageHandler) { - s.mu.Lock() - defer s.mu.Unlock() - - s.handlers[pattern] = append(s.handlers[pattern], handler) - log.Printf("📡 [PUBSUB] Subscribed to pattern: %s", pattern) -} - -// Start begins listening for pub/sub messages -func (s *PubSubService) Start() error { - client := s.redis.Client() - - // Subscribe to all user channels and global channels - s.pubsub = client.PSubscribe(s.ctx, - "user:*:events", // User-specific events - "agent:*:events", // Agent-specific events - "broadcast:*", // Global broadcast - ) - - // Wait for subscription confirmation - _, err := s.pubsub.Receive(s.ctx) - if err != nil { - return err - } - - // Start message processor - go s.processMessages() - - log.Printf("✅ [PUBSUB] Started listening for messages (instance: %s)", s.instanceID) - return nil -} - -// processMessages handles incoming pub/sub messages -func (s *PubSubService) processMessages() { - ch := s.pubsub.Channel() - - for { - select { - case <-s.ctx.Done(): - return - case msg, ok := <-ch: - if !ok { - return - } - s.handleMessage(msg) - } - } -} - -// handleMessage processes a single pub/sub message -func (s *PubSubService) handleMessage(msg *redis.Message) { - var message PubSubMessage - if err := json.Unmarshal([]byte(msg.Payload), &message); err != nil { - log.Printf("⚠️ [PUBSUB] Failed to unmarshal message: %v", err) - return - } - - // Skip messages from this instance (avoid loops) - if message.InstanceID == s.instanceID { - return - } - - // Find matching handlers - s.mu.RLock() - defer s.mu.RUnlock() - - // Check for exact match - if handlers, ok := s.handlers[msg.Channel]; ok { - for _, handler := range handlers { - go handler(msg.Channel, &message) - } - } - - // Check for pattern matches (simplified - real implementation would use glob matching) - for pattern, handlers := range s.handlers { - if matchPattern(pattern, msg.Channel) { - for _, handler := range handlers { - go handler(msg.Channel, &message) - } - } - } -} - -// PublishToUser publishes a message to a user's channel -func (s *PubSubService) PublishToUser(ctx context.Context, userID string, msgType string, payload map[string]interface{}) error { - message := &PubSubMessage{ - Type: msgType, - UserID: userID, - InstanceID: s.instanceID, - Payload: payload, - } - - data, err := json.Marshal(message) - if err != nil { - return err - } - - channel := "user:" + userID + ":events" - return s.redis.Client().Publish(ctx, channel, data).Err() -} - -// PublishToAgent publishes a message to an agent's channel -func (s *PubSubService) PublishToAgent(ctx context.Context, agentID string, msgType string, payload map[string]interface{}) error { - message := &PubSubMessage{ - Type: msgType, - AgentID: agentID, - InstanceID: s.instanceID, - Payload: payload, - } - - data, err := json.Marshal(message) - if err != nil { - return err - } - - channel := "agent:" + agentID + ":events" - return s.redis.Client().Publish(ctx, channel, data).Err() -} - -// Broadcast publishes a message to all instances -func (s *PubSubService) Broadcast(ctx context.Context, topic string, msgType string, payload map[string]interface{}) error { - message := &PubSubMessage{ - Type: msgType, - InstanceID: s.instanceID, - Payload: payload, - } - - data, err := json.Marshal(message) - if err != nil { - return err - } - - channel := "broadcast:" + topic - return s.redis.Client().Publish(ctx, channel, data).Err() -} - -// PublishExecutionUpdate publishes an execution update for a user -func (s *PubSubService) PublishExecutionUpdate(ctx context.Context, userID, agentID, executionID string, update map[string]interface{}) error { - payload := map[string]interface{}{ - "executionId": executionID, - "agentId": agentID, - "update": update, - } - - return s.PublishToUser(ctx, userID, "execution_update", payload) -} - -// Stop stops the pub/sub service -func (s *PubSubService) Stop() error { - s.cancel() - if s.pubsub != nil { - return s.pubsub.Close() - } - return nil -} - -// matchPattern checks if a channel matches a pattern (simplified glob) -func matchPattern(pattern, channel string) bool { - // Simple wildcard matching - if pattern == channel { - return true - } - - // Handle patterns like "user:*:events" - patternParts := splitChannel(pattern) - channelParts := splitChannel(channel) - - if len(patternParts) != len(channelParts) { - return false - } - - for i, part := range patternParts { - if part != "*" && part != channelParts[i] { - return false - } - } - - return true -} - -// splitChannel splits a channel name by ":" -func splitChannel(channel string) []string { - var parts []string - current := "" - for _, c := range channel { - if c == ':' { - parts = append(parts, current) - current = "" - } else { - current += string(c) - } - } - parts = append(parts, current) - return parts -} diff --git a/backend/internal/services/redis_service.go b/backend/internal/services/redis_service.go deleted file mode 100644 index a592e136..00000000 --- a/backend/internal/services/redis_service.go +++ /dev/null @@ -1,185 +0,0 @@ -package services - -import ( - "context" - "fmt" - "log" - "sync" - "time" - - "github.com/redis/go-redis/v9" -) - -// RedisService provides Redis connection and operations -type RedisService struct { - client *redis.Client - mu sync.RWMutex -} - -var ( - redisInstance *RedisService - redisOnce sync.Once -) - -// NewRedisService creates a new Redis service instance -func NewRedisService(redisURL string) (*RedisService, error) { - var initErr error - - redisOnce.Do(func() { - opts, err := redis.ParseURL(redisURL) - if err != nil { - initErr = fmt.Errorf("failed to parse Redis URL: %w", err) - return - } - - // Configure connection pool - opts.PoolSize = 10 - opts.MinIdleConns = 2 - opts.MaxRetries = 3 - opts.DialTimeout = 5 * time.Second - opts.ReadTimeout = 3 * time.Second - opts.WriteTimeout = 3 * time.Second - - client := redis.NewClient(opts) - - // Test connection - ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) - defer cancel() - - if err := client.Ping(ctx).Err(); err != nil { - initErr = fmt.Errorf("failed to connect to Redis: %w", err) - return - } - - redisInstance = &RedisService{ - client: client, - } - - log.Println("✅ Redis connection established") - }) - - if initErr != nil { - return nil, initErr - } - - return redisInstance, nil -} - -// GetRedisService returns the singleton Redis service instance -func GetRedisService() *RedisService { - return redisInstance -} - -// Client returns the underlying Redis client -func (r *RedisService) Client() *redis.Client { - r.mu.RLock() - defer r.mu.RUnlock() - return r.client -} - -// Close closes the Redis connection -func (r *RedisService) Close() error { - r.mu.Lock() - defer r.mu.Unlock() - - if r.client != nil { - return r.client.Close() - } - return nil -} - -// Ping checks if Redis is healthy -func (r *RedisService) Ping(ctx context.Context) error { - return r.client.Ping(ctx).Err() -} - -// Set sets a key-value pair with optional expiration -func (r *RedisService) Set(ctx context.Context, key string, value interface{}, expiration time.Duration) error { - return r.client.Set(ctx, key, value, expiration).Err() -} - -// Get retrieves a value by key -func (r *RedisService) Get(ctx context.Context, key string) (string, error) { - return r.client.Get(ctx, key).Result() -} - -// Delete removes a key -func (r *RedisService) Delete(ctx context.Context, keys ...string) error { - return r.client.Del(ctx, keys...).Err() -} - -// SetNX sets a key only if it doesn't exist (for distributed locking) -func (r *RedisService) SetNX(ctx context.Context, key string, value interface{}, expiration time.Duration) (bool, error) { - return r.client.SetNX(ctx, key, value, expiration).Result() -} - -// Publish publishes a message to a channel -func (r *RedisService) Publish(ctx context.Context, channel string, message interface{}) error { - return r.client.Publish(ctx, channel, message).Err() -} - -// Subscribe subscribes to one or more channels -func (r *RedisService) Subscribe(ctx context.Context, channels ...string) *redis.PubSub { - return r.client.Subscribe(ctx, channels...) -} - -// Incr increments a counter (for rate limiting) -func (r *RedisService) Incr(ctx context.Context, key string) (int64, error) { - return r.client.Incr(ctx, key).Result() -} - -// Expire sets expiration on a key -func (r *RedisService) Expire(ctx context.Context, key string, expiration time.Duration) error { - return r.client.Expire(ctx, key, expiration).Err() -} - -// TTL gets the remaining time to live for a key -func (r *RedisService) TTL(ctx context.Context, key string) (time.Duration, error) { - return r.client.TTL(ctx, key).Result() -} - -// AcquireLock attempts to acquire a distributed lock -// Returns true if lock was acquired, false otherwise -func (r *RedisService) AcquireLock(ctx context.Context, lockKey string, lockValue string, expiration time.Duration) (bool, error) { - return r.client.SetNX(ctx, lockKey, lockValue, expiration).Result() -} - -// ReleaseLock releases a distributed lock if it's still held by the given value -func (r *RedisService) ReleaseLock(ctx context.Context, lockKey string, lockValue string) (bool, error) { - // Lua script to atomically check and delete - script := redis.NewScript(` - if redis.call("get", KEYS[1]) == ARGV[1] then - return redis.call("del", KEYS[1]) - else - return 0 - end - `) - - result, err := script.Run(ctx, r.client, []string{lockKey}, lockValue).Int64() - if err != nil { - return false, err - } - - return result == 1, nil -} - -// CheckRateLimit checks if a rate limit has been exceeded -// Returns remaining requests and whether the limit was exceeded -func (r *RedisService) CheckRateLimit(ctx context.Context, key string, limit int64, window time.Duration) (remaining int64, exceeded bool, err error) { - count, err := r.client.Incr(ctx, key).Result() - if err != nil { - return 0, false, err - } - - // Set expiry on first request - if count == 1 { - r.client.Expire(ctx, key, window) - } - - remaining = limit - count - if remaining < 0 { - remaining = 0 - } - - return remaining, count > limit, nil -} diff --git a/backend/internal/services/scheduler_service.go b/backend/internal/services/scheduler_service.go deleted file mode 100644 index e12ccbd0..00000000 --- a/backend/internal/services/scheduler_service.go +++ /dev/null @@ -1,730 +0,0 @@ -package services - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "context" - "fmt" - "log" - "sync" - "time" - - "github.com/go-co-op/gocron/v2" - "github.com/google/uuid" - "github.com/robfig/cron/v3" - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/bson/primitive" - "go.mongodb.org/mongo-driver/mongo" - "go.mongodb.org/mongo-driver/mongo/options" -) - -// SchedulerService manages scheduled agent executions -type SchedulerService struct { - scheduler gocron.Scheduler - mongoDB *database.MongoDB - redisService *RedisService - agentService *AgentService - executionService *ExecutionService - workflowExecutor models.WorkflowExecuteFunc - instanceID string - mu sync.RWMutex - jobs map[string]gocron.Job // scheduleID -> job -} - -// NewSchedulerService creates a new scheduler service -func NewSchedulerService( - mongoDB *database.MongoDB, - redisService *RedisService, - agentService *AgentService, - executionService *ExecutionService, -) (*SchedulerService, error) { - // Create scheduler with second-level precision - scheduler, err := gocron.NewScheduler( - gocron.WithLocation(time.UTC), - ) - if err != nil { - return nil, fmt.Errorf("failed to create scheduler: %w", err) - } - - return &SchedulerService{ - scheduler: scheduler, - mongoDB: mongoDB, - redisService: redisService, - agentService: agentService, - executionService: executionService, - instanceID: uuid.New().String(), - jobs: make(map[string]gocron.Job), - }, nil -} - -// Start starts the scheduler and loads all enabled schedules -func (s *SchedulerService) Start(ctx context.Context) error { - log.Println("⏰ Starting scheduler service...") - - // Load and register all enabled schedules - if err := s.loadSchedules(ctx); err != nil { - log.Printf("⚠️ Failed to load schedules: %v", err) - } - - // Start the scheduler - s.scheduler.Start() - log.Println("✅ Scheduler service started") - - return nil -} - -// Stop stops the scheduler -func (s *SchedulerService) Stop() error { - log.Println("⏹️ Stopping scheduler service...") - return s.scheduler.Shutdown() -} - -// SetWorkflowExecutor sets the workflow executor function (used for deferred initialization) -func (s *SchedulerService) SetWorkflowExecutor(executor models.WorkflowExecuteFunc) { - s.mu.Lock() - defer s.mu.Unlock() - s.workflowExecutor = executor -} - -// loadSchedules loads all enabled schedules from MongoDB and registers them -func (s *SchedulerService) loadSchedules(ctx context.Context) error { - if s.mongoDB == nil { - log.Println("⚠️ MongoDB not available, skipping schedule loading") - return nil - } - - collection := s.mongoDB.Database().Collection("schedules") - - cursor, err := collection.Find(ctx, bson.M{"enabled": true}) - if err != nil { - return fmt.Errorf("failed to query schedules: %w", err) - } - defer cursor.Close(ctx) - - var count int - for cursor.Next(ctx) { - var schedule models.Schedule - if err := cursor.Decode(&schedule); err != nil { - log.Printf("⚠️ Failed to decode schedule: %v", err) - continue - } - - if err := s.registerJob(&schedule); err != nil { - log.Printf("⚠️ Failed to register schedule %s: %v", schedule.ID.Hex(), err) - continue - } - count++ - } - - log.Printf("✅ Loaded %d schedules", count) - return nil -} - -// registerJob registers a schedule with gocron -func (s *SchedulerService) registerJob(schedule *models.Schedule) error { - s.mu.Lock() - defer s.mu.Unlock() - - // Validate timezone - _, err := time.LoadLocation(schedule.Timezone) - if err != nil { - return fmt.Errorf("invalid timezone %s: %w", schedule.Timezone, err) - } - - // Build cron expression with timezone prefix (CRON_TZ=America/New_York 0 9 * * *) - cronWithTZ := fmt.Sprintf("CRON_TZ=%s %s", schedule.Timezone, schedule.CronExpression) - - // Create the job - job, err := s.scheduler.NewJob( - gocron.CronJob(cronWithTZ, false), - gocron.NewTask(func() { - s.executeScheduledJob(schedule) - }), - gocron.WithName(schedule.ID.Hex()), - gocron.WithTags(schedule.AgentID, schedule.UserID), - ) - if err != nil { - return fmt.Errorf("failed to create job: %w", err) - } - - s.jobs[schedule.ID.Hex()] = job - log.Printf("📅 Registered schedule %s for agent %s (cron: %s, tz: %s)", - schedule.ID.Hex(), schedule.AgentID, schedule.CronExpression, schedule.Timezone) - - return nil -} - -// unregisterJob removes a job from the scheduler -func (s *SchedulerService) unregisterJob(scheduleID string) error { - s.mu.Lock() - defer s.mu.Unlock() - - job, exists := s.jobs[scheduleID] - if !exists { - return nil - } - - if err := s.scheduler.RemoveJob(job.ID()); err != nil { - return fmt.Errorf("failed to remove job: %w", err) - } - - delete(s.jobs, scheduleID) - log.Printf("🗑️ Unregistered schedule %s", scheduleID) - - return nil -} - -// executeScheduledJob executes a scheduled agent workflow -func (s *SchedulerService) executeScheduledJob(schedule *models.Schedule) { - ctx := context.Background() - - // Create a unique lock key for this schedule execution window - // Using minute-level granularity to prevent duplicate runs within the same minute - lockKey := fmt.Sprintf("schedule-lock:%s:%d", schedule.ID.Hex(), time.Now().Unix()/60) - - // Try to acquire distributed lock - acquired, err := s.redisService.AcquireLock(ctx, lockKey, s.instanceID, 5*time.Minute) - if err != nil { - log.Printf("❌ Failed to acquire lock for schedule %s: %v", schedule.ID.Hex(), err) - return - } - - if !acquired { - // Another instance is handling this execution - log.Printf("⏭️ Schedule %s already being executed by another instance", schedule.ID.Hex()) - return - } - - // Release lock when done - defer func() { - if _, err := s.redisService.ReleaseLock(ctx, lockKey, s.instanceID); err != nil { - log.Printf("⚠️ Failed to release lock for schedule %s: %v", schedule.ID.Hex(), err) - } - }() - - log.Printf("▶️ Executing scheduled job for agent %s (schedule: %s)", schedule.AgentID, schedule.ID.Hex()) - - // Get the agent and workflow - // Note: We use a system context here since scheduled jobs run without a user session - agent, err := s.agentService.GetAgentByID(schedule.AgentID) - if err != nil { - log.Printf("❌ Failed to get agent %s: %v", schedule.AgentID, err) - s.updateScheduleStats(ctx, schedule.ID, false, schedule) - return - } - - if agent.Workflow == nil { - log.Printf("❌ Agent %s has no workflow", schedule.AgentID) - s.updateScheduleStats(ctx, schedule.ID, false, schedule) - return - } - - // Build input from template - input := make(map[string]interface{}) - if schedule.InputTemplate != nil { - for k, v := range schedule.InputTemplate { - input[k] = v - } - } - - // CRITICAL: Inject user context for credential resolution and tool execution - // Without this, tools like send_discord_message cannot access user credentials - input["__user_id__"] = schedule.UserID - log.Printf("🔐 [SCHEDULER] Injecting user context: __user_id__=%s for schedule %s", schedule.UserID, schedule.ID.Hex()) - - // Create execution record in MongoDB - var execRecord *ExecutionRecord - if s.executionService != nil { - var err error - execRecord, err = s.executionService.Create(ctx, &CreateExecutionRequest{ - AgentID: schedule.AgentID, - UserID: schedule.UserID, - WorkflowVersion: agent.Workflow.Version, - TriggerType: "scheduled", - ScheduleID: schedule.ID, - Input: input, - }) - if err != nil { - log.Printf("⚠️ Failed to create execution record: %v", err) - } else { - // Mark as running - s.executionService.UpdateStatus(ctx, execRecord.ID, "running") - } - } - - // Check if workflow executor is available - s.mu.RLock() - executor := s.workflowExecutor - s.mu.RUnlock() - - if executor == nil { - log.Printf("❌ Workflow executor not set for schedule %s", schedule.ID.Hex()) - s.updateScheduleStats(ctx, schedule.ID, false, schedule) - if execRecord != nil { - s.executionService.Complete(ctx, execRecord.ID, &ExecutionCompleteRequest{ - Status: "failed", - Error: "Workflow executor not available", - }) - } - return - } - - // Execute the workflow using the callback function - result, execErr := executor(agent.Workflow, input) - - // Determine success status - status := "failed" - if result != nil { - status = result.Status - } - success := status == "completed" && execErr == nil - - // Complete the execution record - if execRecord != nil { - completeReq := &ExecutionCompleteRequest{ - Status: status, - } - if execErr != nil { - completeReq.Error = execErr.Error() - } else if result != nil { - completeReq.Output = result.Output - completeReq.BlockStates = result.BlockStates - if result.Error != "" { - completeReq.Error = result.Error - } - } - s.executionService.Complete(ctx, execRecord.ID, completeReq) - log.Printf("📊 Scheduled execution %s completed with status: %s", execRecord.ID.Hex(), status) - } - - if success { - log.Printf("✅ Scheduled execution completed successfully for agent %s", schedule.AgentID) - } else { - errMsg := "" - if execErr != nil { - errMsg = execErr.Error() - } else if result != nil && result.Error != "" { - errMsg = result.Error - } - log.Printf("❌ Scheduled execution failed for agent %s: %s (status: %s)", schedule.AgentID, errMsg, status) - } - - // Update schedule statistics and next run time - s.updateScheduleStats(ctx, schedule.ID, success, schedule) -} - -// updateScheduleStats updates the schedule's run statistics and next run time -func (s *SchedulerService) updateScheduleStats(ctx context.Context, scheduleID primitive.ObjectID, success bool, schedule *models.Schedule) { - if s.mongoDB == nil { - return - } - - collection := s.mongoDB.Database().Collection("schedules") - - now := time.Now() - - // Calculate next run time - parser := cron.NewParser(cron.Minute | cron.Hour | cron.Dom | cron.Month | cron.Dow) - cronSchedule, err := parser.Parse(schedule.CronExpression) - var nextRun time.Time - if err == nil { - loc, locErr := time.LoadLocation(schedule.Timezone) - if locErr == nil { - nextRun = cronSchedule.Next(now.In(loc)) - } else { - nextRun = cronSchedule.Next(now) - } - } - - update := bson.M{ - "$set": bson.M{ - "lastRunAt": now, - "updatedAt": now, - "nextRunAt": nextRun, - }, - "$inc": bson.M{ - "totalRuns": 1, - }, - } - - if success { - update["$inc"].(bson.M)["successfulRuns"] = 1 - } else { - update["$inc"].(bson.M)["failedRuns"] = 1 - } - - if _, err := collection.UpdateByID(ctx, scheduleID, update); err != nil { - log.Printf("⚠️ Failed to update schedule stats: %v", err) - } else { - log.Printf("📅 Updated next run time to %v for schedule %s", nextRun, scheduleID.Hex()) - } -} - -// CreateSchedule creates a new schedule for an agent -func (s *SchedulerService) CreateSchedule(ctx context.Context, agentID, userID string, req *models.CreateScheduleRequest) (*models.Schedule, error) { - if s.mongoDB == nil { - return nil, fmt.Errorf("MongoDB not available") - } - - // Validate cron expression - parser := cron.NewParser(cron.Minute | cron.Hour | cron.Dom | cron.Month | cron.Dow) - if _, err := parser.Parse(req.CronExpression); err != nil { - return nil, fmt.Errorf("invalid cron expression: %w", err) - } - - // Validate timezone - loc, err := time.LoadLocation(req.Timezone) - if err != nil { - return nil, fmt.Errorf("invalid timezone: %w", err) - } - - // Check user's schedule limit - limits := models.GetTierLimits(s.getUserTier(ctx, userID)) - if limits.MaxSchedules > 0 { - count, err := s.countUserSchedules(ctx, userID) - if err != nil { - return nil, fmt.Errorf("failed to check schedule limit: %w", err) - } - if count >= int64(limits.MaxSchedules) { - return nil, fmt.Errorf("active schedule limit reached (%d/%d). Pause an existing schedule to create a new one", count, limits.MaxSchedules) - } - } - - // Check if agent already has a schedule - existing, _ := s.GetScheduleByAgentID(ctx, agentID, userID) - if existing != nil { - return nil, fmt.Errorf("agent already has a schedule") - } - - // Calculate next run time - schedule, _ := parser.Parse(req.CronExpression) - nextRun := schedule.Next(time.Now().In(loc)) - - // Default enabled to true - enabled := true - if req.Enabled != nil { - enabled = *req.Enabled - } - - now := time.Now() - doc := &models.Schedule{ - ID: primitive.NewObjectID(), - AgentID: agentID, - UserID: userID, - CronExpression: req.CronExpression, - Timezone: req.Timezone, - Enabled: enabled, - InputTemplate: req.InputTemplate, - NextRunAt: &nextRun, - TotalRuns: 0, - SuccessfulRuns: 0, - FailedRuns: 0, - CreatedAt: now, - UpdatedAt: now, - } - - collection := s.mongoDB.Database().Collection("schedules") - if _, err := collection.InsertOne(ctx, doc); err != nil { - return nil, fmt.Errorf("failed to create schedule: %w", err) - } - - // Register with scheduler if enabled - if enabled { - if err := s.registerJob(doc); err != nil { - log.Printf("⚠️ Failed to register new schedule: %v", err) - } - } - - log.Printf("✅ Created schedule %s for agent %s", doc.ID.Hex(), agentID) - return doc, nil -} - -// GetSchedule retrieves a schedule by ID -func (s *SchedulerService) GetSchedule(ctx context.Context, scheduleID, userID string) (*models.Schedule, error) { - if s.mongoDB == nil { - return nil, fmt.Errorf("MongoDB not available") - } - - objID, err := primitive.ObjectIDFromHex(scheduleID) - if err != nil { - return nil, fmt.Errorf("invalid schedule ID") - } - - collection := s.mongoDB.Database().Collection("schedules") - - var schedule models.Schedule - err = collection.FindOne(ctx, bson.M{ - "_id": objID, - "userId": userID, - }).Decode(&schedule) - - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("schedule not found") - } - if err != nil { - return nil, fmt.Errorf("failed to get schedule: %w", err) - } - - return &schedule, nil -} - -// GetScheduleByAgentID retrieves a schedule by agent ID -func (s *SchedulerService) GetScheduleByAgentID(ctx context.Context, agentID, userID string) (*models.Schedule, error) { - if s.mongoDB == nil { - return nil, fmt.Errorf("MongoDB not available") - } - - collection := s.mongoDB.Database().Collection("schedules") - - var schedule models.Schedule - err := collection.FindOne(ctx, bson.M{ - "agentId": agentID, - "userId": userID, - }).Decode(&schedule) - - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("schedule not found") - } - if err != nil { - return nil, fmt.Errorf("failed to get schedule: %w", err) - } - - return &schedule, nil -} - -// UpdateSchedule updates a schedule -func (s *SchedulerService) UpdateSchedule(ctx context.Context, scheduleID, userID string, req *models.UpdateScheduleRequest) (*models.Schedule, error) { - if s.mongoDB == nil { - return nil, fmt.Errorf("MongoDB not available") - } - - // Get existing schedule - schedule, err := s.GetSchedule(ctx, scheduleID, userID) - if err != nil { - return nil, err - } - - // Build update - update := bson.M{ - "updatedAt": time.Now(), - } - - if req.CronExpression != nil { - // Validate cron expression - parser := cron.NewParser(cron.Minute | cron.Hour | cron.Dom | cron.Month | cron.Dow) - if _, err := parser.Parse(*req.CronExpression); err != nil { - return nil, fmt.Errorf("invalid cron expression: %w", err) - } - update["cronExpression"] = *req.CronExpression - schedule.CronExpression = *req.CronExpression - } - - if req.Timezone != nil { - if _, err := time.LoadLocation(*req.Timezone); err != nil { - return nil, fmt.Errorf("invalid timezone: %w", err) - } - update["timezone"] = *req.Timezone - schedule.Timezone = *req.Timezone - } - - if req.InputTemplate != nil { - update["inputTemplate"] = req.InputTemplate - schedule.InputTemplate = req.InputTemplate - } - - if req.Enabled != nil { - update["enabled"] = *req.Enabled - schedule.Enabled = *req.Enabled - } - - // Recalculate next run time - parser := cron.NewParser(cron.Minute | cron.Hour | cron.Dom | cron.Month | cron.Dow) - cronSchedule, _ := parser.Parse(schedule.CronExpression) - loc, _ := time.LoadLocation(schedule.Timezone) - nextRun := cronSchedule.Next(time.Now().In(loc)) - update["nextRunAt"] = nextRun - - collection := s.mongoDB.Database().Collection("schedules") - _, err = collection.UpdateByID(ctx, schedule.ID, bson.M{"$set": update}) - if err != nil { - return nil, fmt.Errorf("failed to update schedule: %w", err) - } - - // Re-register job if cron or timezone changed, or enable/disable - s.unregisterJob(scheduleID) - if schedule.Enabled { - schedule.NextRunAt = &nextRun - if err := s.registerJob(schedule); err != nil { - log.Printf("⚠️ Failed to re-register schedule: %v", err) - } - } - - return schedule, nil -} - -// DeleteSchedule deletes a schedule -func (s *SchedulerService) DeleteSchedule(ctx context.Context, scheduleID, userID string) error { - if s.mongoDB == nil { - return fmt.Errorf("MongoDB not available") - } - - objID, err := primitive.ObjectIDFromHex(scheduleID) - if err != nil { - return fmt.Errorf("invalid schedule ID") - } - - // Unregister from scheduler - s.unregisterJob(scheduleID) - - collection := s.mongoDB.Database().Collection("schedules") - result, err := collection.DeleteOne(ctx, bson.M{ - "_id": objID, - "userId": userID, - }) - - if err != nil { - return fmt.Errorf("failed to delete schedule: %w", err) - } - - if result.DeletedCount == 0 { - return fmt.Errorf("schedule not found") - } - - log.Printf("🗑️ Deleted schedule %s", scheduleID) - return nil -} - -// DeleteAllByUser deletes all schedules for a user (GDPR compliance) -func (s *SchedulerService) DeleteAllByUser(ctx context.Context, userID string) (int64, error) { - if s.mongoDB == nil { - return 0, nil // No MongoDB, no schedules to delete - } - - if userID == "" { - return 0, fmt.Errorf("user ID is required") - } - - // First, unregister all jobs for this user from the scheduler - collection := s.mongoDB.Database().Collection("schedules") - cursor, err := collection.Find(ctx, bson.M{"userId": userID}) - if err != nil { - return 0, fmt.Errorf("failed to find user schedules: %w", err) - } - defer cursor.Close(ctx) - - for cursor.Next(ctx) { - var schedule struct { - ID primitive.ObjectID `bson:"_id"` - } - if err := cursor.Decode(&schedule); err == nil { - s.unregisterJob(schedule.ID.Hex()) - } - } - - // Delete all schedules - result, err := collection.DeleteMany(ctx, bson.M{"userId": userID}) - if err != nil { - return 0, fmt.Errorf("failed to delete user schedules: %w", err) - } - - log.Printf("🗑️ [GDPR] Deleted %d schedules for user %s", result.DeletedCount, userID) - return result.DeletedCount, nil -} - -// TriggerNow triggers an immediate execution of a schedule -func (s *SchedulerService) TriggerNow(ctx context.Context, scheduleID, userID string) error { - schedule, err := s.GetSchedule(ctx, scheduleID, userID) - if err != nil { - return err - } - - // Execute in background - go s.executeScheduledJob(schedule) - - return nil -} - -// countUserSchedules counts the number of ENABLED schedules for a user -// Only enabled schedules count toward the limit - paused schedules don't consume quota -// This allows users to pause schedules to free up slots for new ones -func (s *SchedulerService) countUserSchedules(ctx context.Context, userID string) (int64, error) { - collection := s.mongoDB.Database().Collection("schedules") - return collection.CountDocuments(ctx, bson.M{"userId": userID, "enabled": true}) -} - -// ScheduleUsage represents the user's schedule usage stats -type ScheduleUsage struct { - Active int64 `json:"active"` - Paused int64 `json:"paused"` - Total int64 `json:"total"` - Limit int `json:"limit"` - CanCreate bool `json:"canCreate"` -} - -// GetScheduleUsage returns the user's schedule usage statistics -func (s *SchedulerService) GetScheduleUsage(ctx context.Context, userID string) (*ScheduleUsage, error) { - collection := s.mongoDB.Database().Collection("schedules") - - // Count active (enabled) schedules - active, err := collection.CountDocuments(ctx, bson.M{"userId": userID, "enabled": true}) - if err != nil { - return nil, fmt.Errorf("failed to count active schedules: %w", err) - } - - // Count paused (disabled) schedules - paused, err := collection.CountDocuments(ctx, bson.M{"userId": userID, "enabled": false}) - if err != nil { - return nil, fmt.Errorf("failed to count paused schedules: %w", err) - } - - // Get user's limit - limits := models.GetTierLimits(s.getUserTier(ctx, userID)) - limit := limits.MaxSchedules - - // Can create if active < limit (or limit is -1 for unlimited) - canCreate := limit < 0 || active < int64(limit) - - return &ScheduleUsage{ - Active: active, - Paused: paused, - Total: active + paused, - Limit: limit, - CanCreate: canCreate, - }, nil -} - -// getUserTier gets the user's subscription tier (placeholder - will be implemented with UserService) -func (s *SchedulerService) getUserTier(ctx context.Context, userID string) string { - // TODO: Look up user's tier from MongoDB - return "free" -} - -// InitializeIndexes creates the necessary indexes for the schedules collection -func (s *SchedulerService) InitializeIndexes(ctx context.Context) error { - if s.mongoDB == nil { - return nil - } - - collection := s.mongoDB.Database().Collection("schedules") - - indexes := []mongo.IndexModel{ - { - Keys: bson.D{{Key: "agentId", Value: 1}}, - Options: options.Index().SetUnique(true), - }, - { - Keys: bson.D{{Key: "userId", Value: 1}, {Key: "enabled", Value: 1}}, - }, - { - Keys: bson.D{{Key: "nextRunAt", Value: 1}, {Key: "enabled", Value: 1}}, - }, - } - - _, err := collection.Indexes().CreateMany(ctx, indexes) - if err != nil { - return fmt.Errorf("failed to create indexes: %w", err) - } - - log.Println("✅ Schedule indexes created") - return nil -} diff --git a/backend/internal/services/scraper_client.go b/backend/internal/services/scraper_client.go deleted file mode 100644 index c011e2ef..00000000 --- a/backend/internal/services/scraper_client.go +++ /dev/null @@ -1,76 +0,0 @@ -package services - -import ( - "context" - "fmt" - "net" - "net/http" - "time" -) - -// ScraperClient wraps an HTTP client with optimized settings for web scraping -type ScraperClient struct { - httpClient *http.Client - userAgent string - timeout time.Duration -} - -// NewScraperClient creates a new HTTP client optimized for web scraping -func NewScraperClient() *ScraperClient { - // Custom transport with optimized connection pooling - transport := &http.Transport{ - MaxIdleConns: 100, // Total idle connections across all hosts - MaxIdleConnsPerHost: 20, // CRITICAL: Default is 2! Increase for performance - MaxConnsPerHost: 50, // Maximum connections per host - IdleConnTimeout: 90 * time.Second, // Keep connections alive - TLSHandshakeTimeout: 10 * time.Second, - DisableCompression: false, - - // Dial settings for connection establishment - DialContext: (&net.Dialer{ - Timeout: 30 * time.Second, - KeepAlive: 30 * time.Second, - }).DialContext, - } - - return &ScraperClient{ - httpClient: &http.Client{ - Transport: transport, - Timeout: 60 * time.Second, // Overall request timeout - CheckRedirect: func(req *http.Request, via []*http.Request) error { - if len(via) >= 10 { - return fmt.Errorf("too many redirects (max 10)") - } - return nil - }, - }, - userAgent: "ClaraVerse-Bot/1.0 (+https://claraverse.example.com/bot)", - timeout: 60 * time.Second, - } -} - -// Get performs an HTTP GET request with proper headers -func (c *ScraperClient) Get(ctx context.Context, url string) (*http.Response, error) { - req, err := http.NewRequestWithContext(ctx, "GET", url, nil) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - // Set proper headers - req.Header.Set("User-Agent", c.userAgent) - req.Header.Set("Accept", "text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8") - req.Header.Set("Accept-Language", "en-US,en;q=0.9") - - return c.httpClient.Do(req) -} - -// SetUserAgent updates the user agent string -func (c *ScraperClient) SetUserAgent(userAgent string) { - c.userAgent = userAgent -} - -// SetTimeout updates the request timeout -func (c *ScraperClient) SetTimeout(timeout time.Duration) { - c.timeout = timeout - c.httpClient.Timeout = timeout -} diff --git a/backend/internal/services/scraper_ratelimit.go b/backend/internal/services/scraper_ratelimit.go deleted file mode 100644 index 27b7999e..00000000 --- a/backend/internal/services/scraper_ratelimit.go +++ /dev/null @@ -1,118 +0,0 @@ -package services - -import ( - "context" - "sync" - "time" - - "golang.org/x/time/rate" -) - -// RateLimiter implements three-tier rate limiting for web scraping -type RateLimiter struct { - globalLimiter *rate.Limiter // Overall requests/second for the server - perDomainLimiters *sync.Map // map[string]*rate.Limiter - per domain limits - perUserLimiters *sync.Map // map[string]*rate.Limiter - per user limits - mu sync.RWMutex -} - -// NewRateLimiter creates a new three-tier rate limiter -func NewRateLimiter(globalRate, perUserRate float64) *RateLimiter { - return &RateLimiter{ - globalLimiter: rate.NewLimiter(rate.Limit(globalRate), int(globalRate*2)), // 10 req/s, burst 20 - perDomainLimiters: &sync.Map{}, - perUserLimiters: &sync.Map{}, - } -} - -// Wait applies all three tiers of rate limiting -func (rl *RateLimiter) Wait(ctx context.Context, userID, domain string) error { - // Tier 1: Global rate limit (protect server resources) - if err := rl.globalLimiter.Wait(ctx); err != nil { - return err - } - - // Tier 2: Per-domain rate limit (respect target websites) - domainLimiter := rl.getOrCreateDomainLimiter(domain) - if err := domainLimiter.Wait(ctx); err != nil { - return err - } - - // Tier 3: Per-user rate limit (fair usage) - userLimiter := rl.getOrCreateUserLimiter(userID) - if err := userLimiter.Wait(ctx); err != nil { - return err - } - - return nil -} - -// WaitWithCrawlDelay applies rate limiting with respect to robots.txt crawl-delay -func (rl *RateLimiter) WaitWithCrawlDelay(ctx context.Context, userID, domain string, crawlDelay time.Duration) error { - // Tier 1: Global rate limit - if err := rl.globalLimiter.Wait(ctx); err != nil { - return err - } - - // Tier 2: Per-domain rate limit with crawl-delay - domainLimiter := rl.getOrCreateDomainLimiterWithDelay(domain, crawlDelay) - if err := domainLimiter.Wait(ctx); err != nil { - return err - } - - // Tier 3: Per-user rate limit - userLimiter := rl.getOrCreateUserLimiter(userID) - if err := userLimiter.Wait(ctx); err != nil { - return err - } - - return nil -} - -// getOrCreateDomainLimiter gets or creates a rate limiter for a domain (default 2 req/s) -func (rl *RateLimiter) getOrCreateDomainLimiter(domain string) *rate.Limiter { - return rl.getOrCreateDomainLimiterWithDelay(domain, 500*time.Millisecond) -} - -// getOrCreateDomainLimiterWithDelay gets or creates a rate limiter for a domain with custom delay -func (rl *RateLimiter) getOrCreateDomainLimiterWithDelay(domain string, crawlDelay time.Duration) *rate.Limiter { - if limiter, ok := rl.perDomainLimiters.Load(domain); ok { - return limiter.(*rate.Limiter) - } - - // Create new limiter based on crawl delay - requestsPerSecond := 1.0 / crawlDelay.Seconds() - if requestsPerSecond > 5.0 { - requestsPerSecond = 5.0 // Cap at 5 req/s - } - if requestsPerSecond < 0.2 { - requestsPerSecond = 0.2 // Minimum 1 request per 5 seconds - } - - newLimiter := rate.NewLimiter(rate.Limit(requestsPerSecond), 1) - - // Try to store, but use existing if another goroutine created it first - actual, _ := rl.perDomainLimiters.LoadOrStore(domain, newLimiter) - return actual.(*rate.Limiter) -} - -// getOrCreateUserLimiter gets or creates a rate limiter for a user (5 req/s) -func (rl *RateLimiter) getOrCreateUserLimiter(userID string) *rate.Limiter { - if limiter, ok := rl.perUserLimiters.Load(userID); ok { - return limiter.(*rate.Limiter) - } - - newLimiter := rate.NewLimiter(rate.Limit(5.0), 10) // 5 req/s, burst 10 - - // Try to store, but use existing if another goroutine created it first - actual, _ := rl.perUserLimiters.LoadOrStore(userID, newLimiter) - return actual.(*rate.Limiter) -} - -// SetGlobalRate updates the global rate limit -func (rl *RateLimiter) SetGlobalRate(requestsPerSecond float64) { - rl.mu.Lock() - defer rl.mu.Unlock() - rl.globalLimiter.SetLimit(rate.Limit(requestsPerSecond)) - rl.globalLimiter.SetBurst(int(requestsPerSecond * 2)) -} diff --git a/backend/internal/services/scraper_resource.go b/backend/internal/services/scraper_resource.go deleted file mode 100644 index 9c2ebe29..00000000 --- a/backend/internal/services/scraper_resource.go +++ /dev/null @@ -1,52 +0,0 @@ -package services - -import ( - "context" - "fmt" - "io" -) - -// ResourceManager handles resource limits for concurrent scraping -type ResourceManager struct { - semaphore chan struct{} // Limit concurrent requests - maxBodySize int64 // Max response body size in bytes -} - -// NewResourceManager creates a new resource manager -func NewResourceManager(maxConcurrent int, maxBodySize int64) *ResourceManager { - return &ResourceManager{ - semaphore: make(chan struct{}, maxConcurrent), - maxBodySize: maxBodySize, - } -} - -// Acquire acquires a slot for a scraping operation -func (rm *ResourceManager) Acquire(ctx context.Context) error { - select { - case rm.semaphore <- struct{}{}: - return nil - case <-ctx.Done(): - return fmt.Errorf("context cancelled while waiting for resource: %w", ctx.Err()) - } -} - -// Release releases a slot after scraping completes -func (rm *ResourceManager) Release() { - <-rm.semaphore -} - -// ReadBody reads the response body with size limit to prevent memory exhaustion -func (rm *ResourceManager) ReadBody(body io.Reader) ([]byte, error) { - limitedReader := io.LimitReader(body, rm.maxBodySize) - data, err := io.ReadAll(limitedReader) - if err != nil { - return nil, fmt.Errorf("failed to read body: %w", err) - } - - // Check if we hit the limit - if int64(len(data)) >= rm.maxBodySize { - return nil, fmt.Errorf("response body too large (max %d bytes)", rm.maxBodySize) - } - - return data, nil -} diff --git a/backend/internal/services/scraper_robots.go b/backend/internal/services/scraper_robots.go deleted file mode 100644 index 9085c7a3..00000000 --- a/backend/internal/services/scraper_robots.go +++ /dev/null @@ -1,114 +0,0 @@ -package services - -import ( - "context" - "fmt" - "io" - "net/http" - "net/url" - "time" - - cache "github.com/patrickmn/go-cache" - "github.com/temoto/robotstxt" -) - -// RobotsChecker handles robots.txt fetching and compliance checking -type RobotsChecker struct { - cache *cache.Cache - userAgent string - client *http.Client -} - -// NewRobotsChecker creates a new robots.txt checker -func NewRobotsChecker(userAgent string) *RobotsChecker { - return &RobotsChecker{ - cache: cache.New(24*time.Hour, 1*time.Hour), // Cache robots.txt for 24 hours - userAgent: userAgent, - client: &http.Client{ - Timeout: 10 * time.Second, - }, - } -} - -// CanFetch checks if the URL can be fetched according to robots.txt -// Returns (allowed bool, crawlDelay time.Duration, error) -func (rc *RobotsChecker) CanFetch(ctx context.Context, urlStr string) (bool, time.Duration, error) { - parsedURL, err := url.Parse(urlStr) - if err != nil { - return false, 0, fmt.Errorf("invalid URL: %w", err) - } - - domain := parsedURL.Scheme + "://" + parsedURL.Host - robotsURL := domain + "/robots.txt" - - // Check cache first - if cached, found := rc.cache.Get(domain); found { - robotsData := cached.(*robotstxt.RobotsData) - group := robotsData.FindGroup(rc.userAgent) - allowed := group.Test(parsedURL.Path) - crawlDelay := rc.getCrawlDelay(group) - return allowed, crawlDelay, nil - } - - // Fetch robots.txt - req, err := http.NewRequestWithContext(ctx, "GET", robotsURL, nil) - if err != nil { - return false, 0, fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("User-Agent", rc.userAgent) - - resp, err := rc.client.Do(req) - if err != nil { - // If robots.txt doesn't exist or network error, allow by default - return true, 1 * time.Second, nil - } - defer resp.Body.Close() - - // If robots.txt returns 404 or other error, allow by default - if resp.StatusCode != http.StatusOK { - return true, 1 * time.Second, nil - } - - // Read and parse robots.txt - body, err := io.ReadAll(io.LimitReader(resp.Body, 1*1024*1024)) // Max 1MB - if err != nil { - return true, 1 * time.Second, nil - } - - robotsData, err := robotstxt.FromBytes(body) - if err != nil { - // If parsing fails, be conservative and allow - return true, 1 * time.Second, nil - } - - // Cache the robots.txt data - rc.cache.Set(domain, robotsData, cache.DefaultExpiration) - - // Check if path is allowed - group := robotsData.FindGroup(rc.userAgent) - allowed := group.Test(parsedURL.Path) - crawlDelay := rc.getCrawlDelay(group) - - return allowed, crawlDelay, nil -} - -// getCrawlDelay extracts crawl delay from robots.txt group -func (rc *RobotsChecker) getCrawlDelay(group *robotstxt.Group) time.Duration { - if group.CrawlDelay > 0 { - delay := time.Duration(group.CrawlDelay) * time.Second - // Cap at maximum 10 seconds - if delay > 10*time.Second { - delay = 10 * time.Second - } - return delay - } - - // Default to 1 second if no crawl delay specified - return 1 * time.Second -} - -// SetUserAgent updates the user agent string -func (rc *RobotsChecker) SetUserAgent(userAgent string) { - rc.userAgent = userAgent -} diff --git a/backend/internal/services/scraper_service.go b/backend/internal/services/scraper_service.go deleted file mode 100644 index 56c4fb4e..00000000 --- a/backend/internal/services/scraper_service.go +++ /dev/null @@ -1,232 +0,0 @@ -package services - -import ( - "bytes" - "context" - "fmt" - "log" - "net/url" - "strings" - "sync" - "time" - - "github.com/markusmobius/go-trafilatura" - cache "github.com/patrickmn/go-cache" -) - -const ( - defaultUserAgent = "ClaraVerse-Bot/1.0 (+https://claraverse.example.com/bot)" - defaultMaxBodySize = 10 * 1024 * 1024 // 10MB - defaultMaxConcurrent = 10 - defaultGlobalRate = 10.0 // requests per second - defaultPerUserRate = 5.0 // requests per second -) - -// ScraperService handles web scraping operations -type ScraperService struct { - client *ScraperClient - rateLimiter *RateLimiter - robotsChecker *RobotsChecker - contentCache *cache.Cache - resourceMgr *ResourceManager -} - -var ( - scraperInstance *ScraperService - scraperOnce sync.Once -) - -// GetScraperService returns the singleton scraper service instance -func GetScraperService() *ScraperService { - scraperOnce.Do(func() { - scraperInstance = &ScraperService{ - client: NewScraperClient(), - rateLimiter: NewRateLimiter(defaultGlobalRate, defaultPerUserRate), - robotsChecker: NewRobotsChecker(defaultUserAgent), - contentCache: cache.New(1*time.Hour, 10*time.Minute), // Cache for 1 hour - resourceMgr: NewResourceManager(defaultMaxConcurrent, defaultMaxBodySize), - } - - log.Printf("✅ [SCRAPER] Service initialized: max_concurrent=%d, global_rate=%.1f req/s", - defaultMaxConcurrent, defaultGlobalRate) - }) - return scraperInstance -} - -// ScrapeURL scrapes a web page and returns clean content -func (s *ScraperService) ScrapeURL(ctx context.Context, urlStr, format string, maxLength int, userID string) (string, error) { - startTime := time.Now() - - // 1. Validate URL - if err := s.validateURL(urlStr); err != nil { - return "", err - } - - parsedURL, err := url.Parse(urlStr) - if err != nil { - return "", fmt.Errorf("invalid URL: %w", err) - } - - domain := parsedURL.Host - - // 2. Check cache - cacheKey := s.getCacheKey(urlStr, format) - if cached, found := s.contentCache.Get(cacheKey); found { - log.Printf("✅ [SCRAPER] Cache hit for URL: %s (latency: %dms)", - urlStr, time.Since(startTime).Milliseconds()) - return cached.(string), nil - } - - // 3. Check robots.txt - allowed, crawlDelay, err := s.robotsChecker.CanFetch(ctx, urlStr) - if err != nil { - log.Printf("⚠️ [SCRAPER] Failed to check robots.txt for %s: %v", urlStr, err) - // Continue anyway with default delay - crawlDelay = 1 * time.Second - } - - if !allowed { - return "", fmt.Errorf("access blocked by robots.txt for: %s", urlStr) - } - - // 4. Apply rate limiting - if err := s.rateLimiter.WaitWithCrawlDelay(ctx, userID, domain, crawlDelay); err != nil { - return "", fmt.Errorf("rate limit error: %w", err) - } - - // 5. Acquire resource semaphore - if err := s.resourceMgr.Acquire(ctx); err != nil { - return "", fmt.Errorf("resource limit reached, try again later: %w", err) - } - defer s.resourceMgr.Release() - - // 6. Fetch URL - resp, err := s.client.Get(ctx, urlStr) - if err != nil { - log.Printf("❌ [SCRAPER] Failed to fetch URL %s: %v", urlStr, err) - return "", fmt.Errorf("failed to fetch URL: %w", err) - } - defer resp.Body.Close() - - // 7. Check HTTP status - if resp.StatusCode != 200 { - return "", fmt.Errorf("HTTP error %d: %s", resp.StatusCode, resp.Status) - } - - // 8. Check content type - contentType := resp.Header.Get("Content-Type") - if !s.isSupportedContentType(contentType) { - return "", fmt.Errorf("unsupported content type: %s", contentType) - } - - // 9. Read body with size limit - body, err := s.resourceMgr.ReadBody(resp.Body) - if err != nil { - return "", fmt.Errorf("failed to read response: %w", err) - } - - // 10. Extract main content using trafilatura - opts := trafilatura.Options{ - OriginalURL: parsedURL, - } - - result, err := trafilatura.Extract(bytes.NewReader(body), opts) - if err != nil { - return "", fmt.Errorf("failed to extract content: %w", err) - } - - if result == nil || result.ContentText == "" { - return "", fmt.Errorf("no content extracted from page") - } - - // 11. Use extracted content (already plain text or markdown) - content := result.ContentText - - // 12. Apply length limit - if len(content) > maxLength { - content = content[:maxLength] + "\n\n[Content truncated due to length limit]" - } - - // 13. Add metadata header - metadata := fmt.Sprintf("# %s\n\n", result.Metadata.Title) - if result.Metadata.Author != "" { - metadata += fmt.Sprintf("**Author:** %s \n", result.Metadata.Author) - } - if !result.Metadata.Date.IsZero() { - metadata += fmt.Sprintf("**Published:** %s \n", result.Metadata.Date.Format("January 2, 2006")) - } - metadata += fmt.Sprintf("**Source:** %s \n", urlStr) - metadata += "\n---\n\n" - - finalContent := metadata + content - - // 14. Cache result - s.contentCache.Set(cacheKey, finalContent, cache.DefaultExpiration) - - latency := time.Since(startTime).Milliseconds() - log.Printf("✅ [SCRAPER] Successfully scraped URL: %s (latency: %dms, length: %d chars)", - urlStr, latency, len(finalContent)) - - return finalContent, nil -} - -// validateURL checks if the URL is safe to scrape (SSRF protection) -func (s *ScraperService) validateURL(urlStr string) error { - parsedURL, err := url.Parse(urlStr) - if err != nil { - return fmt.Errorf("invalid URL format: %w", err) - } - - // Only allow HTTP and HTTPS - if parsedURL.Scheme != "http" && parsedURL.Scheme != "https" { - return fmt.Errorf("only HTTP/HTTPS URLs are supported, got: %s", parsedURL.Scheme) - } - - hostname := strings.ToLower(parsedURL.Hostname()) - - // Block localhost - if hostname == "localhost" || hostname == "127.0.0.1" || hostname == "::1" { - return fmt.Errorf("localhost URLs are not allowed") - } - - // Block private IP ranges - privateRanges := []string{ - "192.168.", "10.", "172.16.", "172.17.", "172.18.", "172.19.", - "172.20.", "172.21.", "172.22.", "172.23.", "172.24.", "172.25.", - "172.26.", "172.27.", "172.28.", "172.29.", "172.30.", "172.31.", - "169.254.", // Link-local - "fd", // IPv6 private - } - - for _, prefix := range privateRanges { - if strings.HasPrefix(hostname, prefix) { - return fmt.Errorf("private IP addresses are not allowed") - } - } - - return nil -} - -// isSupportedContentType checks if the content type is supported -func (s *ScraperService) isSupportedContentType(contentType string) bool { - contentType = strings.ToLower(contentType) - - supported := []string{ - "text/html", - "text/plain", - "application/xhtml+xml", - } - - for _, ct := range supported { - if strings.Contains(contentType, ct) { - return true - } - } - - return false -} - -// getCacheKey generates a cache key for URL and format -func (s *ScraperService) getCacheKey(urlStr, format string) string { - return fmt.Sprintf("%s:%s", urlStr, format) -} diff --git a/backend/internal/services/stream_buffer_service.go b/backend/internal/services/stream_buffer_service.go deleted file mode 100644 index 52b64af2..00000000 --- a/backend/internal/services/stream_buffer_service.go +++ /dev/null @@ -1,389 +0,0 @@ -package services - -import ( - "errors" - "log" - "strings" - "sync" - "time" -) - -// Stream buffer constants for production safety -const ( - MaxChunksPerBuffer = 10000 // Prevent memory exhaustion - MaxBufferSize = 1 << 20 // 1MB max per buffer - DefaultBufferTTL = 2 * time.Minute - CleanupInterval = 30 * time.Second -) - -// Error types for stream buffer operations -var ( - ErrBufferNotFound = errors.New("stream buffer not found") - ErrBufferFull = errors.New("stream buffer full: max chunks exceeded") - ErrBufferSizeExceeded = errors.New("stream buffer size exceeded") - ErrResumeTooFast = errors.New("resume rate limit exceeded") -) - -// BufferedMessage represents a message that needs to be delivered on reconnect -type BufferedMessage struct { - Type string `json:"type"` - Content string `json:"content,omitempty"` - ToolName string `json:"tool_name,omitempty"` - ToolDisplayName string `json:"tool_display_name,omitempty"` - ToolIcon string `json:"tool_icon,omitempty"` - ToolDescription string `json:"tool_description,omitempty"` - Status string `json:"status,omitempty"` - Arguments map[string]interface{} `json:"arguments,omitempty"` - Result string `json:"result,omitempty"` - Plots interface{} `json:"plots,omitempty"` // For image artifacts - Delivered bool `json:"-"` // Track if already delivered (not serialized) -} - -// StreamBuffer holds buffered chunks for a streaming conversation -type StreamBuffer struct { - ConversationID string - UserID string - ConnID string // Original connection ID - Chunks []string // Buffered text chunks - PendingMessages []BufferedMessage // Important messages (tool_result, etc.) to deliver on reconnect - TotalSize int // Current total size of all chunks - IsComplete bool // Generation finished? - FullContent string // Full content if complete - CreatedAt time.Time - LastChunkAt time.Time // Last chunk received time - ResumeCount int // Track resume attempts - LastResume time.Time // Prevent rapid resume spam - mutex sync.Mutex -} - -// StreamBufferService manages stream buffers for disconnected clients -type StreamBufferService struct { - buffers map[string]*StreamBuffer // conversationID -> buffer - mutex sync.RWMutex - ttl time.Duration - cleanupTick *time.Ticker - done chan struct{} -} - -// NewStreamBufferService creates a new stream buffer service -func NewStreamBufferService() *StreamBufferService { - svc := &StreamBufferService{ - buffers: make(map[string]*StreamBuffer), - ttl: DefaultBufferTTL, - cleanupTick: time.NewTicker(CleanupInterval), - done: make(chan struct{}), - } - go svc.cleanupLoop() - log.Println("📦 StreamBufferService initialized") - return svc -} - -// cleanupLoop periodically removes expired buffers -func (s *StreamBufferService) cleanupLoop() { - for { - select { - case <-s.done: - return - case <-s.cleanupTick.C: - s.cleanup() - } - } -} - -// cleanup removes expired buffers -func (s *StreamBufferService) cleanup() { - s.mutex.Lock() - defer s.mutex.Unlock() - - now := time.Now() - expired := 0 - for convID, buf := range s.buffers { - if now.Sub(buf.CreatedAt) > s.ttl { - delete(s.buffers, convID) - expired++ - log.Printf("📦 Buffer expired for conversation %s", convID) - } - } - if expired > 0 { - log.Printf("📦 Cleaned up %d expired buffers, %d active", expired, len(s.buffers)) - } -} - -// Shutdown gracefully shuts down the service -func (s *StreamBufferService) Shutdown() { - close(s.done) - s.cleanupTick.Stop() - s.mutex.Lock() - defer s.mutex.Unlock() - s.buffers = nil - log.Println("📦 StreamBufferService shutdown complete") -} - -// CreateBuffer creates a new buffer for a conversation -func (s *StreamBufferService) CreateBuffer(conversationID, userID, connID string) { - s.mutex.Lock() - defer s.mutex.Unlock() - - // If buffer already exists, don't overwrite (prevents race conditions) - if _, exists := s.buffers[conversationID]; exists { - log.Printf("📦 Buffer already exists for conversation %s", conversationID) - return - } - - s.buffers[conversationID] = &StreamBuffer{ - ConversationID: conversationID, - UserID: userID, - ConnID: connID, - Chunks: make([]string, 0, 100), // Pre-allocate for performance - PendingMessages: make([]BufferedMessage, 0, 10), // For tool results, etc. - TotalSize: 0, - CreatedAt: time.Now(), - LastChunkAt: time.Now(), - } - log.Printf("📦 Buffer created for conversation %s (user: %s)", conversationID, userID) -} - -// AppendChunk adds a chunk to the buffer -func (s *StreamBufferService) AppendChunk(conversationID, chunk string) error { - s.mutex.RLock() - buf, exists := s.buffers[conversationID] - s.mutex.RUnlock() - - if !exists { - // Buffer doesn't exist - this is normal if streaming started before disconnect - return nil - } - - buf.mutex.Lock() - defer buf.mutex.Unlock() - - // Safety limits - if len(buf.Chunks) >= MaxChunksPerBuffer { - log.Printf("⚠️ Buffer full for conversation %s (max chunks: %d)", conversationID, MaxChunksPerBuffer) - return ErrBufferFull - } - - if buf.TotalSize+len(chunk) > MaxBufferSize { - log.Printf("⚠️ Buffer size exceeded for conversation %s (max: %d bytes)", conversationID, MaxBufferSize) - return ErrBufferSizeExceeded - } - - buf.Chunks = append(buf.Chunks, chunk) - buf.TotalSize += len(chunk) - buf.LastChunkAt = time.Now() - - return nil -} - -// AppendMessage adds an important message to the buffer for delivery on reconnect -// This is used for tool_result, artifacts, and other critical messages that shouldn't be lost -func (s *StreamBufferService) AppendMessage(conversationID string, msg BufferedMessage) error { - s.mutex.RLock() - buf, exists := s.buffers[conversationID] - s.mutex.RUnlock() - - if !exists { - // Buffer doesn't exist - this is normal if streaming started before disconnect - return nil - } - - buf.mutex.Lock() - defer buf.mutex.Unlock() - - // Limit pending messages to prevent memory issues - if len(buf.PendingMessages) >= 50 { - log.Printf("⚠️ Too many pending messages for conversation %s", conversationID) - return ErrBufferFull - } - - buf.PendingMessages = append(buf.PendingMessages, msg) - buf.LastChunkAt = time.Now() - - log.Printf("📦 Buffered message type=%s for conversation %s (pending: %d)", - msg.Type, conversationID, len(buf.PendingMessages)) - - return nil -} - -// MarkMessagesDelivered marks all pending messages as delivered to prevent duplicate replay -func (s *StreamBufferService) MarkMessagesDelivered(conversationID string) { - s.mutex.RLock() - buf, exists := s.buffers[conversationID] - s.mutex.RUnlock() - - if !exists { - return - } - - buf.mutex.Lock() - defer buf.mutex.Unlock() - - for i := range buf.PendingMessages { - buf.PendingMessages[i].Delivered = true - } - log.Printf("📦 Marked %d pending messages as delivered for conversation %s", len(buf.PendingMessages), conversationID) -} - -// MarkComplete marks the buffer as complete with the full content -func (s *StreamBufferService) MarkComplete(conversationID, fullContent string) { - s.mutex.RLock() - buf, exists := s.buffers[conversationID] - s.mutex.RUnlock() - - if !exists { - return - } - - buf.mutex.Lock() - defer buf.mutex.Unlock() - - buf.IsComplete = true - buf.FullContent = fullContent - log.Printf("📦 Buffer marked complete for conversation %s (size: %d bytes)", conversationID, len(fullContent)) -} - -// GetBuffer retrieves a buffer without clearing it (allows multiple resume attempts) -func (s *StreamBufferService) GetBuffer(conversationID string) (*StreamBuffer, error) { - s.mutex.Lock() - defer s.mutex.Unlock() - - buf, exists := s.buffers[conversationID] - if !exists { - return nil, ErrBufferNotFound - } - - // Rate limit: 1 resume per second - if time.Since(buf.LastResume) < time.Second { - return nil, ErrResumeTooFast - } - - buf.ResumeCount++ - buf.LastResume = time.Now() - - log.Printf("📦 Buffer retrieved for conversation %s (resume #%d, chunks: %d)", - conversationID, buf.ResumeCount, len(buf.Chunks)) - - return buf, nil -} - -// ClearBuffer removes a buffer after successful resume -func (s *StreamBufferService) ClearBuffer(conversationID string) { - s.mutex.Lock() - defer s.mutex.Unlock() - - if _, exists := s.buffers[conversationID]; exists { - delete(s.buffers, conversationID) - log.Printf("📦 Buffer cleared for conversation %s", conversationID) - } -} - -// HasBuffer checks if a buffer exists for a conversation -func (s *StreamBufferService) HasBuffer(conversationID string) bool { - s.mutex.RLock() - defer s.mutex.RUnlock() - _, exists := s.buffers[conversationID] - return exists -} - -// GetBufferStats returns statistics about the buffer service -func (s *StreamBufferService) GetBufferStats() map[string]interface{} { - s.mutex.RLock() - defer s.mutex.RUnlock() - - totalChunks := 0 - totalSize := 0 - for _, buf := range s.buffers { - buf.mutex.Lock() - totalChunks += len(buf.Chunks) - totalSize += buf.TotalSize - buf.mutex.Unlock() - } - - return map[string]interface{}{ - "active_buffers": len(s.buffers), - "total_chunks": totalChunks, - "total_size": totalSize, - } -} - -// GetBufferInfo returns detailed info about a specific buffer (for debugging) -func (s *StreamBufferService) GetBufferInfo(conversationID string) map[string]interface{} { - s.mutex.RLock() - buf, exists := s.buffers[conversationID] - s.mutex.RUnlock() - - if !exists { - return nil - } - - buf.mutex.Lock() - defer buf.mutex.Unlock() - - return map[string]interface{}{ - "conversation_id": buf.ConversationID, - "user_id": buf.UserID, - "conn_id": buf.ConnID, - "chunk_count": len(buf.Chunks), - "total_size": buf.TotalSize, - "is_complete": buf.IsComplete, - "created_at": buf.CreatedAt, - "last_chunk_at": buf.LastChunkAt, - "resume_count": buf.ResumeCount, - "age_seconds": time.Since(buf.CreatedAt).Seconds(), - } -} - -// BufferData represents the data needed for stream resume -type BufferData struct { - ConversationID string - UserID string - CombinedChunks string - IsComplete bool - ChunkCount int - PendingMessages []BufferedMessage // Tool results, artifacts, etc. to replay -} - -// GetBufferData safely retrieves buffer data for resume operations -func (s *StreamBufferService) GetBufferData(conversationID string) (*BufferData, error) { - s.mutex.Lock() - defer s.mutex.Unlock() - - buf, exists := s.buffers[conversationID] - if !exists { - return nil, ErrBufferNotFound - } - - // Rate limit: 1 resume per second - if time.Since(buf.LastResume) < time.Second { - return nil, ErrResumeTooFast - } - - buf.ResumeCount++ - buf.LastResume = time.Now() - - buf.mutex.Lock() - defer buf.mutex.Unlock() - - // Combine all chunks - var combined strings.Builder - for _, chunk := range buf.Chunks { - combined.WriteString(chunk) - } - - // Copy only undelivered pending messages - var pendingMsgs []BufferedMessage - for _, msg := range buf.PendingMessages { - if !msg.Delivered { - pendingMsgs = append(pendingMsgs, msg) - } - } - - return &BufferData{ - ConversationID: buf.ConversationID, - UserID: buf.UserID, - CombinedChunks: combined.String(), - IsComplete: buf.IsComplete, - ChunkCount: len(buf.Chunks), - PendingMessages: pendingMsgs, - }, nil -} diff --git a/backend/internal/services/stream_buffer_service_test.go b/backend/internal/services/stream_buffer_service_test.go deleted file mode 100644 index 943d65fc..00000000 --- a/backend/internal/services/stream_buffer_service_test.go +++ /dev/null @@ -1,286 +0,0 @@ -package services - -import ( - "fmt" - "strings" - "sync" - "testing" - "time" -) - -func TestStreamBuffer_CreateAndAppend(t *testing.T) { - svc := NewStreamBufferService() - defer svc.Shutdown() - - convID := "test-conv-123" - userID := "user-456" - connID := "conn-789" - - // Create buffer - svc.CreateBuffer(convID, userID, connID) - - // Verify buffer exists - if !svc.HasBuffer(convID) { - t.Fatal("Buffer should exist after creation") - } - - // Append chunks - chunks := []string{"Hello", " ", "World", "!"} - for _, chunk := range chunks { - err := svc.AppendChunk(convID, chunk) - if err != nil { - t.Fatalf("Failed to append chunk: %v", err) - } - } - - // Get buffer data - data, err := svc.GetBufferData(convID) - if err != nil { - t.Fatalf("Failed to get buffer data: %v", err) - } - - expected := "Hello World!" - if data.CombinedChunks != expected { - t.Errorf("Expected combined chunks %q, got %q", expected, data.CombinedChunks) - } - - if data.ChunkCount != 4 { - t.Errorf("Expected 4 chunks, got %d", data.ChunkCount) - } - - if data.UserID != userID { - t.Errorf("Expected userID %q, got %q", userID, data.UserID) - } -} - -func TestStreamBuffer_MarkComplete(t *testing.T) { - svc := NewStreamBufferService() - defer svc.Shutdown() - - convID := "test-conv-complete" - userID := "user-456" - connID := "conn-789" - - svc.CreateBuffer(convID, userID, connID) - svc.AppendChunk(convID, "Test content") - svc.MarkComplete(convID, "Full test content") - - data, err := svc.GetBufferData(convID) - if err != nil { - t.Fatalf("Failed to get buffer data: %v", err) - } - - if !data.IsComplete { - t.Error("Buffer should be marked as complete") - } -} - -func TestStreamBuffer_ClearBuffer(t *testing.T) { - svc := NewStreamBufferService() - defer svc.Shutdown() - - convID := "test-conv-clear" - svc.CreateBuffer(convID, "user", "conn") - svc.AppendChunk(convID, "Test") - - // Verify exists - if !svc.HasBuffer(convID) { - t.Fatal("Buffer should exist before clear") - } - - // Clear - svc.ClearBuffer(convID) - - // Verify gone - if svc.HasBuffer(convID) { - t.Error("Buffer should not exist after clear") - } - - // GetBufferData should return error - _, err := svc.GetBufferData(convID) - if err != ErrBufferNotFound { - t.Errorf("Expected ErrBufferNotFound, got %v", err) - } -} - -func TestStreamBuffer_MemoryLimits(t *testing.T) { - svc := NewStreamBufferService() - defer svc.Shutdown() - - svc.CreateBuffer("conv-1", "user-1", "conn-1") - - // Try to exceed max chunks - for i := 0; i < MaxChunksPerBuffer+10; i++ { - err := svc.AppendChunk("conv-1", "x") - if i >= MaxChunksPerBuffer { - if err != ErrBufferFull { - t.Errorf("Expected ErrBufferFull at chunk %d, got %v", i, err) - } - } else { - if err != nil { - t.Errorf("Unexpected error at chunk %d: %v", i, err) - } - } - } -} - -func TestStreamBuffer_SizeLimit(t *testing.T) { - svc := NewStreamBufferService() - defer svc.Shutdown() - - svc.CreateBuffer("conv-size", "user", "conn") - - // Create a large chunk that exceeds the size limit - largeChunk := strings.Repeat("x", MaxBufferSize+1) - err := svc.AppendChunk("conv-size", largeChunk) - if err != ErrBufferSizeExceeded { - t.Errorf("Expected ErrBufferSizeExceeded, got %v", err) - } -} - -func TestStreamBuffer_RateLimiting(t *testing.T) { - svc := NewStreamBufferService() - defer svc.Shutdown() - - svc.CreateBuffer("conv-rate", "user", "conn") - svc.AppendChunk("conv-rate", "test") - - // First GetBufferData should succeed - _, err := svc.GetBufferData("conv-rate") - if err != nil { - t.Fatalf("First GetBufferData should succeed: %v", err) - } - - // Immediate second call should be rate limited - _, err = svc.GetBufferData("conv-rate") - if err != ErrResumeTooFast { - t.Errorf("Expected ErrResumeTooFast, got %v", err) - } - - // Wait for rate limit to expire - time.Sleep(1100 * time.Millisecond) - - // Should succeed now - _, err = svc.GetBufferData("conv-rate") - if err != nil { - t.Errorf("GetBufferData should succeed after rate limit: %v", err) - } -} - -func TestStreamBuffer_ConcurrentAccess(t *testing.T) { - svc := NewStreamBufferService() - defer svc.Shutdown() - - svc.CreateBuffer("conv-concurrent", "user", "conn") - - // Concurrent writes - var wg sync.WaitGroup - numGoroutines := 100 - - for i := 0; i < numGoroutines; i++ { - wg.Add(1) - go func(idx int) { - defer wg.Done() - svc.AppendChunk("conv-concurrent", fmt.Sprintf("chunk-%d-", idx)) - }(i) - } - wg.Wait() - - data, err := svc.GetBufferData("conv-concurrent") - if err != nil { - t.Fatalf("Failed to get buffer data: %v", err) - } - - if data.ChunkCount != numGoroutines { - t.Errorf("Expected %d chunks, got %d", numGoroutines, data.ChunkCount) - } -} - -func TestStreamBuffer_NonExistentBuffer(t *testing.T) { - svc := NewStreamBufferService() - defer svc.Shutdown() - - // Append to non-existent buffer should not error (just no-op) - err := svc.AppendChunk("non-existent", "test") - if err != nil { - t.Errorf("AppendChunk to non-existent buffer should not error: %v", err) - } - - // GetBufferData should return error - _, err = svc.GetBufferData("non-existent") - if err != ErrBufferNotFound { - t.Errorf("Expected ErrBufferNotFound, got %v", err) - } -} - -func TestStreamBuffer_DuplicateCreate(t *testing.T) { - svc := NewStreamBufferService() - defer svc.Shutdown() - - convID := "conv-duplicate" - - // Create buffer - svc.CreateBuffer(convID, "user-1", "conn-1") - svc.AppendChunk(convID, "original") - - // Try to create again - should not overwrite - svc.CreateBuffer(convID, "user-2", "conn-2") - - data, err := svc.GetBufferData(convID) - if err != nil { - t.Fatalf("Failed to get buffer data: %v", err) - } - - // Should still have original user - if data.UserID != "user-1" { - t.Errorf("Buffer should not be overwritten, expected user-1, got %s", data.UserID) - } - - // Should still have original content - if data.CombinedChunks != "original" { - t.Errorf("Buffer content should not be overwritten") - } -} - -func TestStreamBuffer_Stats(t *testing.T) { - svc := NewStreamBufferService() - defer svc.Shutdown() - - // Create multiple buffers - svc.CreateBuffer("conv-1", "user", "conn") - svc.CreateBuffer("conv-2", "user", "conn") - svc.CreateBuffer("conv-3", "user", "conn") - - svc.AppendChunk("conv-1", "hello") - svc.AppendChunk("conv-2", "world") - svc.AppendChunk("conv-3", "!") - - stats := svc.GetBufferStats() - - activeBuffers := stats["active_buffers"].(int) - if activeBuffers != 3 { - t.Errorf("Expected 3 active buffers, got %d", activeBuffers) - } - - totalChunks := stats["total_chunks"].(int) - if totalChunks != 3 { - t.Errorf("Expected 3 total chunks, got %d", totalChunks) - } -} - -func TestStreamBuffer_Shutdown(t *testing.T) { - svc := NewStreamBufferService() - svc.CreateBuffer("conv-shutdown", "user", "conn") - svc.AppendChunk("conv-shutdown", "test") - - // Verify exists - if !svc.HasBuffer("conv-shutdown") { - t.Fatal("Buffer should exist before shutdown") - } - - // Shutdown - svc.Shutdown() - - // After shutdown, HasBuffer should return false (buffers is nil) - // This is a simple check - in production, you'd want more robust handling -} diff --git a/backend/internal/services/tier_service.go b/backend/internal/services/tier_service.go deleted file mode 100644 index a9719718..00000000 --- a/backend/internal/services/tier_service.go +++ /dev/null @@ -1,243 +0,0 @@ -package services - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "context" - "log" - "sync" - "time" - - "go.mongodb.org/mongo-driver/bson" -) - -// CacheEntry stores cached tier with expiration info for TTL-based invalidation -type CacheEntry struct { - Tier string - ExpiresAt *time.Time // For promo users, this is subscriptionExpiresAt; nil for regular users - CachedAt time.Time -} - -// TierService manages subscription tier limits and lookups -type TierService struct { - mongoDB *database.MongoDB - cache map[string]CacheEntry // userID -> CacheEntry with TTL info - mu sync.RWMutex - defaultTTL time.Duration // Default cache TTL for non-promo users -} - -// NewTierService creates a new tier service -func NewTierService(mongoDB *database.MongoDB) *TierService { - return &TierService{ - mongoDB: mongoDB, - cache: make(map[string]CacheEntry), - defaultTTL: 5 * time.Minute, - } -} - -// GetUserTier returns the subscription tier for a user -func (s *TierService) GetUserTier(ctx context.Context, userID string) string { - now := time.Now() - - // Check cache first - s.mu.RLock() - if entry, ok := s.cache[userID]; ok { - // Check if cache entry is still valid - // For promo users: check if promo has expired - if entry.ExpiresAt != nil && entry.ExpiresAt.Before(now) { - s.mu.RUnlock() - // Promo expired - invalidate cache and re-fetch - s.InvalidateCache(userID) - log.Printf("🔄 [TIER] Promo expired for user %s, re-fetching tier", userID) - return s.fetchAndCacheTier(ctx, userID) - } - - // For non-promo users: check default TTL - if entry.ExpiresAt == nil && now.Sub(entry.CachedAt) > s.defaultTTL { - s.mu.RUnlock() - // Cache TTL exceeded - re-fetch - s.InvalidateCache(userID) - return s.fetchAndCacheTier(ctx, userID) - } - - s.mu.RUnlock() - return entry.Tier - } - s.mu.RUnlock() - - return s.fetchAndCacheTier(ctx, userID) -} - -// fetchAndCacheTier fetches the tier from database and caches it -func (s *TierService) fetchAndCacheTier(ctx context.Context, userID string) string { - // v2.0: Default to Pro tier (no payment system) - tier := models.TierPro - var expiresAt *time.Time - - // Check for admin-set tier overrides in MongoDB - if s.mongoDB != nil { - collection := s.mongoDB.Database().Collection("users") - - var user struct { - SubscriptionTier string `bson:"subscriptionTier"` - TierOverride string `bson:"tierOverride"` // Admin manual override - } - - err := collection.FindOne(ctx, bson.M{"_id": userID}).Decode(&user) - if err == nil { - // Admin override takes priority - if user.TierOverride != "" { - tier = user.TierOverride - } else if user.SubscriptionTier != "" { - tier = user.SubscriptionTier - } - } - } - - // Cache the result with expiration info - s.mu.Lock() - s.cache[userID] = CacheEntry{ - Tier: tier, - ExpiresAt: expiresAt, - CachedAt: time.Now(), - } - s.mu.Unlock() - - return tier -} - -// GetLimits returns the limits for a user based on their tier -func (s *TierService) GetLimits(ctx context.Context, userID string) models.TierLimits { - // Get base tier - tier := s.GetUserTier(ctx, userID) - baseLimits := models.GetTierLimits(tier) - - // Check for granular limit overrides - if s.mongoDB != nil { - collection := s.mongoDB.Database().Collection("users") - - var user struct { - LimitOverrides *models.TierLimits `bson:"limitOverrides"` - } - - err := collection.FindOne(ctx, bson.M{"supabaseUserId": userID}).Decode(&user) - if err == nil && user.LimitOverrides != nil { - // Merge overrides with base limits - return s.mergeLimits(baseLimits, *user.LimitOverrides) - } - } - - return baseLimits -} - -// mergeLimits merges override limits with base limits -// Non-zero override values replace base limits -// Zero override values are ignored (use base limit) -func (s *TierService) mergeLimits(base, override models.TierLimits) models.TierLimits { - result := base // Start with base limits - - // Override each field if non-zero - if override.MaxSchedules != 0 { - result.MaxSchedules = override.MaxSchedules - } - if override.MaxAPIKeys != 0 { - result.MaxAPIKeys = override.MaxAPIKeys - } - if override.RequestsPerMinute != 0 { - result.RequestsPerMinute = override.RequestsPerMinute - } - if override.RequestsPerHour != 0 { - result.RequestsPerHour = override.RequestsPerHour - } - if override.RetentionDays != 0 { - result.RetentionDays = override.RetentionDays - } - if override.MaxExecutionsPerDay != 0 { - result.MaxExecutionsPerDay = override.MaxExecutionsPerDay - } - if override.MaxMessagesPerMonth != 0 { - result.MaxMessagesPerMonth = override.MaxMessagesPerMonth - } - if override.MaxFileUploadsPerDay != 0 { - result.MaxFileUploadsPerDay = override.MaxFileUploadsPerDay - } - if override.MaxImageGensPerDay != 0 { - result.MaxImageGensPerDay = override.MaxImageGensPerDay - } - - return result -} - -// InvalidateCache removes a user from the cache (call when tier changes) -func (s *TierService) InvalidateCache(userID string) { - s.mu.Lock() - delete(s.cache, userID) - s.mu.Unlock() - log.Printf("🔄 [TIER] Invalidated cache for user %s", userID) -} - -// CheckScheduleLimit checks if user can create another schedule -func (s *TierService) CheckScheduleLimit(ctx context.Context, userID string, currentCount int64) bool { - limits := s.GetLimits(ctx, userID) - if limits.MaxSchedules < 0 { - return true // Unlimited - } - return currentCount < int64(limits.MaxSchedules) -} - -// CheckAPIKeyLimit checks if user can create another API key -func (s *TierService) CheckAPIKeyLimit(ctx context.Context, userID string, currentCount int64) bool { - limits := s.GetLimits(ctx, userID) - if limits.MaxAPIKeys < 0 { - return true // Unlimited - } - return currentCount < int64(limits.MaxAPIKeys) -} - -// RateLimitConfig holds rate limit values -type RateLimitConfig struct { - RequestsPerMinute int64 - RequestsPerHour int64 -} - -// GetRateLimits returns the rate limit configuration for a user -func (s *TierService) GetRateLimits(ctx context.Context, userID string) RateLimitConfig { - limits := s.GetLimits(ctx, userID) - return RateLimitConfig{ - RequestsPerMinute: limits.RequestsPerMinute, - RequestsPerHour: limits.RequestsPerHour, - } -} - -// GetExecutionRetentionDays returns how long to keep execution history -func (s *TierService) GetExecutionRetentionDays(ctx context.Context, userID string) int { - limits := s.GetLimits(ctx, userID) - return limits.RetentionDays -} - -// CheckMessageLimit checks if user can send another message this month -func (s *TierService) CheckMessageLimit(ctx context.Context, userID string, currentCount int64) bool { - limits := s.GetLimits(ctx, userID) - if limits.MaxMessagesPerMonth < 0 { - return true // Unlimited - } - return currentCount < limits.MaxMessagesPerMonth -} - -// CheckFileUploadLimit checks if user can upload another file today -func (s *TierService) CheckFileUploadLimit(ctx context.Context, userID string, currentCount int64) bool { - limits := s.GetLimits(ctx, userID) - if limits.MaxFileUploadsPerDay < 0 { - return true // Unlimited - } - return currentCount < limits.MaxFileUploadsPerDay -} - -// CheckImageGenLimit checks if user can generate another image today -func (s *TierService) CheckImageGenLimit(ctx context.Context, userID string, currentCount int64) bool { - limits := s.GetLimits(ctx, userID) - if limits.MaxImageGensPerDay < 0 { - return true // Unlimited - } - return currentCount < limits.MaxImageGensPerDay -} diff --git a/backend/internal/services/tier_service_test.go b/backend/internal/services/tier_service_test.go deleted file mode 100644 index f1393461..00000000 --- a/backend/internal/services/tier_service_test.go +++ /dev/null @@ -1,133 +0,0 @@ -package services - -import ( - "context" - "testing" -) - -func TestNewTierService(t *testing.T) { - // Test without MongoDB (nil) - service := NewTierService(nil) - if service == nil { - t.Fatal("Expected non-nil tier service") - } -} - -func TestTierService_GetUserTier_DefaultsToFree(t *testing.T) { - service := NewTierService(nil) - ctx := context.Background() - - tier := service.GetUserTier(ctx, "user-123") - if tier != "pro" { - t.Errorf("Expected 'pro' tier (v2.0 default), got %s", tier) - } -} - -func TestTierService_GetLimits(t *testing.T) { - service := NewTierService(nil) - ctx := context.Background() - - limits := service.GetLimits(ctx, "user-123") - - // Default to pro tier limits (v2.0 default) - if limits.MaxSchedules != 50 { - t.Errorf("Expected MaxSchedules 50, got %d", limits.MaxSchedules) - } - - if limits.MaxAPIKeys != 50 { - t.Errorf("Expected MaxAPIKeys 50, got %d", limits.MaxAPIKeys) - } -} - -func TestTierService_CheckScheduleLimit(t *testing.T) { - service := NewTierService(nil) - ctx := context.Background() - - tests := []struct { - name string - currentCount int64 - expected bool - }{ - {"under limit", 3, true}, - {"at limit", 50, false}, // Pro tier limit is 50 - {"over limit", 100, false}, // Pro tier limit is 50 - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - result := service.CheckScheduleLimit(ctx, "user-123", tt.currentCount) - if result != tt.expected { - t.Errorf("Expected %v for currentCount %d, got %v", tt.expected, tt.currentCount, result) - } - }) - } -} - -func TestTierService_CheckAPIKeyLimit(t *testing.T) { - service := NewTierService(nil) - ctx := context.Background() - - tests := []struct { - name string - currentCount int64 - expected bool - }{ - {"under limit", 1, true}, - {"at limit", 50, false}, // Pro tier limit is 50 - {"over limit", 100, false}, // Pro tier limit is 50 - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - result := service.CheckAPIKeyLimit(ctx, "user-123", tt.currentCount) - if result != tt.expected { - t.Errorf("Expected %v for currentCount %d, got %v", tt.expected, tt.currentCount, result) - } - }) - } -} - -func TestTierService_GetRateLimits(t *testing.T) { - service := NewTierService(nil) - ctx := context.Background() - - rateLimits := service.GetRateLimits(ctx, "user-123") - - // Default to pro tier rate limits (v2.0 default) - if rateLimits.RequestsPerMinute != 300 { - t.Errorf("Expected RequestsPerMinute 300, got %d", rateLimits.RequestsPerMinute) - } - - if rateLimits.RequestsPerHour != 5000 { - t.Errorf("Expected RequestsPerHour 5000, got %d", rateLimits.RequestsPerHour) - } -} - -func TestTierService_GetExecutionRetentionDays(t *testing.T) { - service := NewTierService(nil) - ctx := context.Background() - - days := service.GetExecutionRetentionDays(ctx, "user-123") - - // Default to free tier retention - if days != 30 { - t.Errorf("Expected 30 days retention, got %d", days) - } -} - -func TestTierService_InvalidateCache(t *testing.T) { - service := NewTierService(nil) - ctx := context.Background() - - // Get tier to populate cache - _ = service.GetUserTier(ctx, "user-123") - - // Invalidate cache - service.InvalidateCache("user-123") - - // Should still return pro (v2.0 default) but cache should be empty - tier := service.GetUserTier(ctx, "user-123") - if tier != "pro" { - t.Errorf("Expected 'pro' tier after cache invalidation, got %s", tier) - } -} diff --git a/backend/internal/services/tool_predictor_service.go b/backend/internal/services/tool_predictor_service.go deleted file mode 100644 index 4a659c32..00000000 --- a/backend/internal/services/tool_predictor_service.go +++ /dev/null @@ -1,478 +0,0 @@ -package services - -import ( - "bytes" - "claraverse/internal/database" - "claraverse/internal/models" - "context" - "database/sql" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "strings" - "time" -) - -// ToolPredictorService handles dynamic tool selection for chat requests -type ToolPredictorService struct { - db *database.DB - providerService *ProviderService - chatService *ChatService - defaultPredictorModel string // "gpt-4.1-mini" -} - -// ToolPredictionResult represents selected tools from predictor -type ToolPredictionResult struct { - SelectedTools []string `json:"selected_tools"` // Array of tool names - Reasoning string `json:"reasoning"` -} - -// Tool prediction system prompt (adapted from WorkflowGeneratorV2) -const ToolPredictionSystemPrompt = `You are a tool selection expert for Clara AI chat system. Analyze the user's message and select the MINIMUM set of tools needed to respond effectively. - -CRITICAL RULES: -- Select ONLY tools that are DIRECTLY needed for THIS specific request -- Most requests need 0-3 tools. Rarely should you select more than 5 tools -- If no tools are needed (general conversation, advice, explanation), return empty array -- Don't over-select "just in case" - be precise and minimal - -WHEN TO SELECT TOOLS: -- Search tools: User asks for current info, news, research, "look up", "search for" -- Time tools: User asks "what time", "current date", mentions time-sensitive info -- File tools: User mentions reading/processing files (CSV, PDF, etc.) -- Communication tools: User wants to send message to specific platform (Discord, Slack, email) -- Calculation tools: Complex math, data analysis -- API tools: Interacting with specific services (GitHub, Jira, etc.) - -WHEN NOT TO SELECT TOOLS: -- General questions, explanations, advice, brainstorming -- Coding help (unless explicitly needs to search docs/internet) -- Writing tasks (emails, documents, summaries of provided text) -- Conversation, jokes, casual chat - -Return JSON with selected_tools array (just tool names) and reasoning.` - -// toolPredictionSchema defines structured output for tool selection -var toolPredictionSchema = map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "selected_tools": map[string]interface{}{ - "type": "array", - "items": map[string]interface{}{ - "type": "string", - "description": "Tool name from available tools", - }, - "description": "Array of tool names needed for this request", - }, - "reasoning": map[string]interface{}{ - "type": "string", - "description": "Brief explanation of tool selection", - }, - }, - "required": []string{"selected_tools", "reasoning"}, - "additionalProperties": false, -} - -// NewToolPredictorService creates a new tool predictor service -func NewToolPredictorService( - db *database.DB, - providerService *ProviderService, - chatService *ChatService, -) *ToolPredictorService { - service := &ToolPredictorService{ - db: db, - providerService: providerService, - chatService: chatService, - } - - // Dynamically select first available smart_tool_router model - var modelID string - err := db.QueryRow(` - SELECT m.id - FROM models m - WHERE m.smart_tool_router = 1 AND m.is_visible = 1 - ORDER BY m.id ASC - LIMIT 1 - `).Scan(&modelID) - - if err != nil { - log.Printf("⚠️ [TOOL-PREDICTOR] No smart_tool_router models found, falling back to any available model") - // Fallback: Use any available visible model - err = db.QueryRow(` - SELECT m.id - FROM models m - WHERE m.is_visible = 1 - ORDER BY m.id ASC - LIMIT 1 - `).Scan(&modelID) - - if err != nil { - log.Printf("❌ [TOOL-PREDICTOR] No models available in database at initialization") - modelID = "" // Will be handled later when models are loaded - } - } - - service.defaultPredictorModel = modelID - if modelID != "" { - log.Printf("✅ [TOOL-PREDICTOR] Using default predictor model: %s", modelID) - } - - return service -} - -// PredictTools predicts which tools are needed for a user message -// Returns selected tool definitions and error (nil on success) -// On failure, returns nil (caller should use all tools as fallback) -// conversationHistory: Recent conversation messages for better context-aware tool selection -func (s *ToolPredictorService) PredictTools( - ctx context.Context, - userID string, - userMessage string, - availableTools []map[string]interface{}, - conversationHistory []map[string]interface{}, -) ([]map[string]interface{}, error) { - - // Get predictor model for user (or use default) - predictorModelID, err := s.getPredictorModelForUser(ctx, userID) - if err != nil { - log.Printf("⚠️ [TOOL-PREDICTOR] Could not get predictor model: %v, using default", err) - predictorModelID = s.defaultPredictorModel - } - - // Get provider and model - provider, actualModel, err := s.getProviderAndModel(predictorModelID) - if err != nil { - log.Printf("⚠️ [TOOL-PREDICTOR] Failed to get provider for predictor: %v", err) - return nil, err - } - - log.Printf("🤖 [TOOL-PREDICTOR] Using model: %s (%s)", predictorModelID, actualModel) - - // Build tool list for prompt - toolListPrompt := s.buildToolListPrompt(availableTools) - - // Build user prompt - userPrompt := fmt.Sprintf(`USER MESSAGE: -%s - -AVAILABLE TOOLS: -%s - -Select the minimal set of tools needed. Return JSON with selected_tools and reasoning.`, - userMessage, toolListPrompt) - - // Build messages with conversation history for better context - messages := []map[string]interface{}{ - { - "role": "system", - "content": ToolPredictionSystemPrompt, - }, - } - - // Add recent conversation history for multi-turn context (exclude current message) - // Limit to last 6 messages (3 pairs) to avoid token bloat - historyLimit := 6 - startIdx := len(conversationHistory) - historyLimit - if startIdx < 0 { - startIdx = 0 - } - for i := startIdx; i < len(conversationHistory); i++ { - msg := conversationHistory[i] - messages = append(messages, msg) - } - - // Add current user message with tool selection prompt - messages = append(messages, map[string]interface{}{ - "role": "user", - "content": userPrompt, - }) - - // Build request with structured output - requestBody := map[string]interface{}{ - "model": actualModel, - "messages": messages, - "stream": false, - "temperature": 0.2, // Low temp for consistency - "response_format": map[string]interface{}{ - "type": "json_schema", - "json_schema": map[string]interface{}{ - "name": "tool_prediction", - "strict": true, - "schema": toolPredictionSchema, - }, - }, - } - - reqBody, err := json.Marshal(requestBody) - if err != nil { - return nil, fmt.Errorf("failed to marshal request: %w", err) - } - - log.Printf("📤 [TOOL-PREDICTOR] Sending prediction request to %s", provider.BaseURL) - - // Create HTTP request with timeout - httpReq, err := http.NewRequestWithContext(ctx, "POST", provider.BaseURL+"/chat/completions", bytes.NewBuffer(reqBody)) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - httpReq.Header.Set("Content-Type", "application/json") - httpReq.Header.Set("Authorization", "Bearer "+provider.APIKey) - - // Send request with 30s timeout - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(httpReq) - if err != nil { - return nil, fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - body, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - if resp.StatusCode != http.StatusOK { - log.Printf("⚠️ [TOOL-PREDICTOR] API error: %s", string(body)) - return nil, fmt.Errorf("API error (status %d): %s", resp.StatusCode, string(body)) - } - - // Parse response - var apiResponse struct { - Choices []struct { - Message struct { - Content string `json:"content"` - } `json:"message"` - } `json:"choices"` - } - - if err := json.Unmarshal(body, &apiResponse); err != nil { - return nil, fmt.Errorf("failed to parse API response: %w", err) - } - - if len(apiResponse.Choices) == 0 { - return nil, fmt.Errorf("no response from predictor model") - } - - // Parse the prediction result - var result ToolPredictionResult - content := apiResponse.Choices[0].Message.Content - - if err := json.Unmarshal([]byte(content), &result); err != nil { - log.Printf("⚠️ [TOOL-PREDICTOR] Failed to parse prediction: %v, content: %s", err, content) - return nil, fmt.Errorf("failed to parse prediction: %w", err) - } - - log.Printf("✅ [TOOL-PREDICTOR] Selected %d tools: %v", len(result.SelectedTools), result.SelectedTools) - log.Printf("💭 [TOOL-PREDICTOR] Reasoning: %s", result.Reasoning) - - // Filter available tools to only include selected ones - selectedToolDefs := s.filterToolsByNames(availableTools, result.SelectedTools) - - log.Printf("📊 [TOOL-PREDICTOR] Reduced from %d to %d tools", len(availableTools), len(selectedToolDefs)) - - return selectedToolDefs, nil -} - -// buildToolListPrompt creates a concise list of tools for the prompt -func (s *ToolPredictorService) buildToolListPrompt(tools []map[string]interface{}) string { - var builder strings.Builder - - for i, toolDef := range tools { - fn, ok := toolDef["function"].(map[string]interface{}) - if !ok { - continue - } - - name, _ := fn["name"].(string) - desc, _ := fn["description"].(string) - - builder.WriteString(fmt.Sprintf("%d. %s: %s\n", i+1, name, desc)) - } - - return builder.String() -} - -// filterToolsByNames filters tool definitions by selected names -func (s *ToolPredictorService) filterToolsByNames( - allTools []map[string]interface{}, - selectedNames []string, -) []map[string]interface{} { - - // Build set for O(1) lookup - nameSet := make(map[string]bool) - for _, name := range selectedNames { - nameSet[name] = true - } - - filtered := make([]map[string]interface{}, 0, len(selectedNames)) - - for _, toolDef := range allTools { - fn, ok := toolDef["function"].(map[string]interface{}) - if !ok { - continue - } - - name, ok := fn["name"].(string) - if !ok { - continue - } - - if nameSet[name] { - filtered = append(filtered, toolDef) - } - } - - return filtered -} - -// getPredictorModelForUser gets user's preferred predictor model -func (s *ToolPredictorService) getPredictorModelForUser(ctx context.Context, userID string) (string, error) { - // Query user preferences for tool predictor model - var predictorModelID sql.NullString - err := s.db.QueryRow(` - SELECT preferences->>'toolPredictorModelId' - FROM users - WHERE id = ? - `, userID).Scan(&predictorModelID) - - if err != nil { - // User not found or no preferences - use default - log.Printf("⚠️ [TOOL-PREDICTOR] Could not get user predictor preference: %v, using default", err) - return s.defaultPredictorModel, nil - } - - // If user has a preference and it's not empty, use it - if predictorModelID.Valid && predictorModelID.String != "" { - log.Printf("🎯 [TOOL-PREDICTOR] Using user-preferred model: %s", predictorModelID.String) - return predictorModelID.String, nil - } - - // No preference set - use default - return s.defaultPredictorModel, nil -} - -// getProviderAndModel resolves model ID to provider and actual model name -func (s *ToolPredictorService) getProviderAndModel(modelID string) (*models.Provider, string, error) { - if modelID == "" { - return s.getDefaultPredictorModel() - } - - // Try to find model in database - var providerID int - var modelName string - var smartToolRouter int - - err := s.db.QueryRow(` - SELECT m.name, m.provider_id, COALESCE(m.smart_tool_router, 0) - FROM models m - WHERE m.id = ? AND m.is_visible = 1 - `, modelID).Scan(&modelName, &providerID, &smartToolRouter) - - if err != nil { - // Try as model alias - if s.chatService != nil { - if provider, actualModel, found := s.chatService.ResolveModelAlias(modelID); found { - return provider, actualModel, nil - } - } - // Only fall back to default if this is NOT already the default model (avoid recursion) - if modelID != s.defaultPredictorModel { - return s.getDefaultPredictorModel() - } - return nil, "", fmt.Errorf("default predictor model %s not found in database", modelID) - } - - // Verify model is marked as smart tool router - if smartToolRouter == 0 { - log.Printf("⚠️ [TOOL-PREDICTOR] Model %s not marked as smart_tool_router", modelID) - - // Search for ANY available smart router model as fallback - log.Printf("⚠️ [TOOL-PREDICTOR] Searching for any available smart router model...") - var fallbackModelID string - var fallbackModelName string - var fallbackProviderID int - - err := s.db.QueryRow(` - SELECT m.id, m.name, m.provider_id - FROM models m - WHERE m.smart_tool_router = 1 AND m.is_visible = 1 - ORDER BY m.id ASC - LIMIT 1 - `).Scan(&fallbackModelID, &fallbackModelName, &fallbackProviderID) - - if err != nil { - return nil, "", fmt.Errorf("no smart_tool_router models available in database") - } - - log.Printf("✅ [TOOL-PREDICTOR] Found smart router model: %s (%s)", fallbackModelID, fallbackModelName) - - provider, err := s.providerService.GetByID(fallbackProviderID) - if err != nil { - return nil, "", fmt.Errorf("failed to get provider for smart router model: %w", err) - } - - return provider, fallbackModelName, nil - } - - provider, err := s.providerService.GetByID(providerID) - if err != nil { - return nil, "", fmt.Errorf("failed to get provider: %w", err) - } - - return provider, modelName, nil -} - -// getDefaultPredictorModel returns the default predictor model -// This directly looks up the default model to avoid infinite recursion -func (s *ToolPredictorService) getDefaultPredictorModel() (*models.Provider, string, error) { - // First try to get the hardcoded default model - var providerID int - var modelName string - var smartToolRouter int - - err := s.db.QueryRow(` - SELECT m.name, m.provider_id, COALESCE(m.smart_tool_router, 0) - FROM models m - WHERE m.id = ? AND m.is_visible = 1 - `, s.defaultPredictorModel).Scan(&modelName, &providerID, &smartToolRouter) - - if err != nil { - // Try as model alias - if s.chatService != nil { - if provider, actualModel, found := s.chatService.ResolveModelAlias(s.defaultPredictorModel); found { - return provider, actualModel, nil - } - } - return nil, "", fmt.Errorf("default predictor model %s not found: %w", s.defaultPredictorModel, err) - } - - // If default model is not marked as smart_tool_router, find ANY available smart router model - if smartToolRouter == 0 { - log.Printf("⚠️ [TOOL-PREDICTOR] Default model %s not marked as smart_tool_router, searching for any available smart router model...", s.defaultPredictorModel) - - var fallbackModelID string - err := s.db.QueryRow(` - SELECT m.id, m.name, m.provider_id - FROM models m - WHERE m.smart_tool_router = 1 AND m.is_visible = 1 - ORDER BY m.id ASC - LIMIT 1 - `).Scan(&fallbackModelID, &modelName, &providerID) - - if err != nil { - return nil, "", fmt.Errorf("no smart_tool_router models available in database") - } - - log.Printf("✅ [TOOL-PREDICTOR] Found smart router model: %s (%s)", fallbackModelID, modelName) - } - - provider, err := s.providerService.GetByID(providerID) - if err != nil { - return nil, "", fmt.Errorf("failed to get provider for default model: %w", err) - } - - return provider, modelName, nil -} diff --git a/backend/internal/services/tool_registry.go b/backend/internal/services/tool_registry.go deleted file mode 100644 index 7327e07b..00000000 --- a/backend/internal/services/tool_registry.go +++ /dev/null @@ -1,1110 +0,0 @@ -package services - -import "strings" - -// ToolDefinition represents a single tool in the registry -type ToolDefinition struct { - ID string `json:"id"` - Name string `json:"name"` - Description string `json:"description"` - Category string `json:"category"` - Icon string `json:"icon"` // Icon name for frontend (lucide-react icon name) - Keywords []string `json:"keywords"` - UseCases []string `json:"use_cases"` - Parameters string `json:"parameters,omitempty"` // Brief parameter description - CodeBlockExample string `json:"code_block_example,omitempty"` // Example argumentMapping for code_block usage -} - -// ToolCategory represents a category of tools -type ToolCategory struct { - ID string `json:"id"` - Name string `json:"name"` - Icon string `json:"icon"` - Description string `json:"description"` -} - -// ToolRegistry holds all available tools - easily extensible -var ToolRegistry = []ToolDefinition{ - // ═══════════════════════════════════════════════════════════════ - // 📊 DATA & ANALYSIS - // ═══════════════════════════════════════════════════════════════ - { - ID: "analyze_data", - Name: "Analyze Data", - Description: "Python data analysis with charts and visualizations", - Category: "data_analysis", - Icon: "BarChart2", - Keywords: []string{"analyze", "analysis", "data", "chart", "graph", "statistics", "visualize", "visualization", "metrics", "plot", "pandas", "numpy"}, - UseCases: []string{"Analyze CSV/Excel data", "Generate charts", "Calculate statistics", "Create visualizations"}, - Parameters: "code: Python code to execute", - }, - { - ID: "calculate_math", - Name: "Calculate Math", - Description: "Mathematical calculations and expressions", - Category: "data_analysis", - Icon: "Calculator", - Keywords: []string{"calculate", "math", "formula", "equation", "compute", "arithmetic", "algebra"}, - UseCases: []string{"Solve equations", "Calculate formulas", "Mathematical operations"}, - Parameters: "expression: Math expression to evaluate", - CodeBlockExample: `{"expression": "{{start.input}}"}`, - }, - { - ID: "read_spreadsheet", - Name: "Read Spreadsheet", - Description: "Read Excel/CSV files (xlsx, xls, csv, tsv)", - Category: "data_analysis", - Icon: "FileSpreadsheet", - Keywords: []string{"spreadsheet", "excel", "csv", "xlsx", "xls", "tsv", "read", "import", "table"}, - UseCases: []string{"Read Excel files", "Import CSV data", "Parse spreadsheet data"}, - Parameters: "file_id: ID of uploaded file", - }, - { - ID: "read_data_file", - Name: "Read Data File", - Description: "Read and parse data files (CSV, JSON, text)", - Category: "data_analysis", - Icon: "FileJson", - Keywords: []string{"read", "parse", "data", "file", "json", "csv", "text", "import"}, - UseCases: []string{"Read JSON files", "Parse text data", "Import data files"}, - Parameters: "file_id: ID of uploaded file", - }, - { - ID: "read_document", - Name: "Read Document", - Description: "Extract text from documents (PDF, DOCX, PPTX)", - Category: "data_analysis", - Icon: "FileText", - Keywords: []string{"document", "pdf", "docx", "pptx", "word", "powerpoint", "extract", "read", "text"}, - UseCases: []string{"Extract PDF text", "Read Word documents", "Parse presentations"}, - Parameters: "file_id: ID of uploaded file", - }, - - // ═══════════════════════════════════════════════════════════════ - // 🔍 SEARCH & WEB - // ═══════════════════════════════════════════════════════════════ - { - ID: "search_web", - Name: "Search Web", - Description: "Search the internet for information, news, articles", - Category: "search_web", - Icon: "Search", - Keywords: []string{"search", "google", "web", "internet", "find", "lookup", "query", "news", "articles", "information"}, - UseCases: []string{"Search for information", "Find news articles", "Research topics"}, - Parameters: "query: Search query string", - CodeBlockExample: `{"query": "{{start.input}}"}`, - }, - { - ID: "search_images", - Name: "Search Images", - Description: "Search for images on the web", - Category: "search_web", - Icon: "Image", - Keywords: []string{"image", "images", "photo", "picture", "search", "find", "visual"}, - UseCases: []string{"Find images", "Search for photos", "Visual content search"}, - Parameters: "query: Image search query", - }, - { - ID: "scrape_web", - Name: "Scrape Web", - Description: "Scrape content from a specific URL", - Category: "search_web", - Icon: "Globe", - Keywords: []string{"scrape", "crawl", "url", "website", "extract", "web", "page", "content"}, - UseCases: []string{"Extract webpage content", "Scrape URLs", "Get page data"}, - Parameters: "url: URL to scrape", - CodeBlockExample: `{"url": "{{start.input}}"}`, - }, - { - ID: "download_file", - Name: "Download File", - Description: "Download a file from a URL", - Category: "search_web", - Icon: "Download", - Keywords: []string{"download", "file", "url", "fetch", "get", "retrieve"}, - UseCases: []string{"Download files", "Fetch remote content", "Get assets"}, - Parameters: "url: URL of file to download", - }, - - // ═══════════════════════════════════════════════════════════════ - // 📝 CONTENT CREATION - // ═══════════════════════════════════════════════════════════════ - { - ID: "create_document", - Name: "Create Document", - Description: "Create DOCX or PDF documents", - Category: "content_creation", - Icon: "FileText", - Keywords: []string{"create", "document", "docx", "pdf", "word", "write", "generate", "report"}, - UseCases: []string{"Create Word documents", "Generate PDFs", "Write reports"}, - Parameters: "content: Document content, format: docx|pdf", - }, - { - ID: "create_text_file", - Name: "Create Text File", - Description: "Create plain text files", - Category: "content_creation", - Icon: "FilePlus", - Keywords: []string{"create", "text", "file", "write", "plain", "txt"}, - UseCases: []string{"Create text files", "Write plain text", "Save content"}, - Parameters: "content: Text content, filename: Output filename", - }, - { - ID: "create_presentation", - Name: "Create Presentation", - Description: "Create PowerPoint presentations with slides", - Category: "content_creation", - Icon: "Presentation", - Keywords: []string{"presentation", "powerpoint", "pptx", "slides", "create", "deck"}, - UseCases: []string{"Create presentations", "Generate slide decks", "Make PowerPoints"}, - Parameters: "slides: Array of slide content", - }, - { - ID: "generate_image", - Name: "Generate Image", - Description: "Generate images using AI (DALL-E)", - Category: "content_creation", - Icon: "Wand2", - Keywords: []string{"generate", "image", "create", "ai", "dall-e", "picture", "art", "visual"}, - UseCases: []string{"Generate AI images", "Create artwork", "Visual content generation"}, - Parameters: "prompt: Image description", - }, - { - ID: "edit_image", - Name: "Edit Image", - Description: "Edit/transform images (resize, crop, filters)", - Category: "content_creation", - Icon: "ImagePlus", - Keywords: []string{"edit", "image", "resize", "crop", "filter", "transform", "modify"}, - UseCases: []string{"Resize images", "Crop photos", "Apply filters"}, - Parameters: "file_id: Image file ID, operation: resize|crop|filter", - }, - { - ID: "html_to_pdf", - Name: "HTML to PDF", - Description: "Convert HTML content to PDF", - Category: "content_creation", - Icon: "FileOutput", - Keywords: []string{"html", "pdf", "convert", "render", "export"}, - UseCases: []string{"Convert HTML to PDF", "Export web content", "Generate PDF reports"}, - Parameters: "html: HTML content to convert", - }, - - // ═══════════════════════════════════════════════════════════════ - // 🎤 MEDIA PROCESSING - // ═══════════════════════════════════════════════════════════════ - { - ID: "transcribe_audio", - Name: "Transcribe Audio", - Description: "Transcribe speech from audio (MP3, WAV, M4A, OGG, FLAC, WebM)", - Category: "media_processing", - Icon: "Mic", - Keywords: []string{"transcribe", "audio", "speech", "voice", "mp3", "wav", "recording", "speech-to-text"}, - UseCases: []string{"Transcribe audio files", "Convert speech to text", "Process recordings"}, - Parameters: "file_id: Audio file ID", - }, - { - ID: "describe_image", - Name: "Describe Image", - Description: "Analyze/describe images using AI vision", - Category: "media_processing", - Icon: "Eye", - Keywords: []string{"describe", "image", "vision", "analyze", "see", "look", "visual", "ai"}, - UseCases: []string{"Describe image content", "Analyze visuals", "Image understanding"}, - Parameters: "file_id: Image file ID", - }, - - // ═══════════════════════════════════════════════════════════════ - // ⏰ UTILITIES - // ═══════════════════════════════════════════════════════════════ - { - ID: "get_current_time", - Name: "Get Current Time", - Description: "Get current date/time (REQUIRED for time-sensitive queries)", - Category: "utilities", - Icon: "Clock", - Keywords: []string{"time", "date", "now", "today", "current", "datetime", "timestamp"}, - UseCases: []string{"Get current time", "Date operations", "Time-sensitive queries"}, - Parameters: "timezone: Optional timezone", - CodeBlockExample: `{}`, - }, - { - ID: "ask_user", - Name: "Ask User Questions", - Description: "Ask the user clarifying questions via modal dialog. Waits for response (blocking).", - Category: "utilities", - Icon: "MessageCircleQuestion", - Keywords: []string{"ask", "question", "prompt", "user", "input", "clarify", "modal", "dialog", "form", "interactive", "wait", "blocking"}, - UseCases: []string{"Ask clarifying questions", "Gather user input", "Get user preferences", "Confirm actions", "Multi-choice questions"}, - Parameters: "title: Prompt title, questions: Array of questions (text/number/checkbox/select/multi-select), allow_skip: Optional", - }, - { - ID: "run_python", - Name: "Run Python", - Description: "Execute Python code for custom logic", - Category: "utilities", - Icon: "Code", - Keywords: []string{"python", "code", "script", "execute", "run", "program", "custom"}, - UseCases: []string{"Run custom code", "Execute scripts", "Custom processing"}, - Parameters: "code: Python code to execute", - }, - { - ID: "api_request", - Name: "API Request", - Description: "Make HTTP API requests (GET, POST, PUT, DELETE)", - Category: "utilities", - Icon: "Globe", - Keywords: []string{"api", "http", "request", "rest", "get", "post", "put", "delete", "endpoint"}, - UseCases: []string{"Call external APIs", "HTTP requests", "API integrations"}, - Parameters: "url: API URL, method: GET|POST|PUT|DELETE, body: Request body", - }, - { - ID: "send_webhook", - Name: "Send Webhook", - Description: "Send data to any webhook URL", - Category: "utilities", - Icon: "Webhook", - Keywords: []string{"webhook", "send", "post", "notify", "trigger", "callback"}, - UseCases: []string{"Trigger webhooks", "Send notifications", "External integrations"}, - Parameters: "url: Webhook URL, data: JSON payload", - CodeBlockExample: `{"webhook_url": "https://example.com/hook", "data": {"message": "{{previous-block.response}}"}}`, - }, - - // ═══════════════════════════════════════════════════════════════ - // 💬 MESSAGING & COMMUNICATION - // ═══════════════════════════════════════════════════════════════ - { - ID: "send_discord_message", - Name: "Send Discord Message", - Description: "Send message to Discord channel", - Category: "messaging", - Icon: "MessageCircle", - Keywords: []string{"discord", "message", "send", "chat", "channel", "notify", "bot"}, - UseCases: []string{"Send Discord messages", "Discord notifications", "Bot messaging"}, - Parameters: "content: Message text, embed_title: Optional embed title", - CodeBlockExample: `{"content": "{{previous-block.response}}"}`, - }, - { - ID: "send_slack_message", - Name: "Send Slack Message", - Description: "Send message to Slack channel", - Category: "messaging", - Icon: "Hash", - Keywords: []string{"slack", "message", "send", "channel", "notify", "workspace"}, - UseCases: []string{"Send Slack messages", "Slack notifications", "Team messaging"}, - Parameters: "channel: Channel name, text: Message text", - CodeBlockExample: `{"channel": "#general", "text": "{{previous-block.response}}"}`, - }, - { - ID: "send_telegram_message", - Name: "Send Telegram Message", - Description: "Send message to Telegram chat", - Category: "messaging", - Icon: "Send", - Keywords: []string{"telegram", "message", "send", "chat", "notify", "bot"}, - UseCases: []string{"Send Telegram messages", "Telegram notifications", "Bot messaging"}, - Parameters: "chat_id: Chat ID, text: Message text", - }, - { - ID: "send_google_chat_message", - Name: "Send Google Chat Message", - Description: "Send message to Google Chat", - Category: "messaging", - Icon: "MessageSquare", - Keywords: []string{"google chat", "message", "send", "hangouts", "workspace"}, - UseCases: []string{"Google Chat messages", "Workspace notifications"}, - Parameters: "space: Space ID, text: Message text", - }, - { - ID: "send_teams_message", - Name: "Send Teams Message", - Description: "Send message to Microsoft Teams", - Category: "messaging", - Icon: "Users", - Keywords: []string{"teams", "microsoft", "message", "send", "channel", "notify"}, - UseCases: []string{"Teams messages", "Microsoft Teams notifications"}, - Parameters: "channel: Channel, text: Message text", - }, - { - ID: "send_email", - Name: "Send Email", - Description: "Send email via SendGrid", - Category: "messaging", - Icon: "Mail", - Keywords: []string{"email", "send", "mail", "sendgrid", "notify"}, - UseCases: []string{"Send emails", "Email notifications"}, - Parameters: "to: Recipient, subject: Subject, body: Email body", - }, - { - ID: "send_brevo_email", - Name: "Send Brevo Email", - Description: "Send email via Brevo", - Category: "messaging", - Icon: "Mail", - Keywords: []string{"email", "brevo", "sendinblue", "send", "mail"}, - UseCases: []string{"Send emails via Brevo", "Marketing emails"}, - Parameters: "to: Recipient, subject: Subject, body: Email body", - }, - { - ID: "twilio_send_sms", - Name: "Send SMS", - Description: "Send SMS via Twilio", - Category: "messaging", - Icon: "Smartphone", - Keywords: []string{"sms", "text", "twilio", "send", "phone", "mobile"}, - UseCases: []string{"Send SMS messages", "Text notifications"}, - Parameters: "to: Phone number, body: Message text", - }, - { - ID: "twilio_send_whatsapp", - Name: "Send WhatsApp", - Description: "Send WhatsApp message via Twilio", - Category: "messaging", - Icon: "MessageCircle", - Keywords: []string{"whatsapp", "message", "twilio", "send", "chat"}, - UseCases: []string{"Send WhatsApp messages", "WhatsApp notifications"}, - Parameters: "to: Phone number, body: Message text", - }, - { - ID: "referralmonk_whatsapp", - Name: "ReferralMonk WhatsApp", - Description: "Send WhatsApp message via ReferralMonk with template support", - Category: "messaging", - Icon: "MessageSquare", - Keywords: []string{"whatsapp", "referralmonk", "template", "campaign", "message", "send", "ahaguru"}, - UseCases: []string{"Send templated WhatsApp messages", "WhatsApp campaigns", "Marketing via WhatsApp"}, - Parameters: "mobile: Phone with country code, template_name: Template ID, language: Language code (default: en), param_1/2/3: Template parameters", - CodeBlockExample: `{"mobile": "917550002919", "template_name": "demo_session_01", "language": "en", "param_1": "{{user.name}}", "param_2": "{{lesson.link}}", "param_3": "Team Name"}`, - }, - - // ═══════════════════════════════════════════════════════════════ - // 📹 VIDEO CONFERENCING - // ═══════════════════════════════════════════════════════════════ - { - ID: "zoom_meeting", - Name: "Zoom Meeting", - Description: "Zoom meetings & webinars - create, list, register attendees", - Category: "video_conferencing", - Icon: "Video", - Keywords: []string{"zoom", "meeting", "webinar", "video", "conference", "call", "register", "attendee", "schedule"}, - UseCases: []string{"Create Zoom meetings", "Register for webinars", "List meetings", "Manage attendees"}, - Parameters: "action: create|list|get|register|create_webinar|register_webinar, meeting_id/webinar_id, email, first_name, last_name", - }, - { - ID: "calendly_events", - Name: "Calendly Events", - Description: "List and manage Calendly events", - Category: "video_conferencing", - Icon: "Calendar", - Keywords: []string{"calendly", "calendar", "schedule", "event", "booking", "appointment"}, - UseCases: []string{"List Calendly events", "View scheduled meetings"}, - Parameters: "user: User URI", - }, - { - ID: "calendly_event_types", - Name: "Calendly Event Types", - Description: "List Calendly event types", - Category: "video_conferencing", - Icon: "CalendarDays", - Keywords: []string{"calendly", "event type", "booking type", "schedule"}, - UseCases: []string{"List event types", "Get booking options"}, - Parameters: "user: User URI", - }, - { - ID: "calendly_invitees", - Name: "Calendly Invitees", - Description: "Get event invitees/attendees", - Category: "video_conferencing", - Icon: "Users", - Keywords: []string{"calendly", "invitee", "attendee", "participant"}, - UseCases: []string{"List event invitees", "Get attendee info"}, - Parameters: "event_uuid: Event UUID", - }, - - // ═══════════════════════════════════════════════════════════════ - // 📋 PROJECT MANAGEMENT - // ═══════════════════════════════════════════════════════════════ - { - ID: "jira_issues", - Name: "Jira Issues", - Description: "List/search Jira issues", - Category: "project_management", - Icon: "CheckSquare", - Keywords: []string{"jira", "issue", "ticket", "bug", "task", "search", "list"}, - UseCases: []string{"Search Jira issues", "List tickets", "Find tasks"}, - Parameters: "jql: JQL query string", - }, - { - ID: "jira_create_issue", - Name: "Create Jira Issue", - Description: "Create a new Jira issue", - Category: "project_management", - Icon: "PlusSquare", - Keywords: []string{"jira", "create", "issue", "ticket", "bug", "task", "new"}, - UseCases: []string{"Create Jira tickets", "Report bugs", "Add tasks"}, - Parameters: "project: Project key, summary: Title, description: Description", - }, - { - ID: "jira_update_issue", - Name: "Update Jira Issue", - Description: "Update an existing Jira issue", - Category: "project_management", - Icon: "Edit", - Keywords: []string{"jira", "update", "edit", "issue", "ticket", "modify"}, - UseCases: []string{"Update tickets", "Modify issues", "Edit tasks"}, - Parameters: "issue_key: Issue key, fields: Fields to update", - }, - { - ID: "linear_issues", - Name: "Linear Issues", - Description: "List Linear issues", - Category: "project_management", - Icon: "CheckSquare", - Keywords: []string{"linear", "issue", "ticket", "task", "list"}, - UseCases: []string{"List Linear issues", "View tasks"}, - Parameters: "team_id: Team ID", - }, - { - ID: "linear_create_issue", - Name: "Create Linear Issue", - Description: "Create a new Linear issue", - Category: "project_management", - Icon: "PlusSquare", - Keywords: []string{"linear", "create", "issue", "ticket", "task", "new"}, - UseCases: []string{"Create Linear issues", "Add tasks"}, - Parameters: "team_id: Team ID, title: Title, description: Description", - }, - { - ID: "linear_update_issue", - Name: "Update Linear Issue", - Description: "Update a Linear issue", - Category: "project_management", - Icon: "Edit", - Keywords: []string{"linear", "update", "edit", "issue", "modify"}, - UseCases: []string{"Update Linear issues", "Modify tasks"}, - Parameters: "issue_id: Issue ID, fields: Fields to update", - }, - { - ID: "clickup_tasks", - Name: "ClickUp Tasks", - Description: "List ClickUp tasks", - Category: "project_management", - Icon: "CheckCircle", - Keywords: []string{"clickup", "task", "list", "todo"}, - UseCases: []string{"List ClickUp tasks", "View todos"}, - Parameters: "list_id: List ID", - }, - { - ID: "clickup_create_task", - Name: "Create ClickUp Task", - Description: "Create a new ClickUp task", - Category: "project_management", - Icon: "PlusCircle", - Keywords: []string{"clickup", "create", "task", "new", "todo"}, - UseCases: []string{"Create ClickUp tasks", "Add todos"}, - Parameters: "list_id: List ID, name: Task name, description: Description", - }, - { - ID: "clickup_update_task", - Name: "Update ClickUp Task", - Description: "Update a ClickUp task", - Category: "project_management", - Icon: "Edit", - Keywords: []string{"clickup", "update", "edit", "task", "modify"}, - UseCases: []string{"Update ClickUp tasks", "Modify todos"}, - Parameters: "task_id: Task ID, fields: Fields to update", - }, - { - ID: "trello_boards", - Name: "Trello Boards", - Description: "List Trello boards", - Category: "project_management", - Icon: "Layout", - Keywords: []string{"trello", "board", "list", "kanban"}, - UseCases: []string{"List Trello boards", "View kanban boards"}, - Parameters: "None required", - }, - { - ID: "trello_lists", - Name: "Trello Lists", - Description: "List Trello lists in a board", - Category: "project_management", - Icon: "List", - Keywords: []string{"trello", "list", "column", "board"}, - UseCases: []string{"List Trello lists", "View board columns"}, - Parameters: "board_id: Board ID", - }, - { - ID: "trello_cards", - Name: "Trello Cards", - Description: "List Trello cards", - Category: "project_management", - Icon: "Square", - Keywords: []string{"trello", "card", "task", "list"}, - UseCases: []string{"List Trello cards", "View tasks"}, - Parameters: "list_id: List ID", - }, - { - ID: "trello_create_card", - Name: "Create Trello Card", - Description: "Create a new Trello card", - Category: "project_management", - Icon: "Plus", - Keywords: []string{"trello", "create", "card", "new", "task"}, - UseCases: []string{"Create Trello cards", "Add tasks"}, - Parameters: "list_id: List ID, name: Card name, description: Description", - }, - { - ID: "asana_tasks", - Name: "Asana Tasks", - Description: "List Asana tasks", - Category: "project_management", - Icon: "CheckSquare", - Keywords: []string{"asana", "task", "list", "project"}, - UseCases: []string{"List Asana tasks", "View project tasks"}, - Parameters: "project_id: Project ID", - }, - - // ═══════════════════════════════════════════════════════════════ - // 💼 CRM & SALES - // ═══════════════════════════════════════════════════════════════ - { - ID: "hubspot_contacts", - Name: "HubSpot Contacts", - Description: "List/search HubSpot contacts", - Category: "crm_sales", - Icon: "Users", - Keywords: []string{"hubspot", "contact", "crm", "customer", "lead", "list", "search"}, - UseCases: []string{"List HubSpot contacts", "Search customers", "Find leads"}, - Parameters: "query: Optional search query", - }, - { - ID: "hubspot_deals", - Name: "HubSpot Deals", - Description: "List HubSpot deals", - Category: "crm_sales", - Icon: "DollarSign", - Keywords: []string{"hubspot", "deal", "sales", "pipeline", "opportunity"}, - UseCases: []string{"List deals", "View sales pipeline"}, - Parameters: "None required", - }, - { - ID: "hubspot_companies", - Name: "HubSpot Companies", - Description: "List HubSpot companies", - Category: "crm_sales", - Icon: "Building", - Keywords: []string{"hubspot", "company", "organization", "account"}, - UseCases: []string{"List companies", "View accounts"}, - Parameters: "None required", - }, - { - ID: "leadsquared_leads", - Name: "LeadSquared Leads", - Description: "List LeadSquared leads", - Category: "crm_sales", - Icon: "UserPlus", - Keywords: []string{"leadsquared", "lead", "crm", "prospect"}, - UseCases: []string{"List leads", "View prospects"}, - Parameters: "query: Optional search query", - }, - { - ID: "leadsquared_create_lead", - Name: "Create LeadSquared Lead", - Description: "Create a new LeadSquared lead", - Category: "crm_sales", - Icon: "UserPlus", - Keywords: []string{"leadsquared", "create", "lead", "new", "prospect"}, - UseCases: []string{"Create leads", "Add prospects"}, - Parameters: "email: Email, firstName: First name, lastName: Last name", - }, - { - ID: "leadsquared_activities", - Name: "LeadSquared Activities", - Description: "List LeadSquared activities", - Category: "crm_sales", - Icon: "Activity", - Keywords: []string{"leadsquared", "activity", "history", "timeline"}, - UseCases: []string{"List activities", "View lead history"}, - Parameters: "lead_id: Lead ID", - }, - { - ID: "mailchimp_lists", - Name: "Mailchimp Lists", - Description: "List Mailchimp audiences", - Category: "crm_sales", - Icon: "Users", - Keywords: []string{"mailchimp", "list", "audience", "subscribers", "email"}, - UseCases: []string{"List audiences", "View subscriber lists"}, - Parameters: "None required", - }, - { - ID: "mailchimp_add_subscriber", - Name: "Mailchimp Add Subscriber", - Description: "Add subscriber to Mailchimp list", - Category: "crm_sales", - Icon: "UserPlus", - Keywords: []string{"mailchimp", "subscriber", "add", "email", "list"}, - UseCases: []string{"Add subscribers", "Email list signup"}, - Parameters: "list_id: List ID, email: Email address", - }, - - // ═══════════════════════════════════════════════════════════════ - // 📊 ANALYTICS - // ═══════════════════════════════════════════════════════════════ - { - ID: "posthog_capture", - Name: "PostHog Capture", - Description: "Track PostHog events", - Category: "analytics", - Icon: "BarChart", - Keywords: []string{"posthog", "track", "event", "analytics", "capture"}, - UseCases: []string{"Track events", "Capture user actions"}, - Parameters: "event: Event name, properties: Event properties", - }, - { - ID: "posthog_identify", - Name: "PostHog Identify", - Description: "Identify PostHog user", - Category: "analytics", - Icon: "User", - Keywords: []string{"posthog", "identify", "user", "profile"}, - UseCases: []string{"Identify users", "Set user properties"}, - Parameters: "distinct_id: User ID, properties: User properties", - }, - { - ID: "posthog_query", - Name: "PostHog Query", - Description: "Query PostHog analytics", - Category: "analytics", - Icon: "Database", - Keywords: []string{"posthog", "query", "analytics", "insights", "data"}, - UseCases: []string{"Query analytics", "Get insights"}, - Parameters: "query: HogQL query", - }, - { - ID: "mixpanel_track", - Name: "Mixpanel Track", - Description: "Track Mixpanel events", - Category: "analytics", - Icon: "BarChart", - Keywords: []string{"mixpanel", "track", "event", "analytics"}, - UseCases: []string{"Track Mixpanel events", "Log user actions"}, - Parameters: "event: Event name, properties: Event properties", - }, - { - ID: "mixpanel_user_profile", - Name: "Mixpanel User Profile", - Description: "Update Mixpanel user profile", - Category: "analytics", - Icon: "User", - Keywords: []string{"mixpanel", "user", "profile", "update"}, - UseCases: []string{"Update user profiles", "Set user properties"}, - Parameters: "distinct_id: User ID, properties: Profile properties", - }, - - // ═══════════════════════════════════════════════════════════════ - // 🐙 CODE & DEVOPS - // ═══════════════════════════════════════════════════════════════ - { - ID: "github_create_issue", - Name: "GitHub Create Issue", - Description: "Create a GitHub issue", - Category: "code_devops", - Icon: "CircleDot", - Keywords: []string{"github", "issue", "create", "bug", "feature", "repo"}, - UseCases: []string{"Create GitHub issues", "Report bugs"}, - Parameters: "owner: Repo owner, repo: Repo name, title: Title, body: Description", - }, - { - ID: "github_list_issues", - Name: "GitHub List Issues", - Description: "List GitHub issues", - Category: "code_devops", - Icon: "List", - Keywords: []string{"github", "issue", "list", "bug", "repo"}, - UseCases: []string{"List GitHub issues", "View repo issues"}, - Parameters: "owner: Repo owner, repo: Repo name", - }, - { - ID: "github_get_repo", - Name: "GitHub Get Repo", - Description: "Get GitHub repository info", - Category: "code_devops", - Icon: "GitBranch", - Keywords: []string{"github", "repo", "repository", "info", "details"}, - UseCases: []string{"Get repo info", "View repository details"}, - Parameters: "owner: Repo owner, repo: Repo name", - }, - { - ID: "github_add_comment", - Name: "GitHub Add Comment", - Description: "Add comment to GitHub issue/PR", - Category: "code_devops", - Icon: "MessageSquare", - Keywords: []string{"github", "comment", "issue", "pr", "pull request"}, - UseCases: []string{"Comment on issues", "Reply to PRs"}, - Parameters: "owner: Repo owner, repo: Repo name, issue_number: Issue number, body: Comment", - }, - { - ID: "gitlab_projects", - Name: "GitLab Projects", - Description: "List GitLab projects", - Category: "code_devops", - Icon: "Folder", - Keywords: []string{"gitlab", "project", "list", "repo"}, - UseCases: []string{"List GitLab projects", "View repositories"}, - Parameters: "None required", - }, - { - ID: "gitlab_issues", - Name: "GitLab Issues", - Description: "List GitLab issues", - Category: "code_devops", - Icon: "List", - Keywords: []string{"gitlab", "issue", "list", "bug"}, - UseCases: []string{"List GitLab issues", "View project issues"}, - Parameters: "project_id: Project ID", - }, - { - ID: "gitlab_mrs", - Name: "GitLab Merge Requests", - Description: "List GitLab merge requests", - Category: "code_devops", - Icon: "GitMerge", - Keywords: []string{"gitlab", "merge request", "mr", "pull request", "pr"}, - UseCases: []string{"List merge requests", "View MRs"}, - Parameters: "project_id: Project ID", - }, - { - ID: "netlify_sites", - Name: "Netlify Sites", - Description: "List Netlify sites", - Category: "code_devops", - Icon: "Globe", - Keywords: []string{"netlify", "site", "list", "deploy", "hosting"}, - UseCases: []string{"List Netlify sites", "View deployed sites"}, - Parameters: "None required", - }, - { - ID: "netlify_deploys", - Name: "Netlify Deploys", - Description: "List Netlify deploys", - Category: "code_devops", - Icon: "Rocket", - Keywords: []string{"netlify", "deploy", "list", "build", "release"}, - UseCases: []string{"List deploys", "View build history"}, - Parameters: "site_id: Site ID", - }, - { - ID: "netlify_trigger_build", - Name: "Netlify Trigger Build", - Description: "Trigger a Netlify build", - Category: "code_devops", - Icon: "Play", - Keywords: []string{"netlify", "build", "trigger", "deploy", "release"}, - UseCases: []string{"Trigger builds", "Deploy sites"}, - Parameters: "site_id: Site ID", - }, - - // ═══════════════════════════════════════════════════════════════ - // 📓 PRODUCTIVITY - // ═══════════════════════════════════════════════════════════════ - { - ID: "notion_search", - Name: "Notion Search", - Description: "Search Notion pages/databases", - Category: "productivity", - Icon: "Search", - Keywords: []string{"notion", "search", "page", "database", "find"}, - UseCases: []string{"Search Notion", "Find pages"}, - Parameters: "query: Search query", - }, - { - ID: "notion_query_database", - Name: "Notion Query Database", - Description: "Query a Notion database", - Category: "productivity", - Icon: "Database", - Keywords: []string{"notion", "database", "query", "filter", "table"}, - UseCases: []string{"Query databases", "Filter records"}, - Parameters: "database_id: Database ID, filter: Optional filter", - }, - { - ID: "notion_create_page", - Name: "Notion Create Page", - Description: "Create a Notion page", - Category: "productivity", - Icon: "FilePlus", - Keywords: []string{"notion", "create", "page", "new", "doc"}, - UseCases: []string{"Create pages", "Add documents"}, - Parameters: "parent_id: Parent page/database ID, properties: Page properties", - }, - { - ID: "notion_update_page", - Name: "Notion Update Page", - Description: "Update a Notion page", - Category: "productivity", - Icon: "Edit", - Keywords: []string{"notion", "update", "edit", "page", "modify"}, - UseCases: []string{"Update pages", "Edit documents"}, - Parameters: "page_id: Page ID, properties: Properties to update", - }, - { - ID: "airtable_list", - Name: "Airtable List Records", - Description: "List Airtable records", - Category: "productivity", - Icon: "Table", - Keywords: []string{"airtable", "list", "records", "table", "database"}, - UseCases: []string{"List records", "View table data"}, - Parameters: "base_id: Base ID, table_name: Table name", - }, - { - ID: "airtable_read", - Name: "Airtable Read Record", - Description: "Read a single Airtable record", - Category: "productivity", - Icon: "Eye", - Keywords: []string{"airtable", "read", "record", "get", "single"}, - UseCases: []string{"Read records", "Get single record"}, - Parameters: "base_id: Base ID, table_name: Table name, record_id: Record ID", - }, - { - ID: "airtable_create", - Name: "Airtable Create Record", - Description: "Create an Airtable record", - Category: "productivity", - Icon: "Plus", - Keywords: []string{"airtable", "create", "record", "new", "add"}, - UseCases: []string{"Create records", "Add data"}, - Parameters: "base_id: Base ID, table_name: Table name, fields: Record fields", - }, - { - ID: "airtable_update", - Name: "Airtable Update Record", - Description: "Update an Airtable record", - Category: "productivity", - Icon: "Edit", - Keywords: []string{"airtable", "update", "record", "edit", "modify"}, - UseCases: []string{"Update records", "Modify data"}, - Parameters: "base_id: Base ID, table_name: Table name, record_id: Record ID, fields: Fields to update", - }, - - // ═══════════════════════════════════════════════════════════════ - // 🛒 E-COMMERCE - // ═══════════════════════════════════════════════════════════════ - { - ID: "shopify_products", - Name: "Shopify Products", - Description: "List Shopify products", - Category: "ecommerce", - Icon: "ShoppingBag", - Keywords: []string{"shopify", "product", "list", "inventory", "catalog"}, - UseCases: []string{"List products", "View inventory"}, - Parameters: "None required", - }, - { - ID: "shopify_orders", - Name: "Shopify Orders", - Description: "List Shopify orders", - Category: "ecommerce", - Icon: "ShoppingCart", - Keywords: []string{"shopify", "order", "list", "sales", "purchase"}, - UseCases: []string{"List orders", "View sales"}, - Parameters: "status: Optional status filter", - }, - { - ID: "shopify_customers", - Name: "Shopify Customers", - Description: "List Shopify customers", - Category: "ecommerce", - Icon: "Users", - Keywords: []string{"shopify", "customer", "list", "buyer"}, - UseCases: []string{"List customers", "View buyers"}, - Parameters: "None required", - }, - - // ═══════════════════════════════════════════════════════════════ - // 🗄️ DATABASE - // ═══════════════════════════════════════════════════════════════ - { - ID: "mongodb_query", - Name: "MongoDB Query", - Description: "Query MongoDB collections - find, aggregate, count documents", - Category: "database", - Icon: "Database", - Keywords: []string{"mongodb", "mongo", "database", "query", "find", "aggregate", "nosql", "document", "collection"}, - UseCases: []string{"Query MongoDB collections", "Find documents", "Aggregate data", "Count records"}, - Parameters: "action: find|aggregate|count, collection: Collection name, filter: Query filter, pipeline: Aggregation pipeline", - }, - { - ID: "mongodb_write", - Name: "MongoDB Write", - Description: "Write to MongoDB - insert or update documents (delete not permitted)", - Category: "database", - Icon: "DatabaseBackup", - Keywords: []string{"mongodb", "mongo", "database", "insert", "update", "write", "create", "modify", "insertOne", "insertMany", "updateOne", "updateMany"}, - UseCases: []string{"Insert single document", "Insert multiple documents", "Update single record", "Update multiple records"}, - Parameters: "action: insertOne|insertMany|updateOne|updateMany, collection: Collection name, document: Document to insert, documents: Array for insertMany, filter: Update filter, update: Update operations", - CodeBlockExample: `{"action": "insertOne", "collection": "users", "document": {"name": "John", "email": "john@example.com"}}`, - }, - { - ID: "redis_read", - Name: "Redis Read", - Description: "Read from Redis - get keys, scan, list operations", - Category: "database", - Icon: "Database", - Keywords: []string{"redis", "cache", "key-value", "read", "get", "scan", "list", "hash", "set"}, - UseCases: []string{"Get cached values", "Read keys", "Scan patterns", "List operations"}, - Parameters: "action: get|mget|scan|hgetall|lrange|smembers, key: Redis key, pattern: Scan pattern", - CodeBlockExample: `{"action": "get", "key": "{{start.input}}"}`, - }, - { - ID: "redis_write", - Name: "Redis Write", - Description: "Write to Redis - set keys, lists, hashes, with TTL support", - Category: "database", - Icon: "DatabaseBackup", - Keywords: []string{"redis", "cache", "key-value", "write", "set", "expire", "list", "hash", "push"}, - UseCases: []string{"Set cache values", "Store data", "Queue operations", "Set expiry"}, - Parameters: "action: set|mset|hset|lpush|rpush|sadd|del, key: Redis key, value: Value to set, ttl: Optional TTL in seconds", - CodeBlockExample: `{"action": "set", "key": "{{start.input}}", "value": "{{previous-block.response}}"}`, - }, - - // ═══════════════════════════════════════════════════════════════ - // 🐦 SOCIAL MEDIA - // ═══════════════════════════════════════════════════════════════ - { - ID: "x_search_posts", - Name: "X Search Posts", - Description: "Search X/Twitter posts", - Category: "social_media", - Icon: "Twitter", - Keywords: []string{"twitter", "x", "search", "tweet", "post", "social"}, - UseCases: []string{"Search tweets", "Find posts"}, - Parameters: "query: Search query", - }, - { - ID: "x_post_tweet", - Name: "X Post Tweet", - Description: "Post to X/Twitter", - Category: "social_media", - Icon: "Send", - Keywords: []string{"twitter", "x", "post", "tweet", "publish"}, - UseCases: []string{"Post tweets", "Share content"}, - Parameters: "text: Tweet text", - }, - { - ID: "x_get_user", - Name: "X Get User", - Description: "Get X/Twitter user info", - Category: "social_media", - Icon: "User", - Keywords: []string{"twitter", "x", "user", "profile", "account"}, - UseCases: []string{"Get user info", "View profiles"}, - Parameters: "username: Twitter username", - }, - { - ID: "x_get_user_posts", - Name: "X Get User Posts", - Description: "Get user's X/Twitter posts", - Category: "social_media", - Icon: "List", - Keywords: []string{"twitter", "x", "user", "posts", "tweets", "timeline"}, - UseCases: []string{"Get user tweets", "View timeline"}, - Parameters: "user_id: User ID", - }, -} - -// ToolCategoryRegistry defines all tool categories -var ToolCategoryRegistry = []ToolCategory{ - {ID: "data_analysis", Name: "Data & Analysis", Icon: "BarChart2", Description: "Analyze data, create charts, and work with spreadsheets"}, - {ID: "search_web", Name: "Search & Web", Icon: "Search", Description: "Search the web, scrape URLs, and download files"}, - {ID: "content_creation", Name: "Content Creation", Icon: "FileText", Description: "Create documents, presentations, and images"}, - {ID: "media_processing", Name: "Media Processing", Icon: "Mic", Description: "Transcribe audio and analyze images"}, - {ID: "utilities", Name: "Utilities", Icon: "Clock", Description: "Time, code execution, and API requests"}, - {ID: "messaging", Name: "Messaging", Icon: "MessageCircle", Description: "Send messages via Discord, Slack, email, SMS, etc."}, - {ID: "video_conferencing", Name: "Video Conferencing", Icon: "Video", Description: "Zoom meetings, webinars, and Calendly"}, - {ID: "project_management", Name: "Project Management", Icon: "CheckSquare", Description: "Jira, Linear, ClickUp, Trello, Asana"}, - {ID: "crm_sales", Name: "CRM & Sales", Icon: "Users", Description: "HubSpot, LeadSquared, Mailchimp"}, - {ID: "analytics", Name: "Analytics", Icon: "BarChart", Description: "PostHog and Mixpanel tracking"}, - {ID: "code_devops", Name: "Code & DevOps", Icon: "GitBranch", Description: "GitHub, GitLab, and Netlify"}, - {ID: "productivity", Name: "Productivity", Icon: "Layout", Description: "Notion and Airtable"}, - {ID: "ecommerce", Name: "E-Commerce", Icon: "ShoppingBag", Description: "Shopify products, orders, and customers"}, - {ID: "social_media", Name: "Social Media", Icon: "Twitter", Description: "X/Twitter posts and interactions"}, - {ID: "database", Name: "Database", Icon: "Database", Description: "MongoDB and Redis database operations"}, -} - -// GetToolsByCategory returns all tools in a given category -func GetToolsByCategory(categoryID string) []ToolDefinition { - var tools []ToolDefinition - for _, tool := range ToolRegistry { - if tool.Category == categoryID { - tools = append(tools, tool) - } - } - return tools -} - -// GetToolByID returns a tool by its ID -func GetToolByID(toolID string) *ToolDefinition { - for _, tool := range ToolRegistry { - if tool.ID == toolID { - return &tool - } - } - return nil -} - -// GetAllToolIDs returns all tool IDs -func GetAllToolIDs() []string { - ids := make([]string, len(ToolRegistry)) - for i, tool := range ToolRegistry { - ids[i] = tool.ID - } - return ids -} - -// BuildToolPromptSection builds a prompt section for specific tools -func BuildToolPromptSection(toolIDs []string) string { - if len(toolIDs) == 0 { - return "" - } - - // Group tools by category for better organization - categoryTools := make(map[string][]ToolDefinition) - for _, toolID := range toolIDs { - if tool := GetToolByID(toolID); tool != nil { - categoryTools[tool.Category] = append(categoryTools[tool.Category], *tool) - } - } - - var builder strings.Builder - builder.WriteString("=== AVAILABLE TOOLS (Selected for this workflow) ===\n\n") - - // Get category info for display - categoryInfo := make(map[string]ToolCategory) - for _, cat := range ToolCategoryRegistry { - categoryInfo[cat.ID] = cat - } - - for catID, tools := range categoryTools { - if cat, ok := categoryInfo[catID]; ok { - builder.WriteString(cat.Name + ":\n") - } - for _, tool := range tools { - builder.WriteString("- " + tool.ID + ": " + tool.Description + "\n") - if tool.Parameters != "" { - builder.WriteString(" Parameters: " + tool.Parameters + "\n") - } - // Include code_block example if available - shows how to configure argumentMapping - if tool.CodeBlockExample != "" { - builder.WriteString(" code_block argumentMapping: " + tool.CodeBlockExample + "\n") - } - } - builder.WriteString("\n") - } - - return builder.String() -} diff --git a/backend/internal/services/tool_service.go b/backend/internal/services/tool_service.go deleted file mode 100644 index 96ed6c6c..00000000 --- a/backend/internal/services/tool_service.go +++ /dev/null @@ -1,196 +0,0 @@ -package services - -import ( - "claraverse/internal/tools" - "context" - "log" - "strings" -) - -// ToolService handles tool-related operations with credential awareness. -// It filters tools based on user's configured credentials to ensure -// only usable tools are sent to the LLM. -type ToolService struct { - toolRegistry *tools.Registry - credentialService *CredentialService -} - -// NewToolService creates a new tool service -func NewToolService(registry *tools.Registry, credentialService *CredentialService) *ToolService { - return &ToolService{ - toolRegistry: registry, - credentialService: credentialService, - } -} - -// GetAvailableTools returns tools filtered by user's credentials. -// - Tools not in ToolIntegrationMap are always included (no credential needed) -// - Tools in ToolIntegrationMap are only included if user has a credential for that integration type -func (s *ToolService) GetAvailableTools(ctx context.Context, userID string) []map[string]interface{} { - // Get all tools for user (built-in + MCP) - allTools := s.toolRegistry.GetUserTools(userID) - - // If no credential service, return all tools (fallback for dev mode or tests) - if s.credentialService == nil { - log.Printf("⚠️ [TOOL-SERVICE] No credential service, returning all %d tools", len(allTools)) - return allTools - } - - // Get user's configured integration types - userIntegrations, err := s.GetUserIntegrationTypes(ctx, userID) - if err != nil { - log.Printf("⚠️ [TOOL-SERVICE] Could not fetch user credentials, returning all tools: %v", err) - return allTools // Graceful degradation - } - - // Filter tools based on credential requirements - var filteredTools []map[string]interface{} - excludedCount := 0 - - for _, toolDef := range allTools { - toolName := extractToolName(toolDef) - if toolName == "" { - continue - } - - // Check if this is an MCP tool (user-specific local client tools) - isMCPTool := isUserSpecificTool(toolDef, userID) - - // Check if tool requires a credential - requiredIntegration := tools.GetIntegrationTypeForTool(toolName) - - if requiredIntegration == "" { - // Tool is NOT in integration mapping - if isMCPTool { - // MCP tools without explicit mapping are excluded by default (security) - // Only include MCP tools that are explicitly mapped as not needing credentials - log.Printf("🔒 [TOOL-SERVICE] Excluding unmapped MCP tool: %s (requires explicit mapping)", toolName) - excludedCount++ - } else { - // Built-in tool that doesn't require credentials - always include - filteredTools = append(filteredTools, toolDef) - } - } else if userIntegrations[requiredIntegration] { - // Tool requires credentials AND user has them - include - filteredTools = append(filteredTools, toolDef) - } else { - // Tool requires credentials user doesn't have - exclude - excludedCount++ - } - } - - log.Printf("🔧 [TOOL-SERVICE] Filtered tools for user %s: %d available, %d excluded (missing credentials)", - userID, len(filteredTools), excludedCount) - - return filteredTools -} - -// GetUserIntegrationTypes returns a set of integration types the user has credentials for -func (s *ToolService) GetUserIntegrationTypes(ctx context.Context, userID string) (map[string]bool, error) { - if s.credentialService == nil { - return make(map[string]bool), nil - } - - credentials, err := s.credentialService.ListByUser(ctx, userID) - if err != nil { - return nil, err - } - - integrations := make(map[string]bool) - for _, cred := range credentials { - integrations[cred.IntegrationType] = true - } - - return integrations, nil -} - -// extractToolName extracts the tool name from an OpenAI tool definition -func extractToolName(toolDef map[string]interface{}) string { - fn, ok := toolDef["function"].(map[string]interface{}) - if !ok { - return "" - } - name, ok := fn["name"].(string) - if !ok { - return "" - } - return name -} - -// isUserSpecificTool checks if a tool is an MCP tool (user-specific, not built-in) -// MCP tools are registered per-user and should be filtered by credentials by default -func isUserSpecificTool(toolDef map[string]interface{}, userID string) bool { - // Check if tool has user_id metadata (MCP tools have this) - if metadata, ok := toolDef["metadata"].(map[string]interface{}); ok { - if toolUserID, ok := metadata["user_id"].(string); ok && toolUserID == userID { - return true - } - } - - // Fallback: Check if tool name suggests it's an MCP tool - // MCP tools often have specific naming patterns (e.g., containing "gmail", "calendar", "notion", etc.) - toolName := extractToolName(toolDef) - mcpPatterns := []string{ - "gmail", "calendar", "drive", "sheets", "docs", "slack", "discord", - "notion", "trello", "asana", "jira", "linear", "github", "gitlab", - "spotify", "twitter", "youtube", "reddit", "instagram", - } - - toolNameLower := strings.ToLower(toolName) - for _, pattern := range mcpPatterns { - if strings.Contains(toolNameLower, pattern) { - // Check if it's NOT a built-in Composio tool (which start with the integration name) - // Built-in: "gmail_send_email", MCP: "send_gmail_message" - if !strings.HasPrefix(toolNameLower, pattern+"_") { - return true - } - } - } - - return false -} - -// GetCredentialForTool returns the credential ID for a tool that requires credentials. -// Returns empty string if no credential is needed or not found. -func (s *ToolService) GetCredentialForTool(ctx context.Context, userID string, toolName string) string { - if s.credentialService == nil { - return "" - } - - // Check if tool requires a credential - integrationType := tools.GetIntegrationTypeForTool(toolName) - if integrationType == "" { - return "" // Tool doesn't require credentials - } - - // Get credentials for this integration type - credentials, err := s.credentialService.ListByUserAndType(ctx, userID, integrationType) - if err != nil { - log.Printf("⚠️ [TOOL-SERVICE] Error getting credentials for %s: %v", integrationType, err) - return "" - } - - if len(credentials) == 0 { - log.Printf("⚠️ [TOOL-SERVICE] No %s credentials found for user %s", integrationType, userID) - return "" - } - - // Use the first credential (or the only one) - credentialID := credentials[0].ID - log.Printf("🔐 [TOOL-SERVICE] Found credential %s for tool %s (type: %s)", credentialID, toolName, integrationType) - return credentialID -} - -// CreateCredentialResolver creates a credential resolver function for a user. -// Returns nil if credential service is not available. -func (s *ToolService) CreateCredentialResolver(userID string) tools.CredentialResolver { - if s.credentialService == nil { - return nil - } - return s.credentialService.CreateCredentialResolver(userID) -} - -// GetCredentialService returns the underlying credential service (for advanced use cases) -func (s *ToolService) GetCredentialService() *CredentialService { - return s.credentialService -} diff --git a/backend/internal/services/usage_limiter_service.go b/backend/internal/services/usage_limiter_service.go deleted file mode 100644 index bc25dae5..00000000 --- a/backend/internal/services/usage_limiter_service.go +++ /dev/null @@ -1,384 +0,0 @@ -package services - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "context" - "fmt" - "time" - - "github.com/redis/go-redis/v9" - "go.mongodb.org/mongo-driver/bson" -) - -// UsageLimiterService tracks and enforces usage limits for messages, file uploads, and image generation -type UsageLimiterService struct { - tierService *TierService - redis *redis.Client - mongoDB *database.MongoDB -} - -// UsageLimiterStats holds current usage statistics for a user -type UsageLimiterStats struct { - MessagesUsed int64 `json:"messages_used"` - FileUploadsUsed int64 `json:"file_uploads_used"` - ImageGensUsed int64 `json:"image_gens_used"` - MessageResetAt time.Time `json:"message_reset_at"` - FileUploadResetAt time.Time `json:"file_upload_reset_at"` - ImageGenResetAt time.Time `json:"image_gen_reset_at"` -} - -// LimitExceededError represents a rate limit error -type LimitExceededError struct { - ErrorCode string `json:"error_code"` - Message string `json:"message"` - Limit int64 `json:"limit"` - Used int64 `json:"used"` - ResetAt time.Time `json:"reset_at"` - UpgradeTo string `json:"upgrade_to"` -} - -func (e *LimitExceededError) Error() string { - return e.Message -} - -// NewUsageLimiterService creates a new usage limiter service -func NewUsageLimiterService(tierService *TierService, redis *redis.Client, mongoDB *database.MongoDB) *UsageLimiterService { - return &UsageLimiterService{ - tierService: tierService, - redis: redis, - mongoDB: mongoDB, - } -} - -// ========== Message Limits (Monthly - Billing Cycle Reset) ========== - -// CheckMessageLimit checks if user can send another message this month -func (s *UsageLimiterService) CheckMessageLimit(ctx context.Context, userID string) error { - limits := s.tierService.GetLimits(ctx, userID) - - // Unlimited - if limits.MaxMessagesPerMonth < 0 { - return nil - } - - // Get current count - count, err := s.GetMonthlyMessageCount(ctx, userID) - if err != nil { - // On error, allow request (fail open) - return nil - } - - // Check limit - if count >= limits.MaxMessagesPerMonth { - resetAt, _ := s.getMonthlyResetTime(ctx, userID) - return &LimitExceededError{ - ErrorCode: "message_limit_exceeded", - Message: fmt.Sprintf("Monthly message limit reached (%d/%d). Resets on %s. Upgrade to Pro for 3,000 messages/month.", count, limits.MaxMessagesPerMonth, resetAt.Format("Jan 2")), - Limit: limits.MaxMessagesPerMonth, - Used: count, - ResetAt: resetAt, - UpgradeTo: s.getSuggestedUpgradeTier(s.tierService.GetUserTier(ctx, userID)), - } - } - - return nil -} - -// IncrementMessageCount increments the user's monthly message count -func (s *UsageLimiterService) IncrementMessageCount(ctx context.Context, userID string) error { - key, err := s.getMessageKey(ctx, userID) - if err != nil { - return err - } - - // Increment counter - _, err = s.redis.Incr(ctx, key).Result() - if err != nil { - return err - } - - // Set expiry (billing period end + 30 days buffer) - resetAt, err := s.getMonthlyResetTime(ctx, userID) - if err == nil { - expiry := time.Until(resetAt.AddDate(0, 1, 0)) // Add 30 days buffer - s.redis.Expire(ctx, key, expiry) - } - - return nil -} - -// GetMonthlyMessageCount returns the user's current message count for this billing period -func (s *UsageLimiterService) GetMonthlyMessageCount(ctx context.Context, userID string) (int64, error) { - key, err := s.getMessageKey(ctx, userID) - if err != nil { - return 0, err - } - - count, err := s.redis.Get(ctx, key).Int64() - if err == redis.Nil { - return 0, nil - } - return count, err -} - -// ========== File Upload Limits (Daily - Midnight UTC Reset) ========== - -// CheckFileUploadLimit checks if user can upload another file today -func (s *UsageLimiterService) CheckFileUploadLimit(ctx context.Context, userID string) error { - limits := s.tierService.GetLimits(ctx, userID) - - // Unlimited - if limits.MaxFileUploadsPerDay < 0 { - return nil - } - - // Get current count - count, err := s.GetDailyFileUploadCount(ctx, userID) - if err != nil { - // On error, allow request (fail open) - return nil - } - - // Check limit - if count >= limits.MaxFileUploadsPerDay { - resetAt := s.getNextMidnightUTC() - return &LimitExceededError{ - ErrorCode: "file_upload_limit_exceeded", - Message: fmt.Sprintf("Daily file upload limit reached (%d/%d). Resets at midnight UTC. Upgrade to Pro for 10 uploads/day.", count, limits.MaxFileUploadsPerDay), - Limit: limits.MaxFileUploadsPerDay, - Used: count, - ResetAt: resetAt, - UpgradeTo: s.getSuggestedUpgradeTier(s.tierService.GetUserTier(ctx, userID)), - } - } - - return nil -} - -// IncrementFileUploadCount increments the user's daily file upload count -func (s *UsageLimiterService) IncrementFileUploadCount(ctx context.Context, userID string) error { - key := s.getFileUploadKey(userID) - - // Increment counter - _, err := s.redis.Incr(ctx, key).Result() - if err != nil { - return err - } - - // Set expiry to next midnight + 24 hours buffer - resetAt := s.getNextMidnightUTC() - expiry := time.Until(resetAt.Add(24 * time.Hour)) - s.redis.Expire(ctx, key, expiry) - - return nil -} - -// GetDailyFileUploadCount returns the user's current file upload count for today -func (s *UsageLimiterService) GetDailyFileUploadCount(ctx context.Context, userID string) (int64, error) { - key := s.getFileUploadKey(userID) - - count, err := s.redis.Get(ctx, key).Int64() - if err == redis.Nil { - return 0, nil - } - return count, err -} - -// ========== Image Generation Limits (Daily - Midnight UTC Reset) ========== - -// CheckImageGenLimit checks if user can generate another image today -func (s *UsageLimiterService) CheckImageGenLimit(ctx context.Context, userID string) error { - limits := s.tierService.GetLimits(ctx, userID) - - // Unlimited - if limits.MaxImageGensPerDay < 0 { - return nil - } - - // Get current count - count, err := s.GetDailyImageGenCount(ctx, userID) - if err != nil { - // On error, allow request (fail open) - return nil - } - - // Check limit - if count >= limits.MaxImageGensPerDay { - resetAt := s.getNextMidnightUTC() - return &LimitExceededError{ - ErrorCode: "image_gen_limit_exceeded", - Message: fmt.Sprintf("Daily image generation limit reached (%d/%d). Resets at midnight UTC. Upgrade to Pro for 25 images/day.", count, limits.MaxImageGensPerDay), - Limit: limits.MaxImageGensPerDay, - Used: count, - ResetAt: resetAt, - UpgradeTo: s.getSuggestedUpgradeTier(s.tierService.GetUserTier(ctx, userID)), - } - } - - return nil -} - -// IncrementImageGenCount increments the user's daily image generation count -func (s *UsageLimiterService) IncrementImageGenCount(ctx context.Context, userID string) error { - key := s.getImageGenKey(userID) - - // Increment counter - _, err := s.redis.Incr(ctx, key).Result() - if err != nil { - return err - } - - // Set expiry to next midnight + 24 hours buffer - resetAt := s.getNextMidnightUTC() - expiry := time.Until(resetAt.Add(24 * time.Hour)) - s.redis.Expire(ctx, key, expiry) - - return nil -} - -// GetDailyImageGenCount returns the user's current image generation count for today -func (s *UsageLimiterService) GetDailyImageGenCount(ctx context.Context, userID string) (int64, error) { - key := s.getImageGenKey(userID) - - count, err := s.redis.Get(ctx, key).Int64() - if err == redis.Nil { - return 0, nil - } - return count, err -} - -// ========== Utility Methods ========== - -// GetUsageStats returns comprehensive usage statistics for a user -func (s *UsageLimiterService) GetUsageStats(ctx context.Context, userID string) (*UsageLimiterStats, error) { - msgCount, _ := s.GetMonthlyMessageCount(ctx, userID) - fileCount, _ := s.GetDailyFileUploadCount(ctx, userID) - imageCount, _ := s.GetDailyImageGenCount(ctx, userID) - - msgResetAt, _ := s.getMonthlyResetTime(ctx, userID) - dailyResetAt := s.getNextMidnightUTC() - - return &UsageLimiterStats{ - MessagesUsed: msgCount, - FileUploadsUsed: fileCount, - ImageGensUsed: imageCount, - MessageResetAt: msgResetAt, - FileUploadResetAt: dailyResetAt, - ImageGenResetAt: dailyResetAt, - }, nil -} - -// ResetMonthlyCounters resets the monthly message counter for a user -func (s *UsageLimiterService) ResetMonthlyCounters(ctx context.Context, userID string) error { - key, err := s.getMessageKey(ctx, userID) - if err != nil { - return err - } - return s.redis.Del(ctx, key).Err() -} - -// ResetAllCounters resets all usage counters for a user (used on tier upgrade) -func (s *UsageLimiterService) ResetAllCounters(ctx context.Context, userID string) error { - // Reset monthly message counter - msgKey, _ := s.getMessageKey(ctx, userID) - s.redis.Del(ctx, msgKey) - - // Reset daily file upload counter - fileKey := s.getFileUploadKey(userID) - s.redis.Del(ctx, fileKey) - - // Reset daily image gen counter - imageKey := s.getImageGenKey(userID) - s.redis.Del(ctx, imageKey) - - return nil -} - -// ========== Private Helper Methods ========== - -// getMessageKey generates the Redis key for monthly message count -func (s *UsageLimiterService) getMessageKey(ctx context.Context, userID string) (string, error) { - billingPeriodKey, err := s.getBillingPeriodKey(ctx, userID) - if err != nil { - // Fallback to calendar month for free users - billingPeriodKey = time.Now().UTC().Format("2006-01") - } - return fmt.Sprintf("messages:%s:%s", userID, billingPeriodKey), nil -} - -// getFileUploadKey generates the Redis key for daily file upload count -func (s *UsageLimiterService) getFileUploadKey(userID string) string { - date := time.Now().UTC().Format("2006-01-02") - return fmt.Sprintf("file_uploads:%s:%s", userID, date) -} - -// getImageGenKey generates the Redis key for daily image generation count -func (s *UsageLimiterService) getImageGenKey(userID string) string { - date := time.Now().UTC().Format("2006-01-02") - return fmt.Sprintf("image_gens:%s:%s", userID, date) -} - -// getBillingPeriodKey returns a unique key for the current billing period -func (s *UsageLimiterService) getBillingPeriodKey(ctx context.Context, userID string) (string, error) { - // Get subscription from MongoDB - subscription, err := s.getSubscription(ctx, userID) - if err != nil || subscription == nil { - // Free tier - use calendar month - return time.Now().UTC().Format("2006-01"), nil - } - - // Paid tier - use billing cycle start date - return subscription.CurrentPeriodStart.Format("2006-01-02"), nil -} - -// getSubscription retrieves the user's subscription from MongoDB -func (s *UsageLimiterService) getSubscription(ctx context.Context, userID string) (*models.Subscription, error) { - if s.mongoDB == nil { - return nil, fmt.Errorf("MongoDB not available") - } - - collection := s.mongoDB.Database().Collection("subscriptions") - var subscription models.Subscription - - err := collection.FindOne(ctx, bson.M{"userId": userID}).Decode(&subscription) - if err != nil { - return nil, err - } - - return &subscription, nil -} - -// getMonthlyResetTime returns when the monthly message count will reset -func (s *UsageLimiterService) getMonthlyResetTime(ctx context.Context, userID string) (time.Time, error) { - subscription, err := s.getSubscription(ctx, userID) - if err != nil || subscription == nil { - // Free tier - reset at end of current month (first day of next month at midnight) - now := time.Now().UTC() - // Get first day of next month properly (handles year rollover) - firstDayNextMonth := time.Date(now.Year(), now.Month(), 1, 0, 0, 0, 0, time.UTC).AddDate(0, 1, 0) - return firstDayNextMonth, nil - } - - // Paid tier - reset at billing cycle end - return subscription.CurrentPeriodEnd, nil -} - -// getNextMidnightUTC returns the next midnight UTC time -func (s *UsageLimiterService) getNextMidnightUTC() time.Time { - now := time.Now().UTC() - tomorrow := now.AddDate(0, 0, 1) - return time.Date(tomorrow.Year(), tomorrow.Month(), tomorrow.Day(), 0, 0, 0, 0, time.UTC) -} - -// getSuggestedUpgradeTier suggests which tier to upgrade to based on current tier -func (s *UsageLimiterService) getSuggestedUpgradeTier(currentTier string) string { - switch currentTier { - case models.TierFree: - return "pro" - case models.TierPro: - return "max" - default: - return "max" - } -} diff --git a/backend/internal/services/user_service.go b/backend/internal/services/user_service.go deleted file mode 100644 index c6abad36..00000000 --- a/backend/internal/services/user_service.go +++ /dev/null @@ -1,502 +0,0 @@ -package services - -import ( - "claraverse/internal/config" - "claraverse/internal/database" - "claraverse/internal/models" - "context" - "fmt" - "log" - "time" - - "go.mongodb.org/mongo-driver/bson" - "go.mongodb.org/mongo-driver/bson/primitive" - "go.mongodb.org/mongo-driver/mongo" - "go.mongodb.org/mongo-driver/mongo/options" -) - -// UserService handles user operations with MongoDB -type UserService struct { - db *database.MongoDB - collection *mongo.Collection - config *config.Config - usageLimiter *UsageLimiterService -} - -// NewUserService creates a new user service -// usageLimiter can be nil and set later via SetUsageLimiter -func NewUserService(db *database.MongoDB, cfg *config.Config, usageLimiter *UsageLimiterService) *UserService { - return &UserService{ - db: db, - collection: db.Collection(database.CollectionUsers), - config: cfg, - usageLimiter: usageLimiter, - } -} - -// SetUsageLimiter sets the usage limiter (for deferred initialization) -func (s *UserService) SetUsageLimiter(limiter *UsageLimiterService) { - s.usageLimiter = limiter -} - -// SyncUserFromSupabase creates or updates a user from Supabase authentication -// This should be called on every authenticated request to keep user data in sync -func (s *UserService) SyncUserFromSupabase(ctx context.Context, supabaseUserID, email string) (*models.User, error) { - if supabaseUserID == "" { - return nil, fmt.Errorf("supabase user ID is required") - } - - now := time.Now() - - // Determine subscription tier based on promo eligibility - subscriptionTier := models.TierFree - var subscriptionExpiresAt *time.Time - - if s.isPromoEligible(now) { - subscriptionTier = models.TierPro - expiresAt := now.Add(time.Duration(s.config.PromoDuration) * 24 * time.Hour) - subscriptionExpiresAt = &expiresAt - } - - // Use upsert to create or update user - filter := bson.M{"supabaseUserId": supabaseUserID} - setOnInsertFields := bson.M{ - "supabaseUserId": supabaseUserID, - "createdAt": now, - "subscriptionTier": subscriptionTier, - "subscriptionStatus": models.SubStatusActive, - "preferences": models.UserPreferences{ - StoreBuilderChatHistory: true, // Default to storing chat history - }, - } - - if subscriptionExpiresAt != nil { - setOnInsertFields["subscriptionExpiresAt"] = subscriptionExpiresAt - } - - update := bson.M{ - "$set": bson.M{ - "email": email, - "lastLoginAt": now, - }, - "$setOnInsert": setOnInsertFields, - } - - opts := options.FindOneAndUpdate(). - SetUpsert(true). - SetReturnDocument(options.After) - - var user models.User - err := s.collection.FindOneAndUpdate(ctx, filter, update, opts).Decode(&user) - if err != nil { - return nil, fmt.Errorf("failed to sync user: %w", err) - } - - // Reset usage counters for NEW promo users to ensure clean slate - // A new user is detected by checking if createdAt is very close to now (within 2 seconds) - if user.SubscriptionTier == models.TierPro && user.CreatedAt.After(now.Add(-2*time.Second)) { - if s.usageLimiter != nil { - if err := s.usageLimiter.ResetAllCounters(ctx, supabaseUserID); err != nil { - log.Printf("⚠️ Failed to reset usage counters for new promo user %s: %v", supabaseUserID, err) - } else { - log.Printf("✅ Reset usage counters for new promo user %s", supabaseUserID) - } - } - } - - return &user, nil -} - -// GetUserBySupabaseID retrieves a user by their Supabase user ID -func (s *UserService) GetUserBySupabaseID(ctx context.Context, supabaseUserID string) (*models.User, error) { - var user models.User - err := s.collection.FindOne(ctx, bson.M{"supabaseUserId": supabaseUserID}).Decode(&user) - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("user not found") - } - if err != nil { - return nil, fmt.Errorf("failed to get user: %w", err) - } - return &user, nil -} - -// GetUserByID retrieves a user by their MongoDB ID -func (s *UserService) GetUserByID(ctx context.Context, userID primitive.ObjectID) (*models.User, error) { - var user models.User - err := s.collection.FindOne(ctx, bson.M{"_id": userID}).Decode(&user) - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("user not found") - } - if err != nil { - return nil, fmt.Errorf("failed to get user: %w", err) - } - return &user, nil -} - -// GetUserByEmail retrieves a user by their email address -func (s *UserService) GetUserByEmail(ctx context.Context, email string) (*models.User, error) { - var user models.User - err := s.collection.FindOne(ctx, bson.M{"email": email}).Decode(&user) - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("user not found") - } - if err != nil { - return nil, fmt.Errorf("failed to get user: %w", err) - } - return &user, nil -} - -// CreateUser creates a new user (for local auth registration) -func (s *UserService) CreateUser(ctx context.Context, user *models.User) error { - result, err := s.collection.InsertOne(ctx, user) - if err != nil { - return fmt.Errorf("failed to create user: %w", err) - } - - // Update user ID with the inserted ID - if oid, ok := result.InsertedID.(primitive.ObjectID); ok { - user.ID = oid - } - - return nil -} - -// UpdateUser updates an existing user -func (s *UserService) UpdateUser(ctx context.Context, user *models.User) error { - filter := bson.M{"_id": user.ID} - update := bson.M{"$set": user} - - result, err := s.collection.UpdateOne(ctx, filter, update) - if err != nil { - return fmt.Errorf("failed to update user: %w", err) - } - if result.MatchedCount == 0 { - return fmt.Errorf("user not found") - } - - return nil -} - -// Collection returns the MongoDB collection (for direct access when needed) -func (s *UserService) Collection() *mongo.Collection { - return s.collection -} - -// UpdatePreferences updates a user's preferences -func (s *UserService) UpdatePreferences(ctx context.Context, supabaseUserID string, req *models.UpdateUserPreferencesRequest) (*models.UserPreferences, error) { - // Build update document - updateFields := bson.M{} - if req.StoreBuilderChatHistory != nil { - updateFields["preferences.storeBuilderChatHistory"] = *req.StoreBuilderChatHistory - } - if req.DefaultModelID != nil { - updateFields["preferences.defaultModelId"] = *req.DefaultModelID - } - if req.ToolPredictorModelID != nil { - updateFields["preferences.toolPredictorModelId"] = *req.ToolPredictorModelID - } - if req.ChatPrivacyMode != nil { - updateFields["preferences.chatPrivacyMode"] = *req.ChatPrivacyMode - } - if req.Theme != nil { - updateFields["preferences.theme"] = *req.Theme - } - if req.FontSize != nil { - updateFields["preferences.fontSize"] = *req.FontSize - } - - // Memory system preferences - if req.MemoryEnabled != nil { - updateFields["preferences.memoryEnabled"] = *req.MemoryEnabled - } - if req.MemoryExtractionThreshold != nil { - updateFields["preferences.memoryExtractionThreshold"] = *req.MemoryExtractionThreshold - } - if req.MemoryMaxInjection != nil { - updateFields["preferences.memoryMaxInjection"] = *req.MemoryMaxInjection - } - if req.MemoryExtractorModelID != nil { - updateFields["preferences.memoryExtractorModelId"] = *req.MemoryExtractorModelID - } - if req.MemorySelectorModelID != nil { - updateFields["preferences.memorySelectorModelId"] = *req.MemorySelectorModelID - } - - if len(updateFields) == 0 { - // No changes, just return current preferences - user, err := s.GetUserBySupabaseID(ctx, supabaseUserID) - if err != nil { - return nil, err - } - return &user.Preferences, nil - } - - filter := bson.M{"supabaseUserId": supabaseUserID} - update := bson.M{"$set": updateFields} - - opts := options.FindOneAndUpdate().SetReturnDocument(options.After) - - var user models.User - err := s.collection.FindOneAndUpdate(ctx, filter, update, opts).Decode(&user) - if err == mongo.ErrNoDocuments { - return nil, fmt.Errorf("user not found") - } - if err != nil { - return nil, fmt.Errorf("failed to update preferences: %w", err) - } - - return &user.Preferences, nil -} - -// GetPreferences retrieves a user's preferences -func (s *UserService) GetPreferences(ctx context.Context, supabaseUserID string) (*models.UserPreferences, error) { - user, err := s.GetUserBySupabaseID(ctx, supabaseUserID) - if err != nil { - return nil, err - } - return &user.Preferences, nil -} - -// MarkWelcomePopupSeen marks the welcome popup as seen for a user -func (s *UserService) MarkWelcomePopupSeen(ctx context.Context, supabaseUserID string) error { - filter := bson.M{"supabaseUserId": supabaseUserID} - update := bson.M{ - "$set": bson.M{ - "hasSeenWelcomePopup": true, - }, - } - - result, err := s.collection.UpdateOne(ctx, filter, update) - if err != nil { - return fmt.Errorf("failed to mark welcome popup as seen: %w", err) - } - if result.MatchedCount == 0 { - return fmt.Errorf("user not found") - } - - return nil -} - -// GetUserCount returns the total number of users (for admin analytics) -func (s *UserService) GetUserCount(ctx context.Context) (int64, error) { - count, err := s.collection.CountDocuments(ctx, bson.M{}) - if err != nil { - return 0, fmt.Errorf("failed to count users: %w", err) - } - return count, nil -} - -// ListUsers returns a paginated list of users (for admin) -func (s *UserService) ListUsers(ctx context.Context, skip, limit int64) ([]*models.User, error) { - opts := options.Find(). - SetSkip(skip). - SetLimit(limit). - SetSort(bson.M{"createdAt": -1}) - - cursor, err := s.collection.Find(ctx, bson.M{}, opts) - if err != nil { - return nil, fmt.Errorf("failed to list users: %w", err) - } - defer cursor.Close(ctx) - - var users []*models.User - if err := cursor.All(ctx, &users); err != nil { - return nil, fmt.Errorf("failed to decode users: %w", err) - } - - return users, nil -} - -// UpdateSubscription updates a user's subscription (for payment integration) -func (s *UserService) UpdateSubscription(ctx context.Context, supabaseUserID, tier string, expiresAt *time.Time) error { - return s.UpdateSubscriptionWithStatus(ctx, supabaseUserID, tier, "", expiresAt) -} - -// UpdateSubscriptionWithStatus updates a user's subscription with status -func (s *UserService) UpdateSubscriptionWithStatus(ctx context.Context, supabaseUserID, tier, status string, expiresAt *time.Time) error { - filter := bson.M{"supabaseUserId": supabaseUserID} - updateFields := bson.M{ - "subscriptionTier": tier, - } - if status != "" { - updateFields["subscriptionStatus"] = status - } - if expiresAt != nil { - updateFields["subscriptionExpiresAt"] = expiresAt - } - - update := bson.M{ - "$set": updateFields, - } - - result, err := s.collection.UpdateOne(ctx, filter, update) - if err != nil { - return fmt.Errorf("failed to update subscription: %w", err) - } - if result.MatchedCount == 0 { - return fmt.Errorf("user not found") - } - - return nil -} - -// UpdateDodoCustomer updates a user's DodoPayments customer ID -func (s *UserService) UpdateDodoCustomer(ctx context.Context, supabaseUserID, customerID string) error { - filter := bson.M{"supabaseUserId": supabaseUserID} - update := bson.M{ - "$set": bson.M{ - "dodoCustomerId": customerID, - }, - } - - result, err := s.collection.UpdateOne(ctx, filter, update) - if err != nil { - return fmt.Errorf("failed to update DodoPayments customer ID: %w", err) - } - if result.MatchedCount == 0 { - return fmt.Errorf("user not found") - } - - return nil -} - -// DeleteUser deletes a user and all their data (GDPR compliance) -func (s *UserService) DeleteUser(ctx context.Context, supabaseUserID string) error { - // Get user first to get their MongoDB ID - user, err := s.GetUserBySupabaseID(ctx, supabaseUserID) - if err != nil { - return err - } - - // Delete user document - result, err := s.collection.DeleteOne(ctx, bson.M{"_id": user.ID}) - if err != nil { - return fmt.Errorf("failed to delete user: %w", err) - } - if result.DeletedCount == 0 { - return fmt.Errorf("user not found") - } - - // Note: Related data (agents, conversations, etc.) should be deleted - // in a transaction or by the caller. This could be enhanced with - // cascade delete logic. - - return nil -} - -// isPromoEligible checks if a signup time is within the promotional window -func (s *UserService) isPromoEligible(signupTime time.Time) bool { - if s.config == nil || !s.config.PromoEnabled { - return false - } - - // Check if signup time is within promo window (UTC) - // Use After for start (exclusive) and Before for end (exclusive) - return !signupTime.Before(s.config.PromoStartDate) && - signupTime.Before(s.config.PromoEndDate) -} - -// SetLimitOverrides sets tier OR granular limit overrides for a user (admin only) -func (s *UserService) SetLimitOverrides(ctx context.Context, supabaseUserID, adminUserID, reason string, tier *string, limits *models.TierLimits) error { - if supabaseUserID == "" { - return fmt.Errorf("user ID is required") - } - if tier == nil && limits == nil { - return fmt.Errorf("either tier or limits must be provided") - } - - now := time.Now() - updateFields := bson.M{ - "overrideSetBy": adminUserID, - "overrideSetAt": now, - "overrideReason": reason, - } - - // Set tier override if provided - if tier != nil { - updateFields["tierOverride"] = *tier - // Clear limit overrides when setting tier - updateFields["limitOverrides"] = nil - } - - // Set granular limit overrides if provided - if limits != nil { - updateFields["limitOverrides"] = limits - // Clear tier override when setting limits - updateFields["tierOverride"] = nil - } - - filter := bson.M{"supabaseUserId": supabaseUserID} - update := bson.M{"$set": updateFields} - - result, err := s.collection.UpdateOne(ctx, filter, update) - if err != nil { - return fmt.Errorf("failed to set overrides: %w", err) - } - if result.MatchedCount == 0 { - return fmt.Errorf("user not found") - } - - if tier != nil { - log.Printf("🔐 Admin %s set tier override for user %s: %s (reason: %s)", adminUserID, supabaseUserID, *tier, reason) - } else { - log.Printf("🔐 Admin %s set granular limit overrides for user %s (reason: %s)", adminUserID, supabaseUserID, reason) - } - return nil -} - -// RemoveAllOverrides removes all overrides (tier and limits) for a user (admin only) -func (s *UserService) RemoveAllOverrides(ctx context.Context, supabaseUserID, adminUserID string) error { - if supabaseUserID == "" { - return fmt.Errorf("user ID is required") - } - - update := bson.M{ - "$unset": bson.M{ - "tierOverride": "", - "limitOverrides": "", - "overrideSetBy": "", - "overrideSetAt": "", - "overrideReason": "", - }, - } - - filter := bson.M{"supabaseUserId": supabaseUserID} - result, err := s.collection.UpdateOne(ctx, filter, update) - if err != nil { - return fmt.Errorf("failed to remove overrides: %w", err) - } - if result.MatchedCount == 0 { - return fmt.Errorf("user not found") - } - - log.Printf("🔐 Admin %s removed all overrides for user %s", adminUserID, supabaseUserID) - return nil -} - -// GetAdminUserDetails returns detailed user info for admin (includes override info) -func (s *UserService) GetAdminUserDetails(ctx context.Context, supabaseUserID string, tierService *TierService) (*models.AdminUserResponse, error) { - user, err := s.GetUserBySupabaseID(ctx, supabaseUserID) - if err != nil { - return nil, err - } - - // Get effective tier - effectiveTier := tierService.GetUserTier(ctx, supabaseUserID) - - // Get effective limits (with overrides applied) - effectiveLimits := tierService.GetLimits(ctx, supabaseUserID) - - return &models.AdminUserResponse{ - UserResponse: user.ToResponse(), - EffectiveTier: effectiveTier, - EffectiveLimits: effectiveLimits, - HasTierOverride: user.TierOverride != nil, - HasLimitOverrides: user.LimitOverrides != nil, - TierOverride: user.TierOverride, - LimitOverrides: user.LimitOverrides, - OverrideSetBy: user.OverrideSetBy, - OverrideSetAt: user.OverrideSetAt, - OverrideReason: user.OverrideReason, - }, nil -} diff --git a/backend/internal/services/vision_init.go b/backend/internal/services/vision_init.go deleted file mode 100644 index 6eece449..00000000 --- a/backend/internal/services/vision_init.go +++ /dev/null @@ -1,93 +0,0 @@ -package services - -import ( - "claraverse/internal/database" - "claraverse/internal/vision" - "fmt" - "log" - "sync" -) - -var ( - visionInitOnce sync.Once - visionProviderSvc *ProviderService - visionDB *database.DB -) - -// SetVisionDependencies sets the dependencies needed for vision service -// Must be called before InitVisionService -func SetVisionDependencies(providerService *ProviderService, db *database.DB) { - visionProviderSvc = providerService - visionDB = db -} - -// InitVisionService initializes the vision package with provider access -func InitVisionService() { - if visionProviderSvc == nil { - log.Println("⚠️ [VISION-INIT] Provider service not set, vision service disabled") - return - } - - visionInitOnce.Do(func() { - configService := GetConfigService() - - // Provider getter callback - providerGetter := func(id int) (*vision.Provider, error) { - p, err := visionProviderSvc.GetByID(id) - if err != nil { - return nil, err - } - return &vision.Provider{ - ID: p.ID, - Name: p.Name, - BaseURL: p.BaseURL, - APIKey: p.APIKey, - Enabled: p.Enabled, - }, nil - } - - // Vision model finder callback - visionModelFinder := func() (int, string, error) { - // First check aliases for vision-capable models - allAliases := configService.GetAllModelAliases() - - for providerID, aliases := range allAliases { - for _, aliasInfo := range aliases { - if aliasInfo.SupportsVision != nil && *aliasInfo.SupportsVision { - provider, err := visionProviderSvc.GetByID(providerID) - if err == nil && provider.Enabled { - log.Printf("🖼️ [VISION-INIT] Found vision model via alias: %s -> %s", aliasInfo.DisplayName, aliasInfo.ActualModel) - return providerID, aliasInfo.ActualModel, nil - } - } - } - } - - // Fallback: Check database for vision models - if visionDB == nil { - return 0, "", fmt.Errorf("database not available") - } - - var providerID int - var modelName string - err := visionDB.QueryRow(` - SELECT m.provider_id, m.name - FROM models m - JOIN providers p ON m.provider_id = p.id - WHERE m.supports_vision = 1 AND m.is_visible = 1 AND p.enabled = 1 - ORDER BY m.provider_id ASC - LIMIT 1 - `).Scan(&providerID, &modelName) - - if err != nil { - return 0, "", fmt.Errorf("no vision model found: %w", err) - } - - log.Printf("🖼️ [VISION-INIT] Found vision model from database: %s (provider: %d)", modelName, providerID) - return providerID, modelName, nil - } - - vision.InitService(providerGetter, visionModelFinder) - log.Printf("✅ [VISION-INIT] Vision service initialized") - }) -} diff --git a/backend/internal/services/workflow_generator.go b/backend/internal/services/workflow_generator.go deleted file mode 100644 index 314fbc80..00000000 --- a/backend/internal/services/workflow_generator.go +++ /dev/null @@ -1,1589 +0,0 @@ -package services - -import ( - "bytes" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "regexp" - "strings" - "time" - - "github.com/google/uuid" - - "claraverse/internal/database" - "claraverse/internal/models" -) - -// WorkflowGeneratorService handles workflow generation with structured output -type WorkflowGeneratorService struct { - db *database.DB - providerService *ProviderService - chatService *ChatService -} - -// NewWorkflowGeneratorService creates a new workflow generator service -func NewWorkflowGeneratorService( - db *database.DB, - providerService *ProviderService, - chatService *ChatService, -) *WorkflowGeneratorService { - return &WorkflowGeneratorService{ - db: db, - providerService: providerService, - chatService: chatService, - } -} - -// V1ToolCategory is used for the legacy v1 dynamic tool injection -type V1ToolCategory struct { - Name string - Keywords []string - Tools string - Description string -} - -// v1ToolCategories defines tool categories for legacy v1 workflow generation -var v1ToolCategories = []V1ToolCategory{ - {Name: "data_analysis", Keywords: []string{"analyze", "analysis", "data", "csv", "excel", "spreadsheet", "chart", "graph", "statistics", "visualize", "visualization", "metrics", "calculate", "math"}, Description: "Data analysis and visualization", Tools: "📊 DATA & ANALYSIS:\n- analyze_data: Python data analysis with charts\n- calculate_math: Mathematical calculations\n- read_spreadsheet: Read Excel/CSV files\n- read_data_file: Read and parse data files\n- read_document: Extract text from documents"}, - {Name: "search_web", Keywords: []string{"search", "find", "lookup", "google", "web", "internet", "news", "articles", "scrape", "crawl", "download", "url", "website"}, Description: "Web search and scraping", Tools: "🔍 SEARCH & WEB:\n- search_web: Search the internet\n- search_images: Search for images\n- scrape_web: Scrape content from URL\n- download_file: Download a file from URL"}, - {Name: "content_creation", Keywords: []string{"create", "generate", "write", "document", "pdf", "docx", "presentation", "pptx", "powerpoint", "image", "picture", "photo", "text file", "html"}, Description: "Content creation", Tools: "📝 CONTENT CREATION:\n- create_document: Create DOCX or PDF\n- create_text_file: Create text files\n- create_presentation: Create PowerPoint\n- generate_image: Generate AI images\n- edit_image: Edit images\n- html_to_pdf: Convert HTML to PDF"}, - {Name: "media_processing", Keywords: []string{"audio", "transcribe", "speech", "voice", "mp3", "wav", "video", "image", "describe", "vision", "see", "look"}, Description: "Media processing", Tools: "🎤 MEDIA PROCESSING:\n- transcribe_audio: Transcribe audio\n- describe_image: Analyze images"}, - {Name: "utilities", Keywords: []string{"time", "date", "now", "today", "current", "python", "code", "script", "api", "http", "request", "webhook", "endpoint"}, Description: "Utilities", Tools: "⏰ UTILITIES:\n- get_current_time: Get current time\n- run_python: Execute Python code\n- api_request: Make HTTP requests\n- send_webhook: Send webhook"}, - {Name: "messaging", Keywords: []string{"discord", "slack", "telegram", "teams", "google chat", "email", "sms", "whatsapp", "message", "send", "notify", "notification", "alert", "chat"}, Description: "Messaging", Tools: "💬 MESSAGING:\n- send_discord_message: Discord\n- send_slack_message: Slack\n- send_telegram_message: Telegram\n- send_google_chat_message: Google Chat\n- send_teams_message: Teams\n- send_email: Email\n- twilio_send_sms: SMS\n- twilio_send_whatsapp: WhatsApp"}, - {Name: "video_conferencing", Keywords: []string{"zoom", "meeting", "webinar", "calendly", "calendar", "schedule", "event", "conference", "call", "register", "attendee"}, Description: "Video conferencing", Tools: "📹 VIDEO CONFERENCING:\n- zoom_meeting: Zoom meetings/webinars (actions: create, list, get, register, create_webinar, register_webinar)\n- calendly_events: Calendly events"}, - {Name: "project_management", Keywords: []string{"jira", "linear", "clickup", "trello", "asana", "task", "issue", "ticket", "project", "board", "kanban", "sprint", "backlog"}, Description: "Project management", Tools: "📋 PROJECT MANAGEMENT:\n- jira_issues/jira_create_issue/jira_update_issue\n- linear_issues/linear_create_issue/linear_update_issue\n- clickup_tasks/clickup_create_task/clickup_update_task\n- trello_boards/trello_lists/trello_cards/trello_create_card\n- asana_tasks"}, - {Name: "crm_sales", Keywords: []string{"hubspot", "leadsquared", "mailchimp", "crm", "lead", "contact", "deal", "sales", "customer", "subscriber", "marketing", "audience"}, Description: "CRM & Sales", Tools: "💼 CRM & SALES:\n- hubspot_contacts/hubspot_deals/hubspot_companies\n- leadsquared_leads/leadsquared_create_lead\n- mailchimp_lists/mailchimp_add_subscriber"}, - {Name: "analytics", Keywords: []string{"posthog", "mixpanel", "analytics", "track", "event", "identify", "user profile", "funnel", "cohort", "retention"}, Description: "Analytics", Tools: "📊 ANALYTICS:\n- posthog_capture/posthog_identify/posthog_query\n- mixpanel_track/mixpanel_user_profile"}, - {Name: "code_devops", Keywords: []string{"github", "gitlab", "netlify", "git", "repo", "repository", "issue", "pull request", "pr", "merge", "deploy", "build", "ci", "cd", "code"}, Description: "Code & DevOps", Tools: "🐙 CODE & DEVOPS:\n- github_create_issue/github_list_issues/github_get_repo/github_add_comment\n- gitlab_projects/gitlab_issues/gitlab_mrs\n- netlify_sites/netlify_deploys/netlify_trigger_build"}, - {Name: "productivity", Keywords: []string{"notion", "airtable", "database", "page", "note", "record", "table", "workspace", "wiki"}, Description: "Productivity", Tools: "📓 PRODUCTIVITY:\n- notion_search/notion_query_database/notion_create_page/notion_update_page\n- airtable_list/airtable_read/airtable_create/airtable_update"}, - {Name: "ecommerce", Keywords: []string{"shopify", "shop", "product", "order", "customer", "ecommerce", "store", "inventory", "cart"}, Description: "E-Commerce", Tools: "🛒 E-COMMERCE:\n- shopify_products/shopify_orders/shopify_customers"}, - {Name: "social_media", Keywords: []string{"twitter", "x", "tweet", "post", "social", "media", "follow", "user", "timeline"}, Description: "Social Media", Tools: "🐦 SOCIAL MEDIA:\n- x_search_posts/x_post_tweet/x_get_user/x_get_user_posts"}, -} - -// detectToolCategoriesV1 analyzes user message and returns relevant tool categories (legacy v1) -func detectToolCategoriesV1(userMessage string) []string { - msg := strings.ToLower(userMessage) - detected := make(map[string]bool) - - for _, category := range v1ToolCategories { - for _, keyword := range category.Keywords { - if strings.Contains(msg, keyword) { - detected[category.Name] = true - break - } - } - } - - // Always include utilities for time-sensitive keywords - timeSensitiveKeywords := []string{"today", "daily", "recent", "latest", "current", "now", "this week", "this month", "news", "trending", "breaking"} - for _, keyword := range timeSensitiveKeywords { - if strings.Contains(msg, keyword) { - detected["utilities"] = true - break - } - } - - // If no categories detected, return a default set - if len(detected) == 0 { - detected["data_analysis"] = true - detected["search_web"] = true - detected["utilities"] = true - detected["content_creation"] = true - } - - result := make([]string, 0, len(detected)) - for cat := range detected { - result = append(result, cat) - } - return result -} - -// buildDynamicToolsSectionV1 builds the tools section based on detected categories (legacy v1) -func buildDynamicToolsSectionV1(categories []string) string { - var builder strings.Builder - builder.WriteString("=== AVAILABLE TOOLS (Relevant to your request) ===\n\n") - - categoryMap := make(map[string]V1ToolCategory) - for _, cat := range v1ToolCategories { - categoryMap[cat.Name] = cat - } - - for _, catName := range categories { - if cat, ok := categoryMap[catName]; ok { - builder.WriteString(cat.Tools) - builder.WriteString("\n\n") - } - } - - return builder.String() -} - -// buildDynamicSystemPrompt builds the complete system prompt with dynamically injected tools (legacy v1) -func buildDynamicSystemPrompt(userMessage string) string { - categories := detectToolCategoriesV1(userMessage) - toolsSection := buildDynamicToolsSectionV1(categories) - prompt := strings.Replace(WorkflowSystemPromptBase, "{{DYNAMIC_TOOLS_SECTION}}", toolsSection, 1) - log.Printf("🔧 [WORKFLOW-GEN] Detected tool categories: %v", categories) - return prompt -} - -// WorkflowSystemPromptBase is the base system prompt without tools section -const WorkflowSystemPromptBase = `You are a Clara AI workflow generator. Your ONLY job is to output valid JSON workflow definitions. - -CRITICAL: You must ONLY respond with a JSON object. No explanations, no markdown, no code blocks - JUST the JSON. - -=== WORKFLOW STRUCTURE === -{ - "blocks": [...], - "connections": [...], - "variables": [], - "explanation": "Brief description of what the workflow does or what changed" -} - -=== BLOCK TYPES === -1. "variable" - Input block (Start) - Config: { "operation": "read", "variableName": "input", "defaultValue": "" } - Optional "inputType": "text" (default), "file", or "json" - - Use "json" when workflow needs structured input (API-like endpoints, complex data) - - For JSON input, add "jsonSchema" to define expected structure - Example JSON input block: - { "operation": "read", "variableName": "input", "inputType": "json", - "jsonSchema": { "type": "object", "properties": { "userId": {"type": "string"}, "action": {"type": "string"} } } } - -2. "llm_inference" - AI agent with tools (EXECUTION MODE) - Config: { - "systemPrompt": "IMPERATIVE instructions - what the agent MUST do", - "userPrompt": "{{input}}" or "{{previous-block.response}}", - "temperature": 0.3, - "enabledTools": ["tool_name"], - "requiredTools": ["tool_name"], - "requireToolUsage": true, - "outputFormat": "json", - "outputSchema": { JSON Schema object } - } - -3. "code_block" - Direct tool execution (NO LLM, FAST & DETERMINISTIC) - Config: { - "toolName": "tool_name_here", - "argumentMapping": { "param1": "{{input}}", "param2": "{{block-id.response}}" } - } - - USE code_block WHEN: - - Task is PURELY mechanical (no reasoning/decisions needed) - - All data and parameters are already available - - Examples: get current time, send pre-formatted message, make API call with known params - - USE llm_inference INSTEAD WHEN: - - Need to DECIDE what to search/do - - Need to INTERPRET, FORMAT, or SUMMARIZE data - - ANY intelligent decision-making is required - -=== BLOCK TYPE DECISION GUIDE === -Q: Does the task need ANY reasoning, decisions, or interpretation? - YES → Use "llm_inference" (LLM calls tools with intelligence) - NO → Use "code_block" (direct execution, faster & cheaper) - -EXAMPLES - When to use code_block: -- "Get current time" → code_block with toolName="get_current_time" -- "Send this exact message to Discord" (message already formatted) → code_block -- "Download file from URL" (URL provided) → code_block -- "Calculate 2+2" → code_block with toolName="calculate_math" - -EXAMPLES - When to use llm_inference: -- "Search for news about X" → llm_inference (LLM decides search query) -- "Analyze this data" → llm_inference (LLM interprets results) -- "Format and send to Discord" → llm_inference (needs formatting decision) -- "Summarize the results" → llm_inference (needs interpretation) - -=== SYSTEM PROMPT WRITING RULES (CRITICAL!) === -System prompts MUST be written in IMPERATIVE/COMMAND style, not conversational: - -CORRECT (Imperative - use these patterns): -- "Search for news about the topic. Call search_web. Return top 3 results with titles and summaries." -- "Send this content to Discord NOW using send_discord_message. Include embed_title." -- "Analyze the data. Generate a bar chart. Use analyze_data tool immediately." - -WRONG (Conversational - NEVER use): -- "You should search for news..." (too passive) -- "Please format and send to Discord..." (too polite/optional) -- "Can you analyze this data..." (implies optionality) -- "If you want, you could..." (gives choice - NO!) - -WRONG (Question-asking - NEVER generate prompts that ask questions): -- "What topic would you like to search?" (NO - data is provided) -- "Should I include charts?" (NO - decide based on context) -- "Would you like me to..." (NO - just do it) - -{{DYNAMIC_TOOLS_SECTION}} - -=== TOOL CONFIGURATION (CRITICAL FOR RELIABILITY!) === -For each LLM block with tools, you MUST include: -- "enabledTools": List of tools the block CAN use -- "requiredTools": List of tools the block MUST use (usually same as enabledTools) -- "requireToolUsage": true (forces tool usage, prevents text-only responses) -- "temperature": 0.3 (low for deterministic execution) - -Example for Discord Publisher block: -{ - "enabledTools": ["send_discord_message"], - "requiredTools": ["send_discord_message"], - "requireToolUsage": true, - "temperature": 0.3 -} - -=== STRUCTURED OUTPUT (CRITICAL FOR RELIABILITY!) === -ALWAYS use structured outputs for blocks that return data to be consumed by other blocks or rendered in UIs. -This ensures 100% predictable, parseable outputs. - -When to use structured output: -- Data fetching blocks (news, search results, API data) -- Analysis blocks that return metrics or insights -- Any block whose output will be displayed in a UI -- Blocks that extract specific information - -How to configure structured output: -1. Set "outputFormat": "json" -2. Define "outputSchema" with JSON Schema -3. The schema MUST match what downstream blocks or UI expect - -Example for News Fetcher block: -{ - "systemPrompt": "FIRST call get_current_time. THEN call search_web for news. Return EXACTLY in the schema format.", - "temperature": 0.3, - "enabledTools": ["get_current_time", "search_web"], - "requiredTools": ["get_current_time", "search_web"], - "requireToolUsage": true, - "outputFormat": "json", - "outputSchema": { - "type": "object", - "properties": { - "articles": { - "type": "array", - "items": { - "type": "object", - "properties": { - "title": {"type": "string"}, - "source": {"type": "string"}, - "url": {"type": "string"}, - "summary": {"type": "string"}, - "publishedDate": {"type": "string"} - }, - "required": ["title", "source", "url", "summary", "publishedDate"] - } - }, - "totalResults": {"type": "number"}, - "fetchedAt": {"type": "string"} - }, - "required": ["articles", "totalResults", "fetchedAt"], - "additionalProperties": false - } -} - -Common schema patterns: -- News/Articles WITH metadata: { articles: [{ title, source, url, summary, publishedDate }], totalResults, fetchedAt } -- Simple list (array at root): [{ id, name, value }] - use "type": "array" with "items" schema -- Metrics/Stats: { metrics: { key: value }, summary, analyzedAt } -- List Results: { items: [{ name, description, value }], count, retrievedAt } -- Analysis: { insights: [...], recommendations: [...], confidence: number } - -Example for Simple Product List (array at root): -{ - "systemPrompt": "Call search_products and return the list of products.", - "outputFormat": "json", - "outputSchema": { - "type": "array", - "items": { - "type": "object", - "properties": { - "id": {"type": "string"}, - "name": {"type": "string"}, - "price": {"type": "number"} - }, - "required": ["id", "name", "price"] - } - } -} - -RULES for structured output: -1. Use "additionalProperties": false to prevent extra fields -2. CRITICAL: In "required" arrays, you MUST list ALL properties defined in "properties" - OpenAI's strict mode rejects partial required arrays -3. Use descriptive property names (camelCase) -4. Include metadata (fetchedAt, analyzedAt, etc.) -5. Schema MUST be strict - no optional variations -6. Every nested object needs its own "required" array listing ALL its properties -7. ARRAYS AT ROOT LEVEL: You can use arrays directly without wrapping in an object: - - For simple lists: { "type": "array", "items": { "type": "object", "properties": {...}, "required": [...] } } - - For data + metadata: { "type": "object", "properties": { "items": {...}, "total": {...} }, "required": [...] } - - Use arrays when returning just a list, use objects when you need metadata too - -=== CREDENTIAL HANDLING === -For integration tools (Discord, Slack, webhooks): -- Credentials are AUTO-INJECTED at runtime -- DO NOT include webhook URLs in prompts -- DO NOT tell the agent to ask for credentials -- System prompts should command: "Send to Discord NOW" (not "provide your webhook URL") - -=== TIME-SENSITIVE QUERIES (CRITICAL!) === -When the user's request involves time-sensitive information, the search block MUST also call get_current_time: - -TIME-SENSITIVE KEYWORDS (if any of these appear, add get_current_time): -- "today", "daily", "recent", "latest", "current", "now", "this week", "this month" -- "news", "events", "updates", "trending", "breaking" -- "stock", "price", "weather", "score", "live" - -For time-sensitive search blocks, use BOTH tools: -{ - "enabledTools": ["get_current_time", "search_web"], - "requiredTools": ["get_current_time", "search_web"], - "systemPrompt": "FIRST call get_current_time to get today's date. THEN search for [topic] using that date. Include the date in your search query for accurate results." -} - -EXAMPLE - User asks "Get me today's AI news": -{ - "systemPrompt": "FIRST call get_current_time to get today's date and time. THEN call search_web with the topic AND the current date (e.g., 'AI news December 2024'). Return top 3 results with titles, sources, and the date they were published.", - "enabledTools": ["get_current_time", "search_web"], - "requiredTools": ["get_current_time", "search_web"] -} - -=== TOOL ASSIGNMENT RULES === -Each block = ONE specific task = ONE set of related tools. Never mix unrelated tools! - -TOOL SELECTION BY BLOCK PURPOSE: -- Research/Search block (time-sensitive): enabledTools=["get_current_time", "search_web"], requiredTools=["get_current_time", "search_web"] -- Research/Search block (general): enabledTools=["search_web"], requiredTools=["search_web"] -- Data Analysis block: enabledTools=["analyze_data"], requiredTools=["analyze_data"] -- Spreadsheet Reading block: enabledTools=["read_spreadsheet"], requiredTools=["read_spreadsheet"] -- Audio Transcription block: enabledTools=["transcribe_audio"], requiredTools=["transcribe_audio"] -- Image Analysis block: enabledTools=["describe_image"], requiredTools=["describe_image"] -- Document Reading block: enabledTools=["read_document"], requiredTools=["read_document"] -- Discord Publisher: enabledTools=["send_discord_message"], requiredTools=["send_discord_message"] -- Slack Publisher: enabledTools=["send_slack_message"], requiredTools=["send_slack_message"] -- Telegram Publisher: enabledTools=["send_telegram_message"], requiredTools=["send_telegram_message"] -- Google Chat Publisher: enabledTools=["send_google_chat_message"], requiredTools=["send_google_chat_message"] -- Content Writer: enabledTools=[] (no tools - generates text only, requireToolUsage=false) - -=== DATA FLOW & VARIABLE PATHS (CRITICAL!) === - -UNDERSTANDING VARIABLE BLOCKS: -The Start block (type: "variable") has two important fields: - - "id": "start" (the block's ID) - - "variableName": "input" (creates a global variable) - -The variableName field creates a TOP-LEVEL key in the workflow context! - -Example Start block output structure: -{ - "value": {"email": "test@example.com", "name": "John"}, - "input": {"email": "test@example.com", "name": "John"} ← variableName creates this key -} - -VARIABLE PATH RULES: -1. To access the ENTIRE input: {{input}} -2. To access nested fields: {{input.email}}, {{input.name}}, {{input.phone}} -3. Previous block outputs: {{block-id.response}} -4. Block outputs are ALREADY RESOLVED - no need to fetch data - -CORRECT PATHS: -- {{input}} - Entire workflow input (from start block's variableName) -- {{input.email}} - Nested field from input -- {{news-researcher.response}} - Previous block's response -- {{block-id.response.articles}} - Nested data from previous block - -WRONG PATHS (NEVER use these): -- {{start.email}} - NO! "email" is INSIDE input, not a property of start -- {{start.response.email}} - NO! Start block doesn't have "response" -- {{start.output.value}} - NO! Use {{input}} instead -- {{block.output.value}} - NO! Use {{block.response}} instead - -FOR CODE_BLOCKS with nested data: -When accessing nested fields from previous blocks in argumentMapping: -{ - "toolName": "mongodb_write", - "argumentMapping": { - "action": "insertOne", - "collection": "users", - "document": { - "email": "{{input.email}}", ← Access nested field - "name": "{{input.name}}", ← Access nested field - "phone": "{{input.phone}}", ← Access nested field - "created_at": "{{get-current-time.response}}" - } - } -} - -=== BLOCK ID NAMING === -Block "id" MUST be kebab-case of "name": -- "News Researcher" → id: "news-researcher" -- "Discord Publisher" → id: "discord-publisher" - -=== LAYOUT === -- Start block: position { "x": 250, "y": 50 } -- Space blocks 150px vertically -- timeout: 30 for variable, 120 for LLM blocks - -=== EXAMPLE 1: News Search + Discord (TIME-SENSITIVE) === -User: "Create an agent that searches for news and posts to Discord" - -{ - "blocks": [ - { - "id": "start", - "type": "variable", - "name": "Start", - "description": "Input topic for news search", - "config": { "operation": "read", "variableName": "input", "defaultValue": "AI news" }, - "position": { "x": 250, "y": 50 }, - "timeout": 30 - }, - { - "id": "news-researcher", - "type": "llm_inference", - "name": "News Researcher", - "description": "Search and summarize latest news", - "config": { - "systemPrompt": "FIRST call get_current_time to get today's date. THEN call search_web for news about the given topic, including the current date in your query (e.g., 'AI news December 2024'). Return results EXACTLY in the output schema format with top 3 articles.", - "userPrompt": "{{input}}", - "temperature": 0.3, - "enabledTools": ["get_current_time", "search_web"], - "requiredTools": ["get_current_time", "search_web"], - "requireToolUsage": true, - "outputFormat": "json", - "outputSchema": { - "type": "object", - "properties": { - "articles": { - "type": "array", - "items": { - "type": "object", - "properties": { - "title": {"type": "string"}, - "source": {"type": "string"}, - "url": {"type": "string"}, - "summary": {"type": "string"}, - "publishedDate": {"type": "string"} - }, - "required": ["title", "source", "url", "summary"] - } - }, - "totalResults": {"type": "number"}, - "fetchedAt": {"type": "string"} - }, - "required": ["articles", "totalResults", "fetchedAt"], - "additionalProperties": false - } - }, - "position": { "x": 250, "y": 200 }, - "timeout": 120 - }, - { - "id": "discord-publisher", - "type": "llm_inference", - "name": "Discord Publisher", - "description": "Format and send news to Discord", - "config": { - "systemPrompt": "Send this news summary to Discord NOW. Call send_discord_message with: content containing a brief intro, embed_title set to 'Latest News Update', embed_description with the full summary. Execute immediately.", - "userPrompt": "{{news-researcher.response}}", - "temperature": 0.3, - "enabledTools": ["send_discord_message"], - "requiredTools": ["send_discord_message"], - "requireToolUsage": true - }, - "position": { "x": 250, "y": 350 }, - "timeout": 120 - } - ], - "connections": [ - { "id": "conn-1", "sourceBlockId": "start", "sourceOutput": "output", "targetBlockId": "news-researcher", "targetInput": "input" }, - { "id": "conn-2", "sourceBlockId": "news-researcher", "sourceOutput": "output", "targetBlockId": "discord-publisher", "targetInput": "input" } - ], - "variables": [], - "explanation": "3 blocks: Start→News Researcher (MUST call get_current_time THEN search_web)→Discord Publisher (MUST call send_discord_message)" -} - -=== EXAMPLE 2: Data Analysis + Discord === -User: "Analyze CSV data and send chart to Discord" - -{ - "blocks": [ - { - "id": "start", - "type": "variable", - "name": "Start", - "description": "Receive CSV file input", - "config": { "operation": "read", "variableName": "input", "inputType": "file" }, - "position": { "x": 250, "y": 50 }, - "timeout": 30 - }, - { - "id": "data-analyzer", - "type": "llm_inference", - "name": "Data Analyzer", - "description": "Analyze data and generate charts", - "config": { - "systemPrompt": "Analyze this data immediately. Call analyze_data to generate visualizations. Create at least one meaningful chart showing key insights. Return analysis summary.", - "userPrompt": "{{input}}", - "temperature": 0.3, - "enabledTools": ["analyze_data"], - "requiredTools": ["analyze_data"], - "requireToolUsage": true - }, - "position": { "x": 250, "y": 200 }, - "timeout": 120 - }, - { - "id": "discord-publisher", - "type": "llm_inference", - "name": "Discord Publisher", - "description": "Send analysis results and charts to Discord", - "config": { - "systemPrompt": "Send this analysis to Discord NOW. Call send_discord_message with the summary as content and include the chart image. Execute immediately - do not ask questions.", - "userPrompt": "{{data-analyzer.response}}", - "temperature": 0.3, - "enabledTools": ["send_discord_message"], - "requiredTools": ["send_discord_message"], - "requireToolUsage": true - }, - "position": { "x": 250, "y": 350 }, - "timeout": 120 - } - ], - "connections": [ - { "id": "conn-1", "sourceBlockId": "start", "sourceOutput": "output", "targetBlockId": "data-analyzer", "targetInput": "input" }, - { "id": "conn-2", "sourceBlockId": "data-analyzer", "sourceOutput": "output", "targetBlockId": "discord-publisher", "targetInput": "input" } - ], - "variables": [], - "explanation": "3 blocks: Start (file)→Data Analyzer (MUST call analyze_data)→Discord Publisher (MUST call send_discord_message)" -} - -=== EXAMPLE 3: Audio Transcription + Summary === -User: "Create an agent that transcribes audio files and summarizes them" - -{ - "blocks": [ - { - "id": "start", - "type": "variable", - "name": "Start", - "description": "Receive audio file input", - "config": { "operation": "read", "variableName": "input", "inputType": "file", "acceptedFileTypes": ["audio"] }, - "position": { "x": 250, "y": 50 }, - "timeout": 30 - }, - { - "id": "audio-transcriber", - "type": "llm_inference", - "name": "Audio Transcriber", - "description": "Transcribe audio to text", - "config": { - "systemPrompt": "Transcribe the provided audio file immediately. Call transcribe_audio with the file_id from the input. Return the full transcription text.", - "userPrompt": "{{input}}", - "temperature": 0.3, - "enabledTools": ["transcribe_audio"], - "requiredTools": ["transcribe_audio"], - "requireToolUsage": true - }, - "position": { "x": 250, "y": 200 }, - "timeout": 120 - }, - { - "id": "content-summarizer", - "type": "llm_inference", - "name": "Content Summarizer", - "description": "Summarize the transcribed content", - "config": { - "systemPrompt": "Summarize this transcription. Extract key points, main topics discussed, and important details. Provide a concise summary with bullet points.", - "userPrompt": "{{audio-transcriber.response}}", - "temperature": 0.5, - "enabledTools": [], - "requireToolUsage": false - }, - "position": { "x": 250, "y": 350 }, - "timeout": 120 - } - ], - "connections": [ - { "id": "conn-1", "sourceBlockId": "start", "sourceOutput": "output", "targetBlockId": "audio-transcriber", "targetInput": "input" }, - { "id": "conn-2", "sourceBlockId": "audio-transcriber", "sourceOutput": "output", "targetBlockId": "content-summarizer", "targetInput": "input" } - ], - "variables": [], - "explanation": "3 blocks: Start (audio file)→Audio Transcriber (MUST call transcribe_audio)→Content Summarizer (text generation)" -} - -=== EXAMPLE 4: MongoDB Insert with Nested Field Access === -User: "Insert user data into MongoDB" - -This example shows correct variable path usage for accessing nested input fields: - -{ - "blocks": [ - { - "id": "start", - "type": "variable", - "name": "Start", - "description": "Receive user input as JSON", - "config": { - "operation": "read", - "variableName": "input", - "inputType": "json", - "jsonSchema": { - "type": "object", - "properties": { - "email": {"type": "string"}, - "name": {"type": "string"}, - "phone": {"type": "string"} - } - } - }, - "position": { "x": 250, "y": 50 }, - "timeout": 30 - }, - { - "id": "get-current-time", - "type": "code_block", - "name": "Get Current Time", - "description": "Get timestamp for record", - "config": { - "toolName": "get_current_time", - "argumentMapping": {} - }, - "position": { "x": 250, "y": 200 }, - "timeout": 30 - }, - { - "id": "insert-user", - "type": "code_block", - "name": "Insert User", - "description": "Insert user into MongoDB with nested field access", - "config": { - "toolName": "mongodb_write", - "argumentMapping": { - "action": "insertOne", - "collection": "users", - "document": { - "email": "{{input.email}}", - "name": "{{input.name}}", - "phone": "{{input.phone}}", - "created_at": "{{get-current-time.response}}", - "status": "active" - } - } - }, - "position": { "x": 250, "y": 350 }, - "timeout": 30 - } - ], - "connections": [ - { "id": "conn-1", "sourceBlockId": "start", "sourceOutput": "output", "targetBlockId": "get-current-time", "targetInput": "input" }, - { "id": "conn-2", "sourceBlockId": "get-current-time", "sourceOutput": "output", "targetBlockId": "insert-user", "targetInput": "input" } - ], - "variables": [], - "explanation": "Inserts user data into MongoDB using correct nested field paths: {{input.email}}, {{input.name}}, {{input.phone}}" -} - -KEY POINTS IN THIS EXAMPLE: -- Start block has variableName: "input", so we use {{input.email}}, NOT {{start.email}} -- Code_block argumentMapping can have nested objects -- Deep interpolation resolves {{...}} at any nesting level -- Mixed literal values ("active") and variable references work together - -=== EXAMPLE 5: Mixed LLM + code_block (EFFICIENT HYBRID) === -User: "Search for AI news and send to Discord" - -This example shows how to mix llm_inference (for research) with code_block (for sending). -The llm_inference block does the intelligent work (search + format), then code_block sends directly. - -{ - "blocks": [ - { - "id": "start", - "type": "variable", - "name": "Start", - "description": "Input topic for news search", - "config": { "operation": "read", "variableName": "input", "defaultValue": "AI news" }, - "position": { "x": 250, "y": 50 }, - "timeout": 30 - }, - { - "id": "news-researcher", - "type": "llm_inference", - "name": "News Researcher", - "description": "Search news and format for Discord", - "config": { - "systemPrompt": "FIRST call get_current_time. THEN call search_web for news. Format results as a Discord message with embed fields. Return EXACTLY in the output schema format.", - "userPrompt": "{{input}}", - "temperature": 0.3, - "enabledTools": ["get_current_time", "search_web"], - "requiredTools": ["get_current_time", "search_web"], - "requireToolUsage": true, - "outputFormat": "json", - "outputSchema": { - "type": "object", - "properties": { - "content": {"type": "string", "description": "Brief intro message"}, - "embed_title": {"type": "string", "description": "Discord embed title"}, - "embed_description": {"type": "string", "description": "Full news summary with links"} - }, - "required": ["content", "embed_title", "embed_description"], - "additionalProperties": false - } - }, - "position": { "x": 250, "y": 200 }, - "timeout": 120 - }, - { - "id": "discord-sender", - "type": "code_block", - "name": "Discord Sender", - "description": "Send pre-formatted message to Discord (no LLM needed)", - "config": { - "toolName": "send_discord_message", - "argumentMapping": { - "content": "{{news-researcher.response.content}}", - "embed_title": "{{news-researcher.response.embed_title}}", - "embed_description": "{{news-researcher.response.embed_description}}" - } - }, - "position": { "x": 250, "y": 350 }, - "timeout": 30 - } - ], - "connections": [ - { "id": "conn-1", "sourceBlockId": "start", "sourceOutput": "output", "targetBlockId": "news-researcher", "targetInput": "input" }, - { "id": "conn-2", "sourceBlockId": "news-researcher", "sourceOutput": "output", "targetBlockId": "discord-sender", "targetInput": "input" } - ], - "variables": [], - "explanation": "3 blocks: Start→News Researcher (llm_inference: search + format)→Discord Sender (code_block: direct send, no LLM overhead)" -} - -WHY use code_block for Discord Sender? -- The message is ALREADY formatted by the researcher -- No decisions needed - just send the exact content -- FASTER execution (no LLM API call) -- CHEAPER (no token costs) -- MORE RELIABLE (no LLM interpretation) - -REMEMBER: -- Temperature = 0.3 for all LLM blocks (deterministic) -- requiredTools = same as enabledTools (forces tool usage) -- requireToolUsage = true (validates tool was called) -- System prompts use IMPERATIVE style (commands, not suggestions) -- Use code_block for mechanical tasks with NO reasoning needed (faster, cheaper) -- Use llm_inference when decisions, formatting, or interpretation is required -- code_block timeout = 30 (no LLM), llm_inference timeout = 120 -- Output ONLY valid JSON. No text before or after.` - -// GenerateWorkflow generates a workflow based on user input -func (s *WorkflowGeneratorService) GenerateWorkflow(req *models.WorkflowGenerateRequest, userID string) (*models.WorkflowGenerateResponse, error) { - log.Printf("🔧 [WORKFLOW-GEN] Generating workflow for agent %s, user %s", req.AgentID, userID) - - // Get provider and model - provider, modelID, err := s.getProviderAndModel(req.ModelID) - if err != nil { - return &models.WorkflowGenerateResponse{ - Success: false, - Error: fmt.Sprintf("Failed to get provider: %v", err), - }, nil - } - - // Build the user message - userMessage := s.buildUserMessage(req) - - // Build dynamic system prompt with relevant tools based on user request - systemPrompt := buildDynamicSystemPrompt(req.UserMessage) - - // Build messages array with conversation history for better context - messages := []map[string]interface{}{ - { - "role": "system", - "content": systemPrompt, - }, - } - - // Add conversation history if provided (for multi-turn context) - if len(req.ConversationHistory) > 0 { - for _, msg := range req.ConversationHistory { - messages = append(messages, map[string]interface{}{ - "role": msg.Role, - "content": msg.Content, - }) - } - } - - // Add current user message - messages = append(messages, map[string]interface{}{ - "role": "user", - "content": userMessage, - }) - - // Build request body with structured output - requestBody := map[string]interface{}{ - "model": modelID, - "messages": messages, - "stream": false, - "temperature": 0.3, // Lower temperature for more consistent JSON output - } - - // Add response_format for structured output (OpenAI-compatible) - requestBody["response_format"] = map[string]interface{}{ - "type": "json_object", - } - - reqBody, err := json.Marshal(requestBody) - if err != nil { - return nil, fmt.Errorf("failed to marshal request: %w", err) - } - - log.Printf("📤 [WORKFLOW-GEN] Sending request to %s with model %s", provider.BaseURL, modelID) - - // Create HTTP request - httpReq, err := http.NewRequest("POST", provider.BaseURL+"/chat/completions", bytes.NewBuffer(reqBody)) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - httpReq.Header.Set("Content-Type", "application/json") - httpReq.Header.Set("Authorization", "Bearer "+provider.APIKey) - - // Send request with timeout - client := &http.Client{Timeout: 120 * time.Second} - resp, err := client.Do(httpReq) - if err != nil { - return nil, fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - // Read response - body, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - if resp.StatusCode != http.StatusOK { - log.Printf("⚠️ [WORKFLOW-GEN] API error: %s", string(body)) - return &models.WorkflowGenerateResponse{ - Success: false, - Error: fmt.Sprintf("API error (status %d): %s", resp.StatusCode, string(body)), - }, nil - } - - // Parse API response - var apiResponse struct { - Choices []struct { - Message struct { - Content string `json:"content"` - } `json:"message"` - } `json:"choices"` - } - - if err := json.Unmarshal(body, &apiResponse); err != nil { - return nil, fmt.Errorf("failed to parse API response: %w", err) - } - - if len(apiResponse.Choices) == 0 { - return &models.WorkflowGenerateResponse{ - Success: false, - Error: "No response from model", - }, nil - } - - content := apiResponse.Choices[0].Message.Content - log.Printf("📥 [WORKFLOW-GEN] Received response (%d chars)", len(content)) - - // Parse the workflow JSON from the response - return s.parseWorkflowResponse(content, req.CurrentWorkflow != nil, req.AgentID) -} - -// buildUserMessage constructs the user message for workflow generation -func (s *WorkflowGeneratorService) buildUserMessage(req *models.WorkflowGenerateRequest) string { - if req.CurrentWorkflow != nil && len(req.CurrentWorkflow.Blocks) > 0 { - // Modification request - include current workflow - workflowJSON, _ := json.MarshalIndent(req.CurrentWorkflow, "", " ") - return fmt.Sprintf(`MODIFICATION REQUEST - -Current workflow: -%s - -User request: %s - -Output the complete modified workflow JSON with all blocks (not just changes).`, string(workflowJSON), req.UserMessage) - } - - // New workflow request - return fmt.Sprintf("CREATE NEW WORKFLOW\n\nUser request: %s", req.UserMessage) -} - -// parseWorkflowResponse parses the LLM response into a workflow -func (s *WorkflowGeneratorService) parseWorkflowResponse(content string, isModification bool, agentID string) (*models.WorkflowGenerateResponse, error) { - // Try to extract JSON from the response (handle markdown code blocks) - jsonContent := s.extractJSON(content) - - // Parse the workflow - var workflowData struct { - Blocks []models.Block `json:"blocks"` - Connections []models.Connection `json:"connections"` - Variables []models.Variable `json:"variables"` - Explanation string `json:"explanation"` - } - - if err := json.Unmarshal([]byte(jsonContent), &workflowData); err != nil { - log.Printf("⚠️ [WORKFLOW-GEN] Failed to parse workflow JSON: %v", err) - log.Printf("⚠️ [WORKFLOW-GEN] Content: %s", jsonContent[:min(500, len(jsonContent))]) - return &models.WorkflowGenerateResponse{ - Success: false, - Error: fmt.Sprintf("Failed to parse workflow JSON: %v", err), - Explanation: content, // Return raw content as explanation - }, nil - } - - // Log the generated workflow for debugging - prettyWorkflow, _ := json.MarshalIndent(workflowData, "", " ") - log.Printf("📋 [WORKFLOW-GEN] Generated workflow:\n%s", string(prettyWorkflow)) - - // Post-process blocks: set normalizedId to match id - for i := range workflowData.Blocks { - if workflowData.Blocks[i].NormalizedID == "" { - workflowData.Blocks[i].NormalizedID = workflowData.Blocks[i].ID - } - } - - // Validate the workflow - errors := s.validateWorkflow(&workflowData) - if len(errors) > 0 { - log.Printf("⚠️ [WORKFLOW-GEN] Workflow validation errors: %v", errors) - } - - // Determine action - action := "create" - if isModification { - action = "modify" - } - - // Build the workflow with generated IDs - workflow := &models.Workflow{ - ID: uuid.New().String(), - AgentID: agentID, - Blocks: workflowData.Blocks, - Connections: workflowData.Connections, - Variables: workflowData.Variables, - Version: 1, - } - - log.Printf("✅ [WORKFLOW-GEN] Successfully parsed workflow: %d blocks, %d connections", - len(workflow.Blocks), len(workflow.Connections)) - - return &models.WorkflowGenerateResponse{ - Success: true, - Workflow: workflow, - Explanation: workflowData.Explanation, - Action: action, - Version: 1, - Errors: errors, - }, nil -} - -// extractJSON extracts JSON from a response that might be wrapped in markdown -func (s *WorkflowGeneratorService) extractJSON(content string) string { - content = strings.TrimSpace(content) - - // If it starts with {, assume it's pure JSON - if strings.HasPrefix(content, "{") { - return content - } - - // Try to extract from markdown code block - re := regexp.MustCompile("(?s)```(?:json)?\\s*\\n?(\\{.*\\})\\s*\\n?```") - matches := re.FindStringSubmatch(content) - if len(matches) > 1 { - return matches[1] - } - - // Try to find JSON object anywhere in the content - re = regexp.MustCompile(`(?s)\{.*"blocks".*\}`) - match := re.FindString(content) - if match != "" { - return match - } - - return content -} - -// validateWorkflow validates the workflow structure -func (s *WorkflowGeneratorService) validateWorkflow(workflow *struct { - Blocks []models.Block `json:"blocks"` - Connections []models.Connection `json:"connections"` - Variables []models.Variable `json:"variables"` - Explanation string `json:"explanation"` -}) []models.ValidationError { - var errors []models.ValidationError - - // Check for empty blocks - if len(workflow.Blocks) == 0 { - errors = append(errors, models.ValidationError{ - Type: "schema", - Message: "Workflow must have at least one block", - }) - return errors - } - - // Build block ID set for connection validation - blockIDs := make(map[string]bool) - for _, block := range workflow.Blocks { - blockIDs[block.ID] = true - - // Validate block type - if block.Type != "llm_inference" && block.Type != "variable" { - errors = append(errors, models.ValidationError{ - Type: "schema", - Message: fmt.Sprintf("Invalid block type: %s", block.Type), - BlockID: block.ID, - }) - } - } - - // Validate connections reference valid blocks - for _, conn := range workflow.Connections { - if !blockIDs[conn.SourceBlockID] { - errors = append(errors, models.ValidationError{ - Type: "missing_input", - Message: fmt.Sprintf("Connection references non-existent source block: %s", conn.SourceBlockID), - ConnectionID: conn.ID, - }) - } - if !blockIDs[conn.TargetBlockID] { - errors = append(errors, models.ValidationError{ - Type: "missing_input", - Message: fmt.Sprintf("Connection references non-existent target block: %s", conn.TargetBlockID), - ConnectionID: conn.ID, - }) - } - } - - return errors -} - -// getProviderAndModel gets the provider and model for the request -func (s *WorkflowGeneratorService) getProviderAndModel(modelID string) (*models.Provider, string, error) { - // If no model specified, use default - if modelID == "" { - provider, model, err := s.chatService.GetDefaultProviderWithModel() - if err != nil { - return nil, "", err - } - return provider, model, nil - } - - // Try to find the model in the database - var providerID int - var modelName string - - err := s.db.QueryRow(` - SELECT m.name, m.provider_id - FROM models m - WHERE m.id = ? AND m.is_visible = 1 - `, modelID).Scan(&modelName, &providerID) - - if err != nil { - // Try as model alias - if provider, actualModel, found := s.chatService.ResolveModelAlias(modelID); found { - return provider, actualModel, nil - } - // Fall back to default - return s.chatService.GetDefaultProviderWithModel() - } - - // Get the provider - provider, err := s.providerService.GetByID(providerID) - if err != nil { - return nil, "", fmt.Errorf("failed to get provider: %w", err) - } - - return provider, modelName, nil -} - -func min(a, b int) int { - if a < b { - return a - } - return b -} - -// AgentMetadata holds generated name and description for an agent -type AgentMetadata struct { - Name string `json:"name"` - Description string `json:"description"` -} - -// GenerateAgentMetadata generates a name and description for an agent based on the user's request -func (s *WorkflowGeneratorService) GenerateAgentMetadata(userMessage string) (*AgentMetadata, error) { - // Use ChatService's GetTextProviderWithModel which dynamically finds a text-capable provider - // This method checks model aliases from config and falls back to database providers - provider, modelID, err := s.chatService.GetTextProviderWithModel() - if err != nil { - return nil, fmt.Errorf("failed to get text provider for metadata generation: %w", err) - } - - log.Printf("🔍 [METADATA-GEN] Using dynamic model: %s (provider: %s)", modelID, provider.Name) - - // Build a prompt that generates both name and description in a simple format - messages := []map[string]interface{}{ - { - "role": "system", - "content": `Generate a catchy name and brief description for an AI agent. - -RULES for name: -- 2-4 words maximum -- Action-oriented and memorable (e.g., "News Pulse", "Data Wizard", "Chart Crafter", "Report Runner") -- Use descriptive verbs or nouns that indicate the agent's purpose -- NEVER use generic words like "Agent", "Bot", "AI", "Assistant", "Helper", "Tool" -- Make it sound professional but approachable -- Be creative and specific to the task - -RULES for description: -- One sentence, maximum 100 characters -- Start with a verb (e.g., "Searches...", "Analyzes...", "Monitors...") -- Be specific about what the agent does -- Mention the key output or destination if relevant - -RESPOND with EXACTLY this format (two lines only): -NAME: [Your agent name here] -DESC: [Your one-line description here] - -Example for "search for AI news and post to Discord": -NAME: News Pulse -DESC: Searches for latest tech news and posts summaries to Discord - -Example for "analyze CSV data and create charts": -NAME: Chart Crafter -DESC: Analyzes data files and generates visual charts`, - }, - { - "role": "user", - "content": fmt.Sprintf("Agent purpose: %s", userMessage), - }, - } - - // Simple request like chat title generation - no structured output - requestBody := map[string]interface{}{ - "model": modelID, - "messages": messages, - "stream": false, - "temperature": 0.7, - } - - reqBody, err := json.Marshal(requestBody) - if err != nil { - return nil, fmt.Errorf("failed to marshal request: %w", err) - } - - // Create HTTP request - use base URL with /chat/completions - apiURL := provider.BaseURL + "/chat/completions" - log.Printf("🔍 [METADATA-GEN] Sending request to: %s with model: %s", apiURL, modelID) - - httpReq, err := http.NewRequest("POST", apiURL, bytes.NewBuffer(reqBody)) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - httpReq.Header.Set("Content-Type", "application/json") - httpReq.Header.Set("Authorization", "Bearer "+provider.APIKey) - - // Send request with timeout - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(httpReq) - if err != nil { - return nil, fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - body, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - log.Printf("🔍 [METADATA-GEN] Response status: %d, body length: %d", resp.StatusCode, len(body)) - - if resp.StatusCode != http.StatusOK { - log.Printf("❌ [METADATA-GEN] API error response: %s", string(body)) - return nil, fmt.Errorf("API error (status %d): %s", resp.StatusCode, string(body)) - } - - // Parse API response - var apiResponse struct { - Choices []struct { - Message struct { - Content string `json:"content"` - } `json:"message"` - } `json:"choices"` - } - - if err := json.Unmarshal(body, &apiResponse); err != nil { - return nil, fmt.Errorf("failed to parse API response: %w", err) - } - - if len(apiResponse.Choices) == 0 { - return nil, fmt.Errorf("no response from model") - } - - // Parse the NAME: and DESC: format from response - content := strings.TrimSpace(apiResponse.Choices[0].Message.Content) - log.Printf("🔍 [METADATA-GEN] Raw response: %s", content) - - var name, description string - - // Parse line by line looking for NAME: and DESC: - lines := strings.Split(content, "\n") - for _, line := range lines { - line = strings.TrimSpace(line) - if strings.HasPrefix(strings.ToUpper(line), "NAME:") { - name = strings.TrimSpace(strings.TrimPrefix(line, "NAME:")) - name = strings.TrimSpace(strings.TrimPrefix(line, "name:")) - // Remove the prefix more reliably - if idx := strings.Index(strings.ToLower(line), "name:"); idx != -1 { - name = strings.TrimSpace(line[idx+5:]) - } - } else if strings.HasPrefix(strings.ToUpper(line), "DESC:") { - description = strings.TrimSpace(strings.TrimPrefix(line, "DESC:")) - description = strings.TrimSpace(strings.TrimPrefix(line, "desc:")) - // Remove the prefix more reliably - if idx := strings.Index(strings.ToLower(line), "desc:"); idx != -1 { - description = strings.TrimSpace(line[idx+5:]) - } - } - } - - // Fallback: if parsing failed, try to use first line as name - if name == "" && len(lines) > 0 { - name = strings.TrimSpace(lines[0]) - name = strings.Trim(name, `"'#*-`) - } - - // Clean up name - name = strings.Trim(name, `"'#*-`) - - // Limit name to 5 words - words := strings.Fields(name) - if len(words) > 5 { - words = words[:5] - name = strings.Join(words, " ") - } - - if name == "" { - return nil, fmt.Errorf("empty name from model") - } - - metadata := AgentMetadata{ - Name: name, - Description: description, - } - - // Truncate if too long - if len(metadata.Name) > 50 { - metadata.Name = metadata.Name[:50] - } - if len(metadata.Description) > 150 { - metadata.Description = metadata.Description[:150] - } - - log.Printf("📝 [WORKFLOW-GEN] Generated agent metadata: name=%s, description=%s", metadata.Name, metadata.Description) - return &metadata, nil -} - -// GenerateAgentName generates a short, descriptive name for an agent (backwards compatibility) -func (s *WorkflowGeneratorService) GenerateAgentName(userMessage string) (string, error) { - metadata, err := s.GenerateAgentMetadata(userMessage) - if err != nil { - return "", err - } - return metadata.Name, nil -} - -// GenerateDescriptionFromWorkflow generates a description for an agent based on its workflow blocks -func (s *WorkflowGeneratorService) GenerateDescriptionFromWorkflow(workflow *models.Workflow, agentName string) (string, error) { - if workflow == nil || len(workflow.Blocks) == 0 { - return "", fmt.Errorf("no workflow blocks to analyze") - } - - // Use ChatService's GetTextProviderWithModel which dynamically finds a text-capable provider - // This method checks model aliases from config and falls back to database providers - provider, modelID, err := s.chatService.GetTextProviderWithModel() - if err != nil { - return "", fmt.Errorf("failed to get text provider for description generation: %w", err) - } - - log.Printf("🔍 [DESC-GEN] Using dynamic model: %s (provider: %s)", modelID, provider.Name) - - // Build a summary of the workflow blocks for the LLM - var blockSummary strings.Builder - blockSummary.WriteString("Workflow blocks:\n") - for _, block := range workflow.Blocks { - if block.Type == "llm_inference" { - // Extract key info from LLM blocks - tools := "" - if enabledTools, ok := block.Config["enabledTools"].([]interface{}); ok { - toolNames := make([]string, 0) - for _, t := range enabledTools { - if toolName, ok := t.(string); ok { - toolNames = append(toolNames, toolName) - } - } - tools = strings.Join(toolNames, ", ") - } - if tools != "" { - blockSummary.WriteString(fmt.Sprintf("- %s: %s (tools: %s)\n", block.Name, block.Description, tools)) - } else { - blockSummary.WriteString(fmt.Sprintf("- %s: %s\n", block.Name, block.Description)) - } - } else if block.Type == "variable" { - blockSummary.WriteString(fmt.Sprintf("- %s (input): %s\n", block.Name, block.Description)) - } - } - - messages := []map[string]interface{}{ - { - "role": "system", - "content": `Generate a brief, one-sentence description for an AI agent based on its workflow. - -RULES: -- Maximum 100 characters -- Start with a verb (e.g., "Searches...", "Analyzes...", "Monitors...") -- Be specific about what the agent does -- Mention the key actions or outputs -- Do not include the agent name in the description -- Do not use quotes around the description - -RESPOND with ONLY the description text, nothing else.`, - }, - { - "role": "user", - "content": fmt.Sprintf("Agent name: %s\n\n%s", agentName, blockSummary.String()), - }, - } - - requestBody := map[string]interface{}{ - "model": modelID, - "messages": messages, - "stream": false, - "temperature": 0.5, - } - - reqBody, err := json.Marshal(requestBody) - if err != nil { - return "", fmt.Errorf("failed to marshal request: %w", err) - } - - apiURL := provider.BaseURL + "/chat/completions" - log.Printf("🔍 [DESC-GEN] Generating description for agent: %s", agentName) - - httpReq, err := http.NewRequest("POST", apiURL, bytes.NewBuffer(reqBody)) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - httpReq.Header.Set("Content-Type", "application/json") - httpReq.Header.Set("Authorization", "Bearer "+provider.APIKey) - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(httpReq) - if err != nil { - return "", fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - body, err := io.ReadAll(resp.Body) - if err != nil { - return "", fmt.Errorf("failed to read response: %w", err) - } - - if resp.StatusCode != http.StatusOK { - return "", fmt.Errorf("API error (status %d): %s", resp.StatusCode, string(body)) - } - - var apiResponse struct { - Choices []struct { - Message struct { - Content string `json:"content"` - } `json:"message"` - } `json:"choices"` - } - - if err := json.Unmarshal(body, &apiResponse); err != nil { - return "", fmt.Errorf("failed to parse API response: %w", err) - } - - if len(apiResponse.Choices) == 0 { - return "", fmt.Errorf("no response from model") - } - - description := strings.TrimSpace(apiResponse.Choices[0].Message.Content) - description = strings.Trim(description, `"'`) - - // Truncate if too long - if len(description) > 150 { - description = description[:150] - } - - log.Printf("📝 [DESC-GEN] Generated description: %s", description) - return description, nil -} - -// GenerateSampleInput generates sample JSON input for a workflow based on its blocks -func (s *WorkflowGeneratorService) GenerateSampleInput(workflow *models.Workflow, modelID string, userID string) (map[string]interface{}, error) { - if workflow == nil || len(workflow.Blocks) == 0 { - return nil, fmt.Errorf("no workflow blocks to analyze") - } - - // Get provider and model - provider, resolvedModelID, err := s.getProviderAndModel(modelID) - if err != nil { - return nil, fmt.Errorf("failed to get provider: %w", err) - } - - log.Printf("🎯 [SAMPLE-INPUT] Generating sample input using model: %s (provider: %s)", resolvedModelID, provider.Name) - - // Build a summary of the workflow to understand what input it needs - var workflowSummary strings.Builder - workflowSummary.WriteString("This workflow has the following blocks:\n\n") - - for i, block := range workflow.Blocks { - workflowSummary.WriteString(fmt.Sprintf("Block %d: %s (type: %s)\n", i+1, block.Name, block.Type)) - - if block.Type == "llm_inference" { - // Extract system prompt and enabled tools - if systemPrompt, ok := block.Config["systemPrompt"].(string); ok && systemPrompt != "" { - // Truncate long prompts - if len(systemPrompt) > 500 { - systemPrompt = systemPrompt[:500] + "..." - } - workflowSummary.WriteString(fmt.Sprintf(" System prompt: %s\n", systemPrompt)) - } - - if enabledTools, ok := block.Config["enabledTools"].([]interface{}); ok && len(enabledTools) > 0 { - toolNames := make([]string, 0, len(enabledTools)) - for _, t := range enabledTools { - if ts, ok := t.(string); ok { - toolNames = append(toolNames, ts) - } - } - if len(toolNames) > 0 { - workflowSummary.WriteString(fmt.Sprintf(" Tools: %s\n", strings.Join(toolNames, ", "))) - } - } - } else if block.Type == "variable" { - workflowSummary.WriteString(" This is the start block that receives input\n") - } - workflowSummary.WriteString("\n") - } - - // Build messages for the LLM - messages := []map[string]interface{}{ - { - "role": "system", - "content": `You are a helpful assistant that generates realistic sample JSON input for AI workflow testing. - -Analyze the workflow description and generate appropriate sample JSON input that would be useful for testing this workflow. - -RULES: -1. Output ONLY valid JSON - no text before or after -2. Use realistic, meaningful sample data that matches what the workflow expects -3. If the workflow processes text, include relevant sample text -4. If it handles URLs, include valid example URLs -5. If it handles names/contacts, use realistic placeholder names -6. If it handles numbers/data, use reasonable sample values -7. Keep the JSON concise but complete -8. Use "input" as the top-level key if no specific structure is evident -9. Consider the tools being used - e.g., if web scraping, include a URL; if data analysis, include data points - -EXAMPLES: -- For a news search workflow: {"input": "latest developments in artificial intelligence"} -- For a data analysis workflow: {"data": [{"name": "Q1", "value": 1000}, {"name": "Q2", "value": 1500}]} -- For a web scraping workflow: {"url": "https://example.com/article", "extract": "main content"} -- For a contact workflow: {"name": "John Smith", "email": "john@example.com", "company": "Acme Corp"}`, - }, - { - "role": "user", - "content": fmt.Sprintf("Generate sample JSON input for this workflow:\n\n%s", workflowSummary.String()), - }, - } - - // Build request body - requestBody := map[string]interface{}{ - "model": resolvedModelID, - "messages": messages, - "stream": false, - "temperature": 0.7, - "response_format": map[string]interface{}{ - "type": "json_object", - }, - } - - reqBody, err := json.Marshal(requestBody) - if err != nil { - return nil, fmt.Errorf("failed to marshal request: %w", err) - } - - // Create HTTP request - apiURL := provider.BaseURL + "/chat/completions" - log.Printf("🔍 [SAMPLE-INPUT] Sending request to: %s", apiURL) - - httpReq, err := http.NewRequest("POST", apiURL, bytes.NewBuffer(reqBody)) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - httpReq.Header.Set("Content-Type", "application/json") - httpReq.Header.Set("Authorization", "Bearer "+provider.APIKey) - - // Send request with timeout - client := &http.Client{Timeout: 60 * time.Second} - resp, err := client.Do(httpReq) - if err != nil { - return nil, fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - body, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - if resp.StatusCode != http.StatusOK { - log.Printf("⚠️ [SAMPLE-INPUT] API error: %s", string(body)) - return nil, fmt.Errorf("API error (status %d): %s", resp.StatusCode, string(body)) - } - - // Parse API response - var apiResponse struct { - Choices []struct { - Message struct { - Content string `json:"content"` - } `json:"message"` - } `json:"choices"` - } - - if err := json.Unmarshal(body, &apiResponse); err != nil { - return nil, fmt.Errorf("failed to parse API response: %w", err) - } - - if len(apiResponse.Choices) == 0 { - return nil, fmt.Errorf("no response from model") - } - - content := strings.TrimSpace(apiResponse.Choices[0].Message.Content) - log.Printf("📥 [SAMPLE-INPUT] Received response: %s", content) - - // Parse the JSON response - var sampleInput map[string]interface{} - if err := json.Unmarshal([]byte(content), &sampleInput); err != nil { - // Try to extract JSON from the response - jsonContent := s.extractJSON(content) - if err := json.Unmarshal([]byte(jsonContent), &sampleInput); err != nil { - return nil, fmt.Errorf("failed to parse sample input JSON: %w", err) - } - } - - log.Printf("✅ [SAMPLE-INPUT] Generated sample input with %d keys", len(sampleInput)) - return sampleInput, nil -} diff --git a/backend/internal/services/workflow_generator_v2.go b/backend/internal/services/workflow_generator_v2.go deleted file mode 100644 index c1bb37fe..00000000 --- a/backend/internal/services/workflow_generator_v2.go +++ /dev/null @@ -1,669 +0,0 @@ -package services - -import ( - "bytes" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "strings" - "time" - - "github.com/google/uuid" - - "claraverse/internal/database" - "claraverse/internal/models" -) - -// WorkflowGeneratorV2Service handles multi-step workflow generation -type WorkflowGeneratorV2Service struct { - db *database.DB - providerService *ProviderService - chatService *ChatService -} - -// NewWorkflowGeneratorV2Service creates a new v2 workflow generator service -func NewWorkflowGeneratorV2Service( - db *database.DB, - providerService *ProviderService, - chatService *ChatService, -) *WorkflowGeneratorV2Service { - return &WorkflowGeneratorV2Service{ - db: db, - providerService: providerService, - chatService: chatService, - } -} - -// ToolSelectionResult represents the result of tool selection (Step 1) -type ToolSelectionResult struct { - SelectedTools []SelectedTool `json:"selected_tools"` - Reasoning string `json:"reasoning"` -} - -// SelectedTool represents a selected tool with reasoning -type SelectedTool struct { - ToolID string `json:"tool_id"` - Category string `json:"category"` - Reason string `json:"reason"` -} - -// GenerationStep represents a step in the generation process -type GenerationStep struct { - StepNumber int `json:"step_number"` - StepName string `json:"step_name"` - Status string `json:"status"` // "pending", "running", "completed", "failed" - Description string `json:"description"` - Tools []string `json:"tools,omitempty"` // Tool IDs for step 1 result -} - -// MultiStepGenerateRequest is the request for multi-step generation -type MultiStepGenerateRequest struct { - AgentID string `json:"agent_id"` - UserMessage string `json:"user_message"` - ModelID string `json:"model_id,omitempty"` - CurrentWorkflow *models.Workflow `json:"current_workflow,omitempty"` - ConversationHistory []models.ConversationMessage `json:"conversation_history,omitempty"` // Recent conversation context for better tool selection -} - -// MultiStepGenerateResponse is the response for multi-step generation -type MultiStepGenerateResponse struct { - Success bool `json:"success"` - CurrentStep int `json:"current_step"` - TotalSteps int `json:"total_steps"` - Steps []GenerationStep `json:"steps"` - SelectedTools []SelectedTool `json:"selected_tools,omitempty"` - Workflow *models.Workflow `json:"workflow,omitempty"` - Explanation string `json:"explanation,omitempty"` - Error string `json:"error,omitempty"` - StepInProgress *GenerationStep `json:"step_in_progress,omitempty"` -} - -// Tool selection system prompt - asks LLM to select relevant tools -const ToolSelectionSystemPrompt = `You are a tool selection expert for Clara AI workflow builder. Your job is to analyze user requests and select the MINIMUM set of tools needed to accomplish the task. - -IMPORTANT: Only select tools that are DIRECTLY needed for the workflow. Don't over-select. - -You will be given: -1. A user request describing what workflow they want to build -2. A list of all available tools with their descriptions and use cases - -Your task: Select the specific tools needed and explain why each is needed. - -Rules: -- Select ONLY tools that will be directly used in the workflow -- If the request mentions "news" or time-sensitive info, ALWAYS include "get_current_time" -- If the request mentions sending to a specific platform (Discord, Slack, etc.), select that messaging tool -- Don't select redundant tools - if search_web is enough, don't also select scrape_web unless needed -- For file processing, select the appropriate reader tool based on file type mentioned - -Output format: JSON with selected_tools array and reasoning.` - -// BuildToolSelectionUserPrompt builds the user prompt for tool selection -func (s *WorkflowGeneratorV2Service) BuildToolSelectionUserPrompt(userMessage string) string { - var builder strings.Builder - - builder.WriteString("USER REQUEST:\n") - builder.WriteString(userMessage) - builder.WriteString("\n\n") - builder.WriteString("AVAILABLE TOOLS:\n\n") - - // Group tools by category - for _, category := range ToolCategoryRegistry { - tools := GetToolsByCategory(category.ID) - if len(tools) == 0 { - continue - } - - builder.WriteString(fmt.Sprintf("## %s\n", category.Name)) - for _, tool := range tools { - builder.WriteString(fmt.Sprintf("- **%s**: %s\n", tool.ID, tool.Description)) - if len(tool.UseCases) > 0 { - builder.WriteString(fmt.Sprintf(" Use cases: %s\n", strings.Join(tool.UseCases, ", "))) - } - } - builder.WriteString("\n") - } - - builder.WriteString("\nSelect the tools needed for this workflow. Return JSON with selected_tools array.") - - return builder.String() -} - -// Tool selection JSON schema for structured output -var toolSelectionSchema = map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "selected_tools": map[string]interface{}{ - "type": "array", - "items": map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "tool_id": map[string]interface{}{ - "type": "string", - "description": "The tool ID from the available tools list", - }, - "category": map[string]interface{}{ - "type": "string", - "description": "The category of the tool", - }, - "reason": map[string]interface{}{ - "type": "string", - "description": "Brief reason why this tool is needed", - }, - }, - "required": []string{"tool_id", "category", "reason"}, - "additionalProperties": false, - }, - }, - "reasoning": map[string]interface{}{ - "type": "string", - "description": "Overall reasoning for the tool selection", - }, - }, - "required": []string{"selected_tools", "reasoning"}, - "additionalProperties": false, -} - -// Step1SelectTools performs tool selection using structured output -func (s *WorkflowGeneratorV2Service) Step1SelectTools(req *MultiStepGenerateRequest, userID string) (*ToolSelectionResult, error) { - log.Printf("🔧 [WORKFLOW-GEN-V2] Step 1: Selecting tools for request: %s", req.UserMessage) - - // Get provider and model - provider, modelID, err := s.getProviderAndModel(req.ModelID) - if err != nil { - return nil, fmt.Errorf("failed to get provider: %w", err) - } - - // Build the user prompt with all available tools - userPrompt := s.BuildToolSelectionUserPrompt(req.UserMessage) - - // Build messages with conversation history for better context - messages := []map[string]interface{}{ - { - "role": "system", - "content": ToolSelectionSystemPrompt, - }, - } - - // Add conversation history if provided (for multi-turn context) - if len(req.ConversationHistory) > 0 { - for _, msg := range req.ConversationHistory { - messages = append(messages, map[string]interface{}{ - "role": msg.Role, - "content": msg.Content, - }) - } - } - - // Add current user message - messages = append(messages, map[string]interface{}{ - "role": "user", - "content": userPrompt, - }) - - // Build request with structured output - requestBody := map[string]interface{}{ - "model": modelID, - "messages": messages, - "stream": false, - "temperature": 0.2, // Low temperature for consistent selection - "response_format": map[string]interface{}{ - "type": "json_schema", - "json_schema": map[string]interface{}{ - "name": "tool_selection", - "strict": true, - "schema": toolSelectionSchema, - }, - }, - } - - reqBody, err := json.Marshal(requestBody) - if err != nil { - return nil, fmt.Errorf("failed to marshal request: %w", err) - } - - log.Printf("📤 [WORKFLOW-GEN-V2] Sending tool selection request to %s", provider.BaseURL) - - // Create HTTP request - httpReq, err := http.NewRequest("POST", provider.BaseURL+"/chat/completions", bytes.NewBuffer(reqBody)) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - httpReq.Header.Set("Content-Type", "application/json") - httpReq.Header.Set("Authorization", "Bearer "+provider.APIKey) - - // Send request - client := &http.Client{Timeout: 60 * time.Second} - resp, err := client.Do(httpReq) - if err != nil { - return nil, fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - body, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - if resp.StatusCode != http.StatusOK { - log.Printf("⚠️ [WORKFLOW-GEN-V2] API error: %s", string(body)) - return nil, fmt.Errorf("API error (status %d): %s", resp.StatusCode, string(body)) - } - - // Parse response - var apiResponse struct { - Choices []struct { - Message struct { - Content string `json:"content"` - } `json:"message"` - } `json:"choices"` - } - - if err := json.Unmarshal(body, &apiResponse); err != nil { - return nil, fmt.Errorf("failed to parse API response: %w", err) - } - - if len(apiResponse.Choices) == 0 { - return nil, fmt.Errorf("no response from model") - } - - // Parse the tool selection result - var result ToolSelectionResult - content := apiResponse.Choices[0].Message.Content - - if err := json.Unmarshal([]byte(content), &result); err != nil { - log.Printf("⚠️ [WORKFLOW-GEN-V2] Failed to parse tool selection: %v, content: %s", err, content) - return nil, fmt.Errorf("failed to parse tool selection: %w", err) - } - - // Validate selected tools exist - validTools := make([]SelectedTool, 0) - for _, selected := range result.SelectedTools { - if tool := GetToolByID(selected.ToolID); tool != nil { - selected.Category = tool.Category // Ensure category is correct - validTools = append(validTools, selected) - } else { - log.Printf("⚠️ [WORKFLOW-GEN-V2] Unknown tool selected: %s, skipping", selected.ToolID) - } - } - result.SelectedTools = validTools - - log.Printf("✅ [WORKFLOW-GEN-V2] Selected %d tools: %v", len(result.SelectedTools), getToolIDs(result.SelectedTools)) - - return &result, nil -} - -// Step2GenerateWorkflow generates the workflow using only selected tools -func (s *WorkflowGeneratorV2Service) Step2GenerateWorkflow( - req *MultiStepGenerateRequest, - selectedTools []SelectedTool, - userID string, -) (*models.WorkflowGenerateResponse, error) { - log.Printf("🔧 [WORKFLOW-GEN-V2] Step 2: Generating workflow with %d tools", len(selectedTools)) - - // Get provider and model - provider, modelID, err := s.getProviderAndModel(req.ModelID) - if err != nil { - return &models.WorkflowGenerateResponse{ - Success: false, - Error: fmt.Sprintf("Failed to get provider: %v", err), - }, nil - } - - // Build tool IDs list - toolIDs := getToolIDs(selectedTools) - - // Build system prompt with only selected tools - systemPrompt := s.buildWorkflowSystemPromptWithTools(toolIDs) - - // Build user message - userMessage := s.buildUserMessage(req) - - // Build messages with conversation history for better context - messages := []map[string]interface{}{ - { - "role": "system", - "content": systemPrompt, - }, - } - - // Add conversation history if provided (for multi-turn context) - if len(req.ConversationHistory) > 0 { - for _, msg := range req.ConversationHistory { - messages = append(messages, map[string]interface{}{ - "role": msg.Role, - "content": msg.Content, - }) - } - } - - // Add current user message - messages = append(messages, map[string]interface{}{ - "role": "user", - "content": userMessage, - }) - - // Build request - requestBody := map[string]interface{}{ - "model": modelID, - "messages": messages, - "stream": false, - "temperature": 0.3, - "response_format": map[string]interface{}{ - "type": "json_object", - }, - } - - reqBody, err := json.Marshal(requestBody) - if err != nil { - return nil, fmt.Errorf("failed to marshal request: %w", err) - } - - log.Printf("📤 [WORKFLOW-GEN-V2] Sending workflow generation request") - - // Create HTTP request - httpReq, err := http.NewRequest("POST", provider.BaseURL+"/chat/completions", bytes.NewBuffer(reqBody)) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - httpReq.Header.Set("Content-Type", "application/json") - httpReq.Header.Set("Authorization", "Bearer "+provider.APIKey) - - // Send request - client := &http.Client{Timeout: 120 * time.Second} - resp, err := client.Do(httpReq) - if err != nil { - return nil, fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - body, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - if resp.StatusCode != http.StatusOK { - log.Printf("⚠️ [WORKFLOW-GEN-V2] API error: %s", string(body)) - return &models.WorkflowGenerateResponse{ - Success: false, - Error: fmt.Sprintf("API error (status %d): %s", resp.StatusCode, string(body)), - }, nil - } - - // Parse response - var apiResponse struct { - Choices []struct { - Message struct { - Content string `json:"content"` - } `json:"message"` - } `json:"choices"` - } - - if err := json.Unmarshal(body, &apiResponse); err != nil { - return nil, fmt.Errorf("failed to parse API response: %w", err) - } - - if len(apiResponse.Choices) == 0 { - return &models.WorkflowGenerateResponse{ - Success: false, - Error: "No response from model", - }, nil - } - - content := apiResponse.Choices[0].Message.Content - log.Printf("📥 [WORKFLOW-GEN-V2] Received workflow response (%d chars)", len(content)) - - // Parse the workflow - return s.parseWorkflowResponse(content, req.CurrentWorkflow != nil, req.AgentID) -} - -// GenerateWorkflowMultiStep performs the full multi-step generation -func (s *WorkflowGeneratorV2Service) GenerateWorkflowMultiStep( - req *MultiStepGenerateRequest, - userID string, - stepCallback func(step GenerationStep), -) (*MultiStepGenerateResponse, error) { - response := &MultiStepGenerateResponse{ - TotalSteps: 2, - Steps: []GenerationStep{ - {StepNumber: 1, StepName: "Tool Selection", Status: "pending", Description: "Analyzing request and selecting relevant tools"}, - {StepNumber: 2, StepName: "Workflow Generation", Status: "pending", Description: "Building the workflow with selected tools"}, - }, - } - - // Step 1: Tool Selection - response.Steps[0].Status = "running" - response.CurrentStep = 1 - response.StepInProgress = &response.Steps[0] - if stepCallback != nil { - stepCallback(response.Steps[0]) - } - - toolResult, err := s.Step1SelectTools(req, userID) - if err != nil { - response.Steps[0].Status = "failed" - response.Success = false - response.Error = fmt.Sprintf("Tool selection failed: %v", err) - return response, nil - } - - response.Steps[0].Status = "completed" - response.Steps[0].Tools = getToolIDs(toolResult.SelectedTools) - response.SelectedTools = toolResult.SelectedTools - - if stepCallback != nil { - stepCallback(response.Steps[0]) - } - - // Step 2: Workflow Generation - response.Steps[1].Status = "running" - response.CurrentStep = 2 - response.StepInProgress = &response.Steps[1] - if stepCallback != nil { - stepCallback(response.Steps[1]) - } - - workflowResult, err := s.Step2GenerateWorkflow(req, toolResult.SelectedTools, userID) - if err != nil { - response.Steps[1].Status = "failed" - response.Success = false - response.Error = fmt.Sprintf("Workflow generation failed: %v", err) - return response, nil - } - - if !workflowResult.Success { - response.Steps[1].Status = "failed" - response.Success = false - response.Error = workflowResult.Error - return response, nil - } - - response.Steps[1].Status = "completed" - response.Workflow = workflowResult.Workflow - response.Explanation = workflowResult.Explanation - response.Success = true - response.StepInProgress = nil - - if stepCallback != nil { - stepCallback(response.Steps[1]) - } - - log.Printf("✅ [WORKFLOW-GEN-V2] Multi-step generation completed successfully") - - return response, nil -} - -// buildWorkflowSystemPromptWithTools builds the system prompt with specific tools -func (s *WorkflowGeneratorV2Service) buildWorkflowSystemPromptWithTools(toolIDs []string) string { - toolsSection := BuildToolPromptSection(toolIDs) - return strings.Replace(WorkflowSystemPromptBase, "{{DYNAMIC_TOOLS_SECTION}}", toolsSection, 1) -} - -// buildUserMessage constructs the user message for workflow generation -func (s *WorkflowGeneratorV2Service) buildUserMessage(req *MultiStepGenerateRequest) string { - if req.CurrentWorkflow != nil && len(req.CurrentWorkflow.Blocks) > 0 { - workflowJSON, _ := json.MarshalIndent(req.CurrentWorkflow, "", " ") - return fmt.Sprintf(`MODIFICATION REQUEST - -Current workflow: -%s - -User request: %s - -Output the complete modified workflow JSON with all blocks (not just changes).`, string(workflowJSON), req.UserMessage) - } - - return fmt.Sprintf("CREATE NEW WORKFLOW\n\nUser request: %s", req.UserMessage) -} - -// parseWorkflowResponse parses the LLM response into a workflow -func (s *WorkflowGeneratorV2Service) parseWorkflowResponse(content string, isModification bool, agentID string) (*models.WorkflowGenerateResponse, error) { - // Try to extract JSON from the response - jsonContent := extractJSON(content) - - // Parse the workflow - var workflowData struct { - Blocks []models.Block `json:"blocks"` - Connections []models.Connection `json:"connections"` - Variables []models.Variable `json:"variables"` - Explanation string `json:"explanation"` - } - - if err := json.Unmarshal([]byte(jsonContent), &workflowData); err != nil { - log.Printf("⚠️ [WORKFLOW-GEN-V2] Failed to parse workflow JSON: %v", err) - return &models.WorkflowGenerateResponse{ - Success: false, - Error: fmt.Sprintf("Failed to parse workflow JSON: %v", err), - Explanation: content, - }, nil - } - - // Log the generated workflow for debugging - prettyWorkflow, _ := json.MarshalIndent(workflowData, "", " ") - log.Printf("📋 [WORKFLOW-GEN-V2] Generated workflow:\n%s", string(prettyWorkflow)) - - // Post-process blocks - for i := range workflowData.Blocks { - if workflowData.Blocks[i].NormalizedID == "" { - workflowData.Blocks[i].NormalizedID = workflowData.Blocks[i].ID - } - } - - // Determine action - action := "create" - if isModification { - action = "modify" - } - - // Build the workflow - workflow := &models.Workflow{ - ID: uuid.New().String(), - AgentID: agentID, - Blocks: workflowData.Blocks, - Connections: workflowData.Connections, - Variables: workflowData.Variables, - Version: 1, - } - - log.Printf("✅ [WORKFLOW-GEN-V2] Parsed workflow: %d blocks, %d connections", - len(workflow.Blocks), len(workflow.Connections)) - - return &models.WorkflowGenerateResponse{ - Success: true, - Workflow: workflow, - Explanation: workflowData.Explanation, - Action: action, - Version: 1, - }, nil -} - -// getProviderAndModel gets the provider and model for the request -func (s *WorkflowGeneratorV2Service) getProviderAndModel(modelID string) (*models.Provider, string, error) { - if modelID == "" { - return s.chatService.GetDefaultProviderWithModel() - } - - // Try to find the model in the database - var providerID int - var modelName string - - err := s.db.QueryRow(` - SELECT m.name, m.provider_id - FROM models m - WHERE m.id = ? AND m.is_visible = 1 - `, modelID).Scan(&modelName, &providerID) - - if err != nil { - if provider, actualModel, found := s.chatService.ResolveModelAlias(modelID); found { - return provider, actualModel, nil - } - return s.chatService.GetDefaultProviderWithModel() - } - - provider, err := s.providerService.GetByID(providerID) - if err != nil { - return nil, "", fmt.Errorf("failed to get provider: %w", err) - } - - return provider, modelName, nil -} - -// Helper function to get tool IDs from selected tools -func getToolIDs(tools []SelectedTool) []string { - ids := make([]string, len(tools)) - for i, t := range tools { - ids[i] = t.ToolID - } - return ids -} - -// extractJSON extracts JSON from a response (handles markdown code blocks) -func extractJSON(content string) string { - content = strings.TrimSpace(content) - - if strings.HasPrefix(content, "{") { - return content - } - - // Try to extract from markdown code block - if idx := strings.Index(content, "```json"); idx != -1 { - start := idx + 7 - end := strings.Index(content[start:], "```") - if end != -1 { - return strings.TrimSpace(content[start : start+end]) - } - } - - if idx := strings.Index(content, "```"); idx != -1 { - start := idx + 3 - // Skip language identifier if present - if newline := strings.Index(content[start:], "\n"); newline != -1 { - start = start + newline + 1 - } - end := strings.Index(content[start:], "```") - if end != -1 { - return strings.TrimSpace(content[start : start+end]) - } - } - - // Find JSON object - if start := strings.Index(content, "{"); start != -1 { - depth := 0 - for i := start; i < len(content); i++ { - if content[i] == '{' { - depth++ - } else if content[i] == '}' { - depth-- - if depth == 0 { - return content[start : i+1] - } - } - } - } - - return content -} diff --git a/backend/internal/tests/legacy_migration_test.go b/backend/internal/tests/legacy_migration_test.go deleted file mode 100644 index 8fa279a1..00000000 --- a/backend/internal/tests/legacy_migration_test.go +++ /dev/null @@ -1,537 +0,0 @@ -package tests - -import ( - "claraverse/internal/handlers" - "claraverse/internal/models" - "context" - "encoding/json" - "io" - "net/http/httptest" - "testing" - "time" - - "github.com/gofiber/fiber/v2" - "go.mongodb.org/mongo-driver/bson" -) - -// ============================================================================ -// Test: Production User (Main Branch) Before Migration -// ============================================================================ -// This test verifies what happens when a user from production (main branch) -// without any subscription fields logs in. They should get "free" tier. - -func TestE2E_ProductionUserLoginBeforeMigration(t *testing.T) { - if testing.Short() { - t.Skip("Skipping E2E test in short mode") - } - - // Setup with promo disabled to avoid interference - ts := SetupE2ETestWithMongoDB(t, &PromoTestConfig{ - PromoEnabled: false, - }) - if ts == nil { - return - } - defer ts.Cleanup() - - ctx := context.Background() - userID := "test-e2e-prod-before-" + time.Now().Format("20060102150405") - email := "test-e2e-prod-before@example.com" - - // Create user with ONLY main branch fields (simulating production user) - // Main branch users have: _id, supabaseUserId, email, createdAt, lastLoginAt, preferences - // NO subscriptionTier, subscriptionStatus, or other new fields - collection := ts.MongoDB.Database().Collection("users") - _, err := collection.InsertOne(ctx, bson.M{ - "supabaseUserId": userID, - "email": email, - "createdAt": time.Now().Add(-90 * 24 * time.Hour), // Created 90 days ago - "lastLoginAt": time.Now().Add(-24 * time.Hour), - "preferences": bson.M{ - "storeBuilderChatHistory": true, - "chatPrivacyMode": "cloud", - }, - // Explicitly NOT setting: subscriptionTier, subscriptionStatus, etc. - }) - if err != nil { - t.Fatalf("Failed to create production-like user: %v", err) - } - - // Verify user was created without subscription fields - var createdUser bson.M - err = collection.FindOne(ctx, bson.M{"supabaseUserId": userID}).Decode(&createdUser) - if err != nil { - t.Fatalf("Failed to fetch created user: %v", err) - } - if _, exists := createdUser["subscriptionTier"]; exists { - t.Fatal("User should NOT have subscriptionTier field (simulating main branch)") - } - - // Create app with auth middleware for this user - app := fiber.New(fiber.Config{DisableStartupMessage: true}) - app.Use(testAuthMiddleware(userID, email)) - - subHandler := handlers.NewSubscriptionHandler(ts.PaymentService, ts.UserService) - app.Get("/api/subscriptions/current", subHandler.GetCurrent) - - // Call GET /api/subscriptions/current - req := httptest.NewRequest("GET", "/api/subscriptions/current", nil) - resp, err := app.Test(req, -1) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != 200 { - body, _ := io.ReadAll(resp.Body) - t.Fatalf("Expected 200, got %d: %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - if err := json.NewDecoder(resp.Body).Decode(&result); err != nil { - t.Fatalf("Failed to decode response: %v", err) - } - - // Without migration, TierService defaults to "free" for users without tier - tier := ts.TierService.GetUserTier(ctx, userID) - if tier != models.TierFree { - t.Errorf("Expected TierService to return 'free' for unmigrated user, got '%s'", tier) - } - - t.Logf("Response tier: %v", result["tier"]) - t.Logf("TierService tier: %s", tier) - t.Log("Confirmed: Production users without subscription fields get 'free' tier until migrated") -} - -// ============================================================================ -// Test: Production User Migration to Legacy Unlimited -// ============================================================================ -// This test simulates the migration script logic and verifies it works correctly. - -func TestE2E_ProductionUserMigration(t *testing.T) { - if testing.Short() { - t.Skip("Skipping E2E test in short mode") - } - - ts := SetupE2ETestWithMongoDB(t, &PromoTestConfig{ - PromoEnabled: false, - }) - if ts == nil { - return - } - defer ts.Cleanup() - - ctx := context.Background() - userID := "test-e2e-migrate-" + time.Now().Format("20060102150405") - email := "test-e2e-migrate@example.com" - - collection := ts.MongoDB.Database().Collection("users") - - // Step 1: Create production-like user (no subscription fields) - _, err := collection.InsertOne(ctx, bson.M{ - "supabaseUserId": userID, - "email": email, - "createdAt": time.Now().Add(-90 * 24 * time.Hour), - "lastLoginAt": time.Now().Add(-24 * time.Hour), - "preferences": bson.M{ - "storeBuilderChatHistory": true, - }, - }) - if err != nil { - t.Fatalf("Failed to create production-like user: %v", err) - } - - // Step 2: Apply migration logic (same as migrate_legacy_users.go) - migrationFilter := bson.M{ - "$or": []bson.M{ - {"subscriptionTier": bson.M{"$exists": false}}, - {"subscriptionTier": ""}, - }, - } - migrationTime := time.Now() - migrationUpdate := bson.M{ - "$set": bson.M{ - "subscriptionTier": models.TierLegacyUnlimited, - "subscriptionStatus": models.SubStatusActive, - "migratedToLegacyAt": migrationTime, - }, - } - - result, err := collection.UpdateMany(ctx, migrationFilter, migrationUpdate) - if err != nil { - t.Fatalf("Migration failed: %v", err) - } - if result.ModifiedCount == 0 { - t.Fatal("Expected at least 1 user to be migrated") - } - t.Logf("Migrated %d user(s)", result.ModifiedCount) - - // Step 3: Verify user now has legacy_unlimited tier - var migratedUser models.User - err = collection.FindOne(ctx, bson.M{"supabaseUserId": userID}).Decode(&migratedUser) - if err != nil { - t.Fatalf("Failed to fetch migrated user: %v", err) - } - - if migratedUser.SubscriptionTier != models.TierLegacyUnlimited { - t.Errorf("Expected tier '%s', got '%s'", models.TierLegacyUnlimited, migratedUser.SubscriptionTier) - } - if migratedUser.SubscriptionStatus != models.SubStatusActive { - t.Errorf("Expected status '%s', got '%s'", models.SubStatusActive, migratedUser.SubscriptionStatus) - } - if migratedUser.MigratedToLegacyAt == nil { - t.Error("Expected migratedToLegacyAt to be set") - } - - // Step 4: Verify TierService returns correct tier - // Invalidate cache first since we updated directly in DB - ts.TierService.InvalidateCache(userID) - tier := ts.TierService.GetUserTier(ctx, userID) - if tier != models.TierLegacyUnlimited { - t.Errorf("Expected TierService to return '%s', got '%s'", models.TierLegacyUnlimited, tier) - } - - // Step 5: Verify limits are unlimited - limits := models.GetTierLimits(tier) - if limits.MaxSchedules != -1 { - t.Errorf("Expected unlimited schedules (-1), got %d", limits.MaxSchedules) - } - if limits.MaxAPIKeys != -1 { - t.Errorf("Expected unlimited API keys (-1), got %d", limits.MaxAPIKeys) - } - - t.Log("Migration test passed: Production user successfully migrated to legacy_unlimited") -} - -// ============================================================================ -// Test: Production User Login After Migration -// ============================================================================ -// This tests the full flow: production user already migrated, then logs in. - -func TestE2E_ProductionUserLoginAfterMigration(t *testing.T) { - if testing.Short() { - t.Skip("Skipping E2E test in short mode") - } - - // Setup with ACTIVE promo (to verify legacy users are NOT affected by promo) - now := time.Now() - ts := SetupE2ETestWithMongoDB(t, &PromoTestConfig{ - PromoEnabled: true, - PromoStartDate: now.Add(-1 * time.Hour), - PromoEndDate: now.Add(24 * time.Hour), - PromoDuration: 30, - }) - if ts == nil { - return - } - defer ts.Cleanup() - - ctx := context.Background() - userID := "test-e2e-post-migrate-" + time.Now().Format("20060102150405") - email := "test-e2e-post-migrate@example.com" - - collection := ts.MongoDB.Database().Collection("users") - - // Create user that has ALREADY been migrated (simulating post-migration state) - migratedAt := time.Now().Add(-7 * 24 * time.Hour) // Migrated 7 days ago - _, err := collection.InsertOne(ctx, bson.M{ - "supabaseUserId": userID, - "email": email, - "createdAt": time.Now().Add(-120 * 24 * time.Hour), // Created 120 days ago - "lastLoginAt": time.Now().Add(-24 * time.Hour), - "subscriptionTier": models.TierLegacyUnlimited, - "subscriptionStatus": models.SubStatusActive, - "migratedToLegacyAt": migratedAt, - "preferences": bson.M{ - "storeBuilderChatHistory": true, - }, - }) - if err != nil { - t.Fatalf("Failed to create migrated user: %v", err) - } - - // Create app with auth middleware - app := fiber.New(fiber.Config{DisableStartupMessage: true}) - app.Use(testAuthMiddleware(userID, email)) - - subHandler := handlers.NewSubscriptionHandler(ts.PaymentService, ts.UserService) - app.Get("/api/subscriptions/current", subHandler.GetCurrent) - - // Call GET /api/subscriptions/current - req := httptest.NewRequest("GET", "/api/subscriptions/current", nil) - resp, err := app.Test(req, -1) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != 200 { - body, _ := io.ReadAll(resp.Body) - t.Fatalf("Expected 200, got %d: %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - if err := json.NewDecoder(resp.Body).Decode(&result); err != nil { - t.Fatalf("Failed to decode response: %v", err) - } - - // Verify tier is legacy_unlimited (NOT converted to promo even though promo is active) - if result["tier"] != models.TierLegacyUnlimited { - t.Errorf("Expected tier '%s', got '%v'", models.TierLegacyUnlimited, result["tier"]) - } - - // Verify is_promo_user is false - if result["is_promo_user"] != false { - t.Errorf("Expected is_promo_user false for legacy user, got '%v'", result["is_promo_user"]) - } - - // Verify via TierService - tier := ts.TierService.GetUserTier(ctx, userID) - if tier != models.TierLegacyUnlimited { - t.Errorf("TierService should return '%s', got '%s'", models.TierLegacyUnlimited, tier) - } - - t.Log("Post-migration login test passed: Legacy user retains legacy_unlimited tier") -} - -// ============================================================================ -// Test: Migration Idempotency -// ============================================================================ -// Running migration twice should not break anything or update migratedToLegacyAt. - -func TestE2E_MigrationIdempotent(t *testing.T) { - if testing.Short() { - t.Skip("Skipping E2E test in short mode") - } - - ts := SetupE2ETestWithMongoDB(t, &PromoTestConfig{ - PromoEnabled: false, - }) - if ts == nil { - return - } - defer ts.Cleanup() - - ctx := context.Background() - userID := "test-e2e-idempotent-" + time.Now().Format("20060102150405") - email := "test-e2e-idempotent@example.com" - - collection := ts.MongoDB.Database().Collection("users") - - // Create production-like user - _, err := collection.InsertOne(ctx, bson.M{ - "supabaseUserId": userID, - "email": email, - "createdAt": time.Now().Add(-60 * 24 * time.Hour), - "lastLoginAt": time.Now().Add(-24 * time.Hour), - }) - if err != nil { - t.Fatalf("Failed to create user: %v", err) - } - - // First migration - migrationFilter := bson.M{ - "$or": []bson.M{ - {"subscriptionTier": bson.M{"$exists": false}}, - {"subscriptionTier": ""}, - }, - } - firstMigrationTime := time.Now() - migrationUpdate := bson.M{ - "$set": bson.M{ - "subscriptionTier": models.TierLegacyUnlimited, - "subscriptionStatus": models.SubStatusActive, - "migratedToLegacyAt": firstMigrationTime, - }, - } - - result1, err := collection.UpdateMany(ctx, migrationFilter, migrationUpdate) - if err != nil { - t.Fatalf("First migration failed: %v", err) - } - if result1.ModifiedCount != 1 { - t.Fatalf("Expected 1 user migrated on first run, got %d", result1.ModifiedCount) - } - - // Get migratedToLegacyAt after first migration - var userAfterFirst models.User - err = collection.FindOne(ctx, bson.M{"supabaseUserId": userID}).Decode(&userAfterFirst) - if err != nil { - t.Fatalf("Failed to fetch user after first migration: %v", err) - } - firstMigratedAt := userAfterFirst.MigratedToLegacyAt - - // Wait a moment to ensure time difference would be detectable - time.Sleep(10 * time.Millisecond) - - // Second migration (should not match the user since they already have a tier) - result2, err := collection.UpdateMany(ctx, migrationFilter, bson.M{ - "$set": bson.M{ - "subscriptionTier": models.TierLegacyUnlimited, - "subscriptionStatus": models.SubStatusActive, - "migratedToLegacyAt": time.Now(), // Different time - }, - }) - if err != nil { - t.Fatalf("Second migration failed: %v", err) - } - if result2.ModifiedCount != 0 { - t.Errorf("Expected 0 users migrated on second run (already migrated), got %d", result2.ModifiedCount) - } - - // Verify migratedToLegacyAt unchanged - var userAfterSecond models.User - err = collection.FindOne(ctx, bson.M{"supabaseUserId": userID}).Decode(&userAfterSecond) - if err != nil { - t.Fatalf("Failed to fetch user after second migration: %v", err) - } - - if userAfterSecond.MigratedToLegacyAt == nil || firstMigratedAt == nil { - t.Error("migratedToLegacyAt should be set") - } else if !userAfterSecond.MigratedToLegacyAt.Equal(*firstMigratedAt) { - t.Errorf("migratedToLegacyAt should not change: first=%v, second=%v", - firstMigratedAt, userAfterSecond.MigratedToLegacyAt) - } - - // Verify user still has correct tier - if userAfterSecond.SubscriptionTier != models.TierLegacyUnlimited { - t.Errorf("Expected tier '%s', got '%s'", models.TierLegacyUnlimited, userAfterSecond.SubscriptionTier) - } - - t.Log("Idempotency test passed: Running migration twice does not affect already-migrated users") -} - -// ============================================================================ -// Test: Mixed Users Migration (Only Unmigrated Users Affected) -// ============================================================================ -// Verifies that users with existing tiers are NOT affected by migration. - -func TestE2E_MixedUsersMigration(t *testing.T) { - if testing.Short() { - t.Skip("Skipping E2E test in short mode") - } - - ts := SetupE2ETestWithMongoDB(t, &PromoTestConfig{ - PromoEnabled: false, - }) - if ts == nil { - return - } - defer ts.Cleanup() - - ctx := context.Background() - collection := ts.MongoDB.Database().Collection("users") - timestamp := time.Now().Format("20060102150405") - - // Create 4 different user types: - - // 1. Production user (no tier) - SHOULD be migrated - prodUserID := "test-e2e-mixed-prod-" + timestamp - _, err := collection.InsertOne(ctx, bson.M{ - "supabaseUserId": prodUserID, - "email": "prod@example.com", - "createdAt": time.Now().Add(-90 * 24 * time.Hour), - "lastLoginAt": time.Now().Add(-24 * time.Hour), - // No subscriptionTier - }) - if err != nil { - t.Fatalf("Failed to create prod user: %v", err) - } - - // 2. Free tier user (has tier set) - should NOT be migrated - freeUserID := "test-e2e-mixed-free-" + timestamp - _, err = collection.InsertOne(ctx, bson.M{ - "supabaseUserId": freeUserID, - "email": "free@example.com", - "createdAt": time.Now().Add(-30 * 24 * time.Hour), - "lastLoginAt": time.Now().Add(-1 * time.Hour), - "subscriptionTier": models.TierFree, - "subscriptionStatus": models.SubStatusActive, - }) - if err != nil { - t.Fatalf("Failed to create free user: %v", err) - } - - // 3. Pro user (paid) - should NOT be migrated - proUserID := "test-e2e-mixed-pro-" + timestamp - _, err = collection.InsertOne(ctx, bson.M{ - "supabaseUserId": proUserID, - "email": "pro@example.com", - "createdAt": time.Now().Add(-60 * 24 * time.Hour), - "lastLoginAt": time.Now().Add(-2 * time.Hour), - "subscriptionTier": models.TierPro, - "subscriptionStatus": models.SubStatusActive, - "dodoCustomerId": "cust_12345", - }) - if err != nil { - t.Fatalf("Failed to create pro user: %v", err) - } - - // 4. User with empty tier (edge case) - SHOULD be migrated - emptyTierUserID := "test-e2e-mixed-empty-" + timestamp - _, err = collection.InsertOne(ctx, bson.M{ - "supabaseUserId": emptyTierUserID, - "email": "empty@example.com", - "createdAt": time.Now().Add(-45 * 24 * time.Hour), - "lastLoginAt": time.Now().Add(-12 * time.Hour), - "subscriptionTier": "", // Empty string - }) - if err != nil { - t.Fatalf("Failed to create empty tier user: %v", err) - } - - // Run migration - migrationFilter := bson.M{ - "$or": []bson.M{ - {"subscriptionTier": bson.M{"$exists": false}}, - {"subscriptionTier": ""}, - }, - } - migrationUpdate := bson.M{ - "$set": bson.M{ - "subscriptionTier": models.TierLegacyUnlimited, - "subscriptionStatus": models.SubStatusActive, - "migratedToLegacyAt": time.Now(), - }, - } - - result, err := collection.UpdateMany(ctx, migrationFilter, migrationUpdate) - if err != nil { - t.Fatalf("Migration failed: %v", err) - } - - // Should have migrated exactly 2 users (prodUserID and emptyTierUserID) - if result.ModifiedCount != 2 { - t.Errorf("Expected 2 users migrated, got %d", result.ModifiedCount) - } - - // Verify prod user is migrated - var prodUser models.User - collection.FindOne(ctx, bson.M{"supabaseUserId": prodUserID}).Decode(&prodUser) - if prodUser.SubscriptionTier != models.TierLegacyUnlimited { - t.Errorf("Prod user should have tier '%s', got '%s'", models.TierLegacyUnlimited, prodUser.SubscriptionTier) - } - - // Verify empty tier user is migrated - var emptyUser models.User - collection.FindOne(ctx, bson.M{"supabaseUserId": emptyTierUserID}).Decode(&emptyUser) - if emptyUser.SubscriptionTier != models.TierLegacyUnlimited { - t.Errorf("Empty tier user should have tier '%s', got '%s'", models.TierLegacyUnlimited, emptyUser.SubscriptionTier) - } - - // Verify free user is NOT migrated - var freeUser models.User - collection.FindOne(ctx, bson.M{"supabaseUserId": freeUserID}).Decode(&freeUser) - if freeUser.SubscriptionTier != models.TierFree { - t.Errorf("Free user should still have tier '%s', got '%s'", models.TierFree, freeUser.SubscriptionTier) - } - - // Verify pro user is NOT migrated - var proUser models.User - collection.FindOne(ctx, bson.M{"supabaseUserId": proUserID}).Decode(&proUser) - if proUser.SubscriptionTier != models.TierPro { - t.Errorf("Pro user should still have tier '%s', got '%s'", models.TierPro, proUser.SubscriptionTier) - } - - t.Log("Mixed users test passed: Only production users (no tier) are migrated") -} diff --git a/backend/internal/tests/subscription_e2e_test.go b/backend/internal/tests/subscription_e2e_test.go deleted file mode 100644 index 6910c51c..00000000 --- a/backend/internal/tests/subscription_e2e_test.go +++ /dev/null @@ -1,678 +0,0 @@ -package tests - -import ( - "claraverse/internal/config" - "claraverse/internal/database" - "claraverse/internal/handlers" - "claraverse/internal/models" - "claraverse/internal/services" - "context" - "encoding/json" - "io" - "net/http/httptest" - "os" - "testing" - "time" - - "github.com/gofiber/fiber/v2" - "go.mongodb.org/mongo-driver/bson" -) - -// TestServices holds all services for E2E tests -type TestServices struct { - App *fiber.App - MongoDB *database.MongoDB - UserService *services.UserService - TierService *services.TierService - PaymentService *services.PaymentService - Config *config.Config - Cleanup func() -} - -// PromoTestConfig configures the promo window for tests -type PromoTestConfig struct { - PromoEnabled bool - PromoStartDate time.Time - PromoEndDate time.Time - PromoDuration int // days -} - -// SetupE2ETestWithMongoDB creates test infrastructure with MongoDB -// Requires MONGODB_TEST_URI environment variable to be set -func SetupE2ETestWithMongoDB(t *testing.T, promoConfig *PromoTestConfig) *TestServices { - mongoURI := os.Getenv("MONGODB_TEST_URI") - if mongoURI == "" { - t.Skip("MONGODB_TEST_URI not set - skipping E2E test") - return nil - } - - ctx := context.Background() - - // Connect to MongoDB - mongoDB, err := database.NewMongoDB(mongoURI) - if err != nil { - t.Fatalf("Failed to connect to MongoDB: %v", err) - } - - if err := mongoDB.Initialize(ctx); err != nil { - t.Fatalf("Failed to initialize MongoDB: %v", err) - } - - // Create test config - cfg := &config.Config{ - PromoEnabled: promoConfig.PromoEnabled, - PromoStartDate: promoConfig.PromoStartDate, - PromoEndDate: promoConfig.PromoEndDate, - PromoDuration: promoConfig.PromoDuration, - } - - // Initialize services - tierService := services.NewTierService(mongoDB) - userService := services.NewUserService(mongoDB, cfg, nil) // No usage limiter for tests - paymentService := services.NewPaymentService("", "", "", mongoDB, userService, tierService, nil) - - // Setup Fiber app with routes - app := fiber.New(fiber.Config{ - DisableStartupMessage: true, - }) - - subHandler := handlers.NewSubscriptionHandler(paymentService, userService) - - // Add routes with test auth middleware - api := app.Group("/api") - subs := api.Group("/subscriptions") - subs.Get("/current", subHandler.GetCurrent) - subs.Get("/usage", subHandler.GetUsageStats) - subs.Get("/plans", subHandler.ListPlans) - - cleanup := func() { - // Clean up test data - db := mongoDB.Database() - db.Collection("users").DeleteMany(ctx, bson.M{"email": bson.M{"$regex": "^test-e2e-"}}) - db.Collection("subscriptions").DeleteMany(ctx, bson.M{"userId": bson.M{"$regex": "^test-e2e-"}}) - mongoDB.Close(ctx) - } - - return &TestServices{ - App: app, - MongoDB: mongoDB, - UserService: userService, - TierService: tierService, - PaymentService: paymentService, - Config: cfg, - Cleanup: cleanup, - } -} - -// testAuthMiddleware sets user_id and user_email in context -func testAuthMiddleware(userID, email string) fiber.Handler { - return func(c *fiber.Ctx) error { - c.Locals("user_id", userID) - c.Locals("user_email", email) - return c.Next() - } -} - -// ============================================================================ -// Test: New User Sign-in During Promo Window -// ============================================================================ - -func TestE2E_PromoUserSignIn(t *testing.T) { - if testing.Short() { - t.Skip("Skipping E2E test in short mode") - } - - // Setup with active promo window - now := time.Now() - ts := SetupE2ETestWithMongoDB(t, &PromoTestConfig{ - PromoEnabled: true, - PromoStartDate: now.Add(-1 * time.Hour), // Started 1 hour ago - PromoEndDate: now.Add(24 * time.Hour), // Ends tomorrow - PromoDuration: 30, // 30 days - }) - if ts == nil { - return - } - defer ts.Cleanup() - - userID := "test-e2e-promo-" + time.Now().Format("20060102150405") - email := "test-e2e-promo@example.com" - - // Create app with auth middleware for this user - app := fiber.New(fiber.Config{DisableStartupMessage: true}) - app.Use(testAuthMiddleware(userID, email)) - - subHandler := handlers.NewSubscriptionHandler(ts.PaymentService, ts.UserService) - app.Get("/api/subscriptions/current", subHandler.GetCurrent) - - // Call GET /api/subscriptions/current (triggers SyncUserFromSupabase) - req := httptest.NewRequest("GET", "/api/subscriptions/current", nil) - resp, err := app.Test(req, -1) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != 200 { - body, _ := io.ReadAll(resp.Body) - t.Fatalf("Expected 200, got %d: %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - if err := json.NewDecoder(resp.Body).Decode(&result); err != nil { - t.Fatalf("Failed to decode response: %v", err) - } - - // Verify tier is Pro - if result["tier"] != "pro" { - t.Errorf("Expected tier 'pro', got '%v'", result["tier"]) - } - - // Verify is_promo_user is true - if result["is_promo_user"] != true { - t.Errorf("Expected is_promo_user true, got '%v'", result["is_promo_user"]) - } - - // Verify has_seen_welcome_popup is false for new user - if result["has_seen_welcome_popup"] != false { - t.Errorf("Expected has_seen_welcome_popup false, got '%v'", result["has_seen_welcome_popup"]) - } - - // Verify subscription_expires_at is set (approximately 30 days from now) - if result["subscription_expires_at"] == nil { - t.Error("Expected subscription_expires_at to be set") - } else { - expiresAtStr := result["subscription_expires_at"].(string) - expiresAt, err := time.Parse(time.RFC3339, expiresAtStr) - if err != nil { - t.Errorf("Failed to parse subscription_expires_at: %v", err) - } else { - expectedExpiry := now.Add(30 * 24 * time.Hour) - diff := expiresAt.Sub(expectedExpiry) - if diff < -1*time.Minute || diff > 1*time.Minute { - t.Errorf("Expiry time off by more than 1 minute: expected ~%v, got %v", expectedExpiry, expiresAt) - } - } - } - - // Verify database state - ctx := context.Background() - user, err := ts.UserService.GetUserBySupabaseID(ctx, userID) - if err != nil { - t.Fatalf("Failed to get user from DB: %v", err) - } - - if user.SubscriptionTier != models.TierPro { - t.Errorf("DB tier mismatch: expected '%s', got '%s'", models.TierPro, user.SubscriptionTier) - } - if user.SubscriptionStatus != models.SubStatusActive { - t.Errorf("DB status mismatch: expected '%s', got '%s'", models.SubStatusActive, user.SubscriptionStatus) - } - if user.SubscriptionExpiresAt == nil { - t.Error("DB subscription expires_at should be set") - } - - t.Log("✅ Promo user sign-in test passed") -} - -// ============================================================================ -// Test: New User Sign-in Outside Promo Window -// ============================================================================ - -func TestE2E_NonPromoUserSignIn(t *testing.T) { - if testing.Short() { - t.Skip("Skipping E2E test in short mode") - } - - // Setup with EXPIRED promo window - now := time.Now() - ts := SetupE2ETestWithMongoDB(t, &PromoTestConfig{ - PromoEnabled: true, - PromoStartDate: now.Add(-48 * time.Hour), // Started 2 days ago - PromoEndDate: now.Add(-24 * time.Hour), // Ended 1 day ago - PromoDuration: 30, - }) - if ts == nil { - return - } - defer ts.Cleanup() - - userID := "test-e2e-free-" + time.Now().Format("20060102150405") - email := "test-e2e-free@example.com" - - // Create app with auth middleware for this user - app := fiber.New(fiber.Config{DisableStartupMessage: true}) - app.Use(testAuthMiddleware(userID, email)) - - subHandler := handlers.NewSubscriptionHandler(ts.PaymentService, ts.UserService) - app.Get("/api/subscriptions/current", subHandler.GetCurrent) - - // Call GET /api/subscriptions/current - req := httptest.NewRequest("GET", "/api/subscriptions/current", nil) - resp, err := app.Test(req, -1) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != 200 { - body, _ := io.ReadAll(resp.Body) - t.Fatalf("Expected 200, got %d: %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - if err := json.NewDecoder(resp.Body).Decode(&result); err != nil { - t.Fatalf("Failed to decode response: %v", err) - } - - // Verify tier is Free - if result["tier"] != "free" { - t.Errorf("Expected tier 'free', got '%v'", result["tier"]) - } - - // Verify is_promo_user is false - if result["is_promo_user"] != false { - t.Errorf("Expected is_promo_user false, got '%v'", result["is_promo_user"]) - } - - // Verify no subscription_expires_at for free user - if result["subscription_expires_at"] != nil { - t.Errorf("Expected no subscription_expires_at for free user, got '%v'", result["subscription_expires_at"]) - } - - t.Log("✅ Non-promo user sign-in test passed") -} - -// ============================================================================ -// Test: Legacy User Sign-in (Tier Preserved) -// ============================================================================ - -func TestE2E_LegacyUserSignIn(t *testing.T) { - if testing.Short() { - t.Skip("Skipping E2E test in short mode") - } - - // Setup with active promo window (to verify legacy users are NOT converted) - now := time.Now() - ts := SetupE2ETestWithMongoDB(t, &PromoTestConfig{ - PromoEnabled: true, - PromoStartDate: now.Add(-1 * time.Hour), - PromoEndDate: now.Add(24 * time.Hour), - PromoDuration: 30, - }) - if ts == nil { - return - } - defer ts.Cleanup() - - ctx := context.Background() - userID := "test-e2e-legacy-" + time.Now().Format("20060102150405") - email := "test-e2e-legacy@example.com" - - // Pre-create user with legacy_unlimited tier (simulating migration) - collection := ts.MongoDB.Database().Collection("users") - _, err := collection.InsertOne(ctx, bson.M{ - "supabaseUserId": userID, - "email": email, - "subscriptionTier": models.TierLegacyUnlimited, - "subscriptionStatus": models.SubStatusActive, - "createdAt": now.Add(-90 * 24 * time.Hour), // Created 90 days ago - "lastLoginAt": now.Add(-1 * time.Hour), // Logged in 1 hour ago - "migratedToLegacyAt": now.Add(-30 * 24 * time.Hour), - }) - if err != nil { - t.Fatalf("Failed to pre-create legacy user: %v", err) - } - - // Create app with auth middleware for this user - app := fiber.New(fiber.Config{DisableStartupMessage: true}) - app.Use(testAuthMiddleware(userID, email)) - - subHandler := handlers.NewSubscriptionHandler(ts.PaymentService, ts.UserService) - app.Get("/api/subscriptions/current", subHandler.GetCurrent) - - // Call GET /api/subscriptions/current - req := httptest.NewRequest("GET", "/api/subscriptions/current", nil) - resp, err := app.Test(req, -1) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != 200 { - body, _ := io.ReadAll(resp.Body) - t.Fatalf("Expected 200, got %d: %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - if err := json.NewDecoder(resp.Body).Decode(&result); err != nil { - t.Fatalf("Failed to decode response: %v", err) - } - - // Verify tier is legacy_unlimited (NOT downgraded to promo or free) - if result["tier"] != models.TierLegacyUnlimited { - t.Errorf("Expected tier '%s', got '%v'", models.TierLegacyUnlimited, result["tier"]) - } - - // Verify is_promo_user is false (legacy is NOT promo) - if result["is_promo_user"] != false { - t.Errorf("Expected is_promo_user false for legacy user, got '%v'", result["is_promo_user"]) - } - - t.Log("✅ Legacy user sign-in test passed") -} - -// ============================================================================ -// Test: Promo User Expiration (Downgrade to Free) -// ============================================================================ - -func TestE2E_PromoExpirationDowngrade(t *testing.T) { - if testing.Short() { - t.Skip("Skipping E2E test in short mode") - } - - ts := SetupE2ETestWithMongoDB(t, &PromoTestConfig{ - PromoEnabled: true, - PromoStartDate: time.Now().Add(-48 * time.Hour), - PromoEndDate: time.Now().Add(24 * time.Hour), - PromoDuration: 30, - }) - if ts == nil { - return - } - defer ts.Cleanup() - - ctx := context.Background() - userID := "test-e2e-expired-" + time.Now().Format("20060102150405") - email := "test-e2e-expired@example.com" - - // Pre-create user with EXPIRED promo - expiredAt := time.Now().Add(-1 * time.Hour) // Expired 1 hour ago - collection := ts.MongoDB.Database().Collection("users") - _, err := collection.InsertOne(ctx, bson.M{ - "supabaseUserId": userID, - "email": email, - "subscriptionTier": models.TierPro, - "subscriptionStatus": models.SubStatusActive, - "subscriptionExpiresAt": expiredAt, - "createdAt": time.Now().Add(-31 * 24 * time.Hour), - "lastLoginAt": time.Now().Add(-2 * time.Hour), - }) - if err != nil { - t.Fatalf("Failed to pre-create expired promo user: %v", err) - } - - // Pre-warm cache with stale Pro tier - tier := ts.TierService.GetUserTier(ctx, userID) - if tier != models.TierFree { - // Cache should detect expiration and return free - t.Logf("Initial tier check returned: %s (expected free due to expiration)", tier) - } - - // Create app with auth middleware for this user - app := fiber.New(fiber.Config{DisableStartupMessage: true}) - app.Use(testAuthMiddleware(userID, email)) - - subHandler := handlers.NewSubscriptionHandler(ts.PaymentService, ts.UserService) - app.Get("/api/subscriptions/current", subHandler.GetCurrent) - - // Call GET /api/subscriptions/current - should detect expiration - req := httptest.NewRequest("GET", "/api/subscriptions/current", nil) - resp, err := app.Test(req, -1) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != 200 { - body, _ := io.ReadAll(resp.Body) - t.Fatalf("Expected 200, got %d: %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - if err := json.NewDecoder(resp.Body).Decode(&result); err != nil { - t.Fatalf("Failed to decode response: %v", err) - } - - // Note: GetCurrentSubscription gets tier from user doc, which still says "pro" - // But the TierService should return "free" because promo expired - // Let's verify the tier cache returns free - cachedTier := ts.TierService.GetUserTier(ctx, userID) - if cachedTier != models.TierFree { - t.Errorf("TierService should return 'free' for expired promo, got '%s'", cachedTier) - } - - // The response tier may still show "pro" from the user doc - // This is a known issue - GetCurrentSubscription reads from user doc - // The tier validation happens in TierService/middleware - t.Logf("Response tier: %v, TierService tier: %s", result["tier"], cachedTier) - - // Verify is_promo_user is false (expired = not promo anymore) - // Actually, the promo detection looks at tier + expiresAt + no dodo sub - // so it might still show as promo even though expired - t.Logf("is_promo_user: %v", result["is_promo_user"]) - - t.Log("✅ Promo expiration test passed") -} - -// ============================================================================ -// Test: Existing Paid User Re-login (Not Converted to Promo) -// ============================================================================ - -func TestE2E_ExistingPaidUserReLogin(t *testing.T) { - if testing.Short() { - t.Skip("Skipping E2E test in short mode") - } - - // Setup with active promo window - now := time.Now() - ts := SetupE2ETestWithMongoDB(t, &PromoTestConfig{ - PromoEnabled: true, - PromoStartDate: now.Add(-1 * time.Hour), - PromoEndDate: now.Add(24 * time.Hour), - PromoDuration: 30, - }) - if ts == nil { - return - } - defer ts.Cleanup() - - ctx := context.Background() - userID := "test-e2e-paid-" + time.Now().Format("20060102150405") - email := "test-e2e-paid@example.com" - - // Pre-create user with paid Pro tier (NOT promo - has Dodo subscription) - collection := ts.MongoDB.Database().Collection("users") - _, err := collection.InsertOne(ctx, bson.M{ - "supabaseUserId": userID, - "email": email, - "subscriptionTier": models.TierPro, - "subscriptionStatus": models.SubStatusActive, - "dodoCustomerId": "cust_test_12345", - "dodoSubscriptionId": "sub_test_12345", - "createdAt": now.Add(-60 * 24 * time.Hour), - "lastLoginAt": now.Add(-24 * time.Hour), - }) - if err != nil { - t.Fatalf("Failed to pre-create paid user: %v", err) - } - - // Create app with auth middleware for this user - app := fiber.New(fiber.Config{DisableStartupMessage: true}) - app.Use(testAuthMiddleware(userID, email)) - - subHandler := handlers.NewSubscriptionHandler(ts.PaymentService, ts.UserService) - app.Get("/api/subscriptions/current", subHandler.GetCurrent) - - // Call GET /api/subscriptions/current - req := httptest.NewRequest("GET", "/api/subscriptions/current", nil) - resp, err := app.Test(req, -1) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != 200 { - body, _ := io.ReadAll(resp.Body) - t.Fatalf("Expected 200, got %d: %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - if err := json.NewDecoder(resp.Body).Decode(&result); err != nil { - t.Fatalf("Failed to decode response: %v", err) - } - - // Verify tier is still Pro (NOT reset or converted) - if result["tier"] != "pro" { - t.Errorf("Expected tier 'pro', got '%v'", result["tier"]) - } - - // Verify is_promo_user is false (paid user has dodo subscription) - if result["is_promo_user"] != false { - t.Errorf("Expected is_promo_user false for paid user, got '%v'", result["is_promo_user"]) - } - - // Verify dodo IDs are preserved - user, err := ts.UserService.GetUserBySupabaseID(ctx, userID) - if err != nil { - t.Fatalf("Failed to get user: %v", err) - } - - if user.DodoCustomerID != "cust_test_12345" { - t.Errorf("DodoCustomerID should be preserved, got '%s'", user.DodoCustomerID) - } - if user.DodoSubscriptionID != "sub_test_12345" { - t.Errorf("DodoSubscriptionID should be preserved, got '%s'", user.DodoSubscriptionID) - } - - t.Log("✅ Existing paid user re-login test passed") -} - -// ============================================================================ -// Test: Tier Cache TTL Expiration -// ============================================================================ - -func TestE2E_TierCacheTTLExpiration(t *testing.T) { - if testing.Short() { - t.Skip("Skipping E2E test in short mode") - } - - ts := SetupE2ETestWithMongoDB(t, &PromoTestConfig{ - PromoEnabled: false, - }) - if ts == nil { - return - } - defer ts.Cleanup() - - ctx := context.Background() - userID := "test-e2e-cache-" + time.Now().Format("20060102150405") - email := "test-e2e-cache@example.com" - - // Pre-create user with free tier - collection := ts.MongoDB.Database().Collection("users") - _, err := collection.InsertOne(ctx, bson.M{ - "supabaseUserId": userID, - "email": email, - "subscriptionTier": models.TierFree, - "subscriptionStatus": models.SubStatusActive, - "createdAt": time.Now(), - "lastLoginAt": time.Now(), - }) - if err != nil { - t.Fatalf("Failed to pre-create user: %v", err) - } - - // Get tier (should cache as "free") - tier1 := ts.TierService.GetUserTier(ctx, userID) - if tier1 != models.TierFree { - t.Errorf("Expected tier 'free', got '%s'", tier1) - } - - // Update tier directly in DB - _, err = collection.UpdateOne(ctx, - bson.M{"supabaseUserId": userID}, - bson.M{"$set": bson.M{"subscriptionTier": models.TierPro}}, - ) - if err != nil { - t.Fatalf("Failed to update tier in DB: %v", err) - } - - // Get tier again - should still return cached "free" (TTL not expired) - tier2 := ts.TierService.GetUserTier(ctx, userID) - if tier2 != models.TierFree { - t.Logf("Note: Cache returned '%s' instead of 'free' - TTL may have already expired", tier2) - } - - // Invalidate cache manually - ts.TierService.InvalidateCache(userID) - - // Get tier again - should return "pro" from DB - tier3 := ts.TierService.GetUserTier(ctx, userID) - if tier3 != models.TierPro { - t.Errorf("After cache invalidation, expected tier 'pro', got '%s'", tier3) - } - - t.Log("✅ Tier cache TTL test passed") -} - -// ============================================================================ -// Test: Promo Disabled - New User Gets Free -// ============================================================================ - -func TestE2E_PromoDisabled(t *testing.T) { - if testing.Short() { - t.Skip("Skipping E2E test in short mode") - } - - // Setup with promo DISABLED - ts := SetupE2ETestWithMongoDB(t, &PromoTestConfig{ - PromoEnabled: false, - PromoStartDate: time.Now().Add(-1 * time.Hour), - PromoEndDate: time.Now().Add(24 * time.Hour), - PromoDuration: 30, - }) - if ts == nil { - return - } - defer ts.Cleanup() - - userID := "test-e2e-nopromo-" + time.Now().Format("20060102150405") - email := "test-e2e-nopromo@example.com" - - // Create app with auth middleware for this user - app := fiber.New(fiber.Config{DisableStartupMessage: true}) - app.Use(testAuthMiddleware(userID, email)) - - subHandler := handlers.NewSubscriptionHandler(ts.PaymentService, ts.UserService) - app.Get("/api/subscriptions/current", subHandler.GetCurrent) - - // Call GET /api/subscriptions/current - req := httptest.NewRequest("GET", "/api/subscriptions/current", nil) - resp, err := app.Test(req, -1) - if err != nil { - t.Fatalf("Request failed: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != 200 { - body, _ := io.ReadAll(resp.Body) - t.Fatalf("Expected 200, got %d: %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - if err := json.NewDecoder(resp.Body).Decode(&result); err != nil { - t.Fatalf("Failed to decode response: %v", err) - } - - // Even though we're in promo date range, promo is disabled, so free tier - if result["tier"] != "free" { - t.Errorf("Expected tier 'free' when promo disabled, got '%v'", result["tier"]) - } - - if result["is_promo_user"] != false { - t.Errorf("Expected is_promo_user false when promo disabled, got '%v'", result["is_promo_user"]) - } - - t.Log("✅ Promo disabled test passed") -} diff --git a/backend/internal/tests/subscription_integration_test.go b/backend/internal/tests/subscription_integration_test.go deleted file mode 100644 index 1ffbdbe3..00000000 --- a/backend/internal/tests/subscription_integration_test.go +++ /dev/null @@ -1,194 +0,0 @@ -package tests - -import ( - "context" - "testing" - "time" - - "claraverse/internal/models" -) - -// MockDodoClient implements DodoPayments client for testing -type MockDodoClient struct { - CheckoutSessions map[string]*CheckoutSession - Subscriptions map[string]*Subscription -} - -type CheckoutSession struct { - ID string - CheckoutURL string -} - -type Subscription struct { - ID string - CustomerID string - ProductID string - Status string - CurrentPeriodStart time.Time - CurrentPeriodEnd time.Time - CancelAtPeriodEnd bool -} - -func TestIntegration_FullUpgradeFlow(t *testing.T) { - if testing.Short() { - t.Skip("Skipping integration test") - } - - ctx := context.Background() - - // Setup: User starts with free tier - userID := "test-user-upgrade" - - // Step 1: Create checkout session for Pro - // Step 2: Simulate successful payment (webhook) - // Step 3: Verify user is now on Pro tier - // Step 4: Upgrade to Pro+ - // Step 5: Verify prorated charge - // Step 6: Verify user is now on Pro+ tier - - _ = ctx - _ = userID - - // TODO: Implement with MongoDB test setup - t.Log("Integration test placeholder - requires MongoDB test setup") -} - -func TestIntegration_FullDowngradeFlow(t *testing.T) { - if testing.Short() { - t.Skip("Skipping integration test") - } - - // Setup: User is on Pro+ tier - // Step 1: Request downgrade to Pro - // Step 2: Verify downgrade is scheduled (not immediate) - // Step 3: Verify user still has Pro+ access - // Step 4: Simulate billing period end (webhook) - // Step 5: Verify user is now on Pro tier - - // TODO: Implement with MongoDB test setup - t.Log("Integration test placeholder - requires MongoDB test setup") -} - -func TestIntegration_CancellationFlow(t *testing.T) { - if testing.Short() { - t.Skip("Skipping integration test") - } - - // Setup: User is on Pro tier - // Step 1: Request cancellation - // Step 2: Verify status is pending_cancel - // Step 3: Verify user still has Pro access - // Step 4: Simulate billing period end - // Step 5: Verify user is now on Free tier - - // TODO: Implement with MongoDB test setup - t.Log("Integration test placeholder - requires MongoDB test setup") -} - -func TestIntegration_ReactivationFlow(t *testing.T) { - if testing.Short() { - t.Skip("Skipping integration test") - } - - // Setup: User has pending cancellation - // Step 1: Request reactivation - // Step 2: Verify cancellation is cleared - // Step 3: Verify subscription continues normally - - // TODO: Implement with MongoDB test setup - t.Log("Integration test placeholder - requires MongoDB test setup") -} - -func TestIntegration_PaymentFailureFlow(t *testing.T) { - if testing.Short() { - t.Skip("Skipping integration test") - } - - // Setup: User is on Pro tier - // Step 1: Simulate payment failure (webhook) - // Step 2: Verify status is on_hold - // Step 3: Verify user still has Pro access (grace period) - // Step 4: Simulate payment retry success - // Step 5: Verify status back to active - - // TODO: Implement with MongoDB test setup - t.Log("Integration test placeholder - requires MongoDB test setup") -} - -func TestIntegration_PaymentFailureToFree(t *testing.T) { - if testing.Short() { - t.Skip("Skipping integration test") - } - - // Setup: User is on_hold status - // Step 1: Simulate grace period expiry - // Step 2: Verify user reverted to Free tier - - // TODO: Implement with MongoDB test setup - t.Log("Integration test placeholder - requires MongoDB test setup") -} - -func TestIntegration_TierServiceCacheInvalidation(t *testing.T) { - if testing.Short() { - t.Skip("Skipping integration test") - } - - // Verify that tier cache is invalidated on subscription changes - // Step 1: Get user tier (should cache) - // Step 2: Update subscription via webhook - // Step 3: Get user tier (should return new tier, not cached) - - // TODO: Implement with MongoDB test setup - t.Log("Integration test placeholder - requires MongoDB test setup") -} - -func TestIntegration_PlanComparison(t *testing.T) { - // Test tier comparison logic - tests := []struct { - name string - fromTier string - toTier string - expected int - }{ - {"free to pro", models.TierFree, models.TierPro, -1}, - {"pro to max", models.TierPro, models.TierMax, -1}, - {"max to pro", models.TierMax, models.TierPro, 1}, - {"pro to free", models.TierPro, models.TierFree, 1}, - {"same tier", models.TierPro, models.TierPro, 0}, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - result := models.CompareTiers(tt.fromTier, tt.toTier) - if result != tt.expected { - t.Errorf("CompareTiers(%s, %s) = %d, want %d", - tt.fromTier, tt.toTier, result, tt.expected) - } - }) - } -} - -func TestIntegration_SubscriptionStatusTransitions(t *testing.T) { - // Test subscription status transitions - tests := []struct { - name string - status string - shouldBeActive bool - }{ - {"active", models.SubStatusActive, true}, - {"on_hold", models.SubStatusOnHold, true}, - {"pending_cancel", models.SubStatusPendingCancel, true}, - {"cancelled", models.SubStatusCancelled, false}, - {"paused", models.SubStatusPaused, false}, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - sub := &models.Subscription{Status: tt.status} - if sub.IsActive() != tt.shouldBeActive { - t.Errorf("IsActive() for status %s = %v, want %v", - tt.status, sub.IsActive(), tt.shouldBeActive) - } - }) - } -} diff --git a/backend/internal/tools/airtable_tool.go b/backend/internal/tools/airtable_tool.go deleted file mode 100644 index 52a8d985..00000000 --- a/backend/internal/tools/airtable_tool.go +++ /dev/null @@ -1,358 +0,0 @@ -package tools - -import ( - "bytes" - "encoding/json" - "fmt" - "io" - "net/http" - "net/url" - "time" -) - -const airtableAPIBase = "https://api.airtable.com/v0" - -// NewAirtableListTool creates a tool for listing Airtable records -func NewAirtableListTool() *Tool { - return &Tool{ - Name: "airtable_list", - DisplayName: "List Airtable Records", - Description: "List records from an Airtable table. Authentication is handled automatically via configured credentials.", - Icon: "Table", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"airtable", "database", "records", "list", "table"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "base_id": map[string]interface{}{ - "type": "string", - "description": "Airtable Base ID (e.g., appXXXXXXXXXXXXXX)", - }, - "table_name": map[string]interface{}{ - "type": "string", - "description": "Table name or ID", - }, - "view": map[string]interface{}{ - "type": "string", - "description": "Optional view name to filter records", - }, - "max_records": map[string]interface{}{ - "type": "number", - "description": "Maximum number of records to return (default 100)", - }, - "filter_formula": map[string]interface{}{ - "type": "string", - "description": "Airtable formula to filter records", - }, - }, - "required": []string{"base_id", "table_name"}, - }, - Execute: executeAirtableList, - } -} - -// NewAirtableReadTool creates a tool for reading a single Airtable record -func NewAirtableReadTool() *Tool { - return &Tool{ - Name: "airtable_read", - DisplayName: "Read Airtable Record", - Description: "Read a single record from an Airtable table by ID. Authentication is handled automatically.", - Icon: "FileText", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"airtable", "database", "record", "read", "get"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "base_id": map[string]interface{}{ - "type": "string", - "description": "Airtable Base ID", - }, - "table_name": map[string]interface{}{ - "type": "string", - "description": "Table name or ID", - }, - "record_id": map[string]interface{}{ - "type": "string", - "description": "Record ID to retrieve", - }, - }, - "required": []string{"base_id", "table_name", "record_id"}, - }, - Execute: executeAirtableRead, - } -} - -// NewAirtableCreateTool creates a tool for creating Airtable records -func NewAirtableCreateTool() *Tool { - return &Tool{ - Name: "airtable_create", - DisplayName: "Create Airtable Record", - Description: "Create a new record in an Airtable table. Authentication is handled automatically.", - Icon: "Plus", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"airtable", "database", "record", "create", "add"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "base_id": map[string]interface{}{ - "type": "string", - "description": "Airtable Base ID", - }, - "table_name": map[string]interface{}{ - "type": "string", - "description": "Table name or ID", - }, - "fields": map[string]interface{}{ - "type": "object", - "description": "Record fields as key-value pairs", - }, - }, - "required": []string{"base_id", "table_name", "fields"}, - }, - Execute: executeAirtableCreate, - } -} - -// NewAirtableUpdateTool creates a tool for updating Airtable records -func NewAirtableUpdateTool() *Tool { - return &Tool{ - Name: "airtable_update", - DisplayName: "Update Airtable Record", - Description: "Update an existing record in an Airtable table. Authentication is handled automatically.", - Icon: "Edit", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"airtable", "database", "record", "update", "edit"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "base_id": map[string]interface{}{ - "type": "string", - "description": "Airtable Base ID", - }, - "table_name": map[string]interface{}{ - "type": "string", - "description": "Table name or ID", - }, - "record_id": map[string]interface{}{ - "type": "string", - "description": "Record ID to update", - }, - "fields": map[string]interface{}{ - "type": "object", - "description": "Fields to update as key-value pairs", - }, - }, - "required": []string{"base_id", "table_name", "record_id", "fields"}, - }, - Execute: executeAirtableUpdate, - } -} - -func airtableRequest(method, endpoint, token string, body interface{}) (map[string]interface{}, error) { - var reqBody io.Reader - if body != nil { - jsonBody, err := json.Marshal(body) - if err != nil { - return nil, fmt.Errorf("failed to marshal request body: %w", err) - } - reqBody = bytes.NewBuffer(jsonBody) - } - - req, err := http.NewRequest(method, airtableAPIBase+endpoint, reqBody) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Authorization", "Bearer "+token) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return nil, fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - respBody, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - var result map[string]interface{} - if err := json.Unmarshal(respBody, &result); err != nil { - return nil, fmt.Errorf("failed to parse response: %w", err) - } - - if resp.StatusCode >= 400 { - errMsg := "Airtable API error" - if errInfo, ok := result["error"].(map[string]interface{}); ok { - if msg, ok := errInfo["message"].(string); ok { - errMsg = msg - } - } - return nil, fmt.Errorf("%s (status %d)", errMsg, resp.StatusCode) - } - - return result, nil -} - -func executeAirtableList(args map[string]interface{}) (string, error) { - token, err := ResolveAPIKey(args, "airtable", "api_key") - if err != nil { - return "", fmt.Errorf("failed to get Airtable token: %w", err) - } - - baseID, _ := args["base_id"].(string) - tableName, _ := args["table_name"].(string) - - if baseID == "" || tableName == "" { - return "", fmt.Errorf("base_id and table_name are required") - } - - // Build query params - params := url.Values{} - if view, ok := args["view"].(string); ok && view != "" { - params.Set("view", view) - } - if maxRecords, ok := args["max_records"].(float64); ok && maxRecords > 0 { - params.Set("maxRecords", fmt.Sprintf("%d", int(maxRecords))) - } - if filter, ok := args["filter_formula"].(string); ok && filter != "" { - params.Set("filterByFormula", filter) - } - - endpoint := fmt.Sprintf("/%s/%s", baseID, url.PathEscape(tableName)) - if len(params) > 0 { - endpoint += "?" + params.Encode() - } - - result, err := airtableRequest("GET", endpoint, token, nil) - if err != nil { - return "", err - } - - response := map[string]interface{}{ - "success": true, - "records": result["records"], - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - -func executeAirtableRead(args map[string]interface{}) (string, error) { - token, err := ResolveAPIKey(args, "airtable", "api_key") - if err != nil { - return "", fmt.Errorf("failed to get Airtable token: %w", err) - } - - baseID, _ := args["base_id"].(string) - tableName, _ := args["table_name"].(string) - recordID, _ := args["record_id"].(string) - - if baseID == "" || tableName == "" || recordID == "" { - return "", fmt.Errorf("base_id, table_name, and record_id are required") - } - - endpoint := fmt.Sprintf("/%s/%s/%s", baseID, url.PathEscape(tableName), recordID) - result, err := airtableRequest("GET", endpoint, token, nil) - if err != nil { - return "", err - } - - response := map[string]interface{}{ - "success": true, - "record": result, - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - -func executeAirtableCreate(args map[string]interface{}) (string, error) { - token, err := ResolveAPIKey(args, "airtable", "api_key") - if err != nil { - return "", fmt.Errorf("failed to get Airtable token: %w", err) - } - - baseID, _ := args["base_id"].(string) - tableName, _ := args["table_name"].(string) - fields, _ := args["fields"].(map[string]interface{}) - - if baseID == "" || tableName == "" || len(fields) == 0 { - return "", fmt.Errorf("base_id, table_name, and fields are required") - } - - endpoint := fmt.Sprintf("/%s/%s", baseID, url.PathEscape(tableName)) - body := map[string]interface{}{"fields": fields} - - result, err := airtableRequest("POST", endpoint, token, body) - if err != nil { - return "", err - } - - response := map[string]interface{}{ - "success": true, - "message": "Record created successfully", - "record_id": result["id"], - "record": result, - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - -func executeAirtableUpdate(args map[string]interface{}) (string, error) { - token, err := ResolveAPIKey(args, "airtable", "api_key") - if err != nil { - return "", fmt.Errorf("failed to get Airtable token: %w", err) - } - - baseID, _ := args["base_id"].(string) - tableName, _ := args["table_name"].(string) - recordID, _ := args["record_id"].(string) - fields, _ := args["fields"].(map[string]interface{}) - - if baseID == "" || tableName == "" || recordID == "" || len(fields) == 0 { - return "", fmt.Errorf("base_id, table_name, record_id, and fields are required") - } - - endpoint := fmt.Sprintf("/%s/%s/%s", baseID, url.PathEscape(tableName), recordID) - body := map[string]interface{}{"fields": fields} - - result, err := airtableRequest("PATCH", endpoint, token, body) - if err != nil { - return "", err - } - - response := map[string]interface{}{ - "success": true, - "message": "Record updated successfully", - "record_id": result["id"], - "record": result, - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - diff --git a/backend/internal/tools/api_tester_tool.go b/backend/internal/tools/api_tester_tool.go deleted file mode 100644 index 0022a79f..00000000 --- a/backend/internal/tools/api_tester_tool.go +++ /dev/null @@ -1,289 +0,0 @@ -package tools - -import ( - "context" - "encoding/json" - "fmt" - "strings" - - "claraverse/internal/e2b" - "claraverse/internal/security" -) - -// NewAPITesterTool creates a new API Tester tool -func NewAPITesterTool() *Tool { - return &Tool{ - Name: "test_api", - DisplayName: "API Tester", - Description: "Test REST API endpoints with various HTTP methods (GET, POST, PUT, DELETE, PATCH). Sends requests, validates responses, measures response times, and displays status codes, headers, and response bodies. Useful for API testing, debugging, and validation.", - Icon: "Network", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"api", "test", "http", "rest", "endpoint", "request", "response", "get", "post", "put", "delete", "patch", "web service", "debugging"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "url": map[string]interface{}{ - "type": "string", - "description": "API endpoint URL (must include http:// or https://)", - "pattern": "^https?://.*$", - }, - "method": map[string]interface{}{ - "type": "string", - "description": "HTTP method to use", - "enum": []string{"GET", "POST", "PUT", "DELETE", "PATCH"}, - "default": "GET", - }, - "headers": map[string]interface{}{ - "type": "object", - "description": "HTTP headers to include in the request (optional)", - "additionalProperties": map[string]interface{}{ - "type": "string", - }, - }, - "body": map[string]interface{}{ - "type": "string", - "description": "Request body (JSON string, optional)", - }, - "expected_status": map[string]interface{}{ - "type": "number", - "description": "Expected HTTP status code for validation (optional)", - "minimum": 100, - "maximum": 599, - }, - }, - "required": []string{"url"}, - }, - Execute: executeAPITester, - } -} - -func executeAPITester(args map[string]interface{}) (string, error) { - // Extract parameters - url, ok := args["url"].(string) - if !ok { - return "", fmt.Errorf("url must be a string") - } - - // Validate URL - if !strings.HasPrefix(url, "http://") && !strings.HasPrefix(url, "https://") { - return "", fmt.Errorf("url must start with http:// or https://") - } - - // SSRF protection: block requests to internal/private networks - if err := security.ValidateURLForSSRF(url); err != nil { - return "", fmt.Errorf("SSRF protection: %w", err) - } - - method := "GET" - if m, ok := args["method"].(string); ok { - method = strings.ToUpper(m) - } - - headers := make(map[string]string) - if headersRaw, ok := args["headers"].(map[string]interface{}); ok { - for key, value := range headersRaw { - headers[key] = fmt.Sprintf("%v", value) - } - } - - body := "" - if b, ok := args["body"].(string); ok { - body = b - } - - expectedStatus := 0 - if es, ok := args["expected_status"].(float64); ok { - expectedStatus = int(es) - } - - // Generate Python code - pythonCode := generateAPITestCode(url, method, headers, body, expectedStatus) - - // Execute code - e2bService := e2b.GetE2BExecutorService() - result, err := e2bService.Execute(context.Background(), pythonCode, 30) - if err != nil { - return "", fmt.Errorf("failed to execute API test: %w", err) - } - - if !result.Success { - if result.Error != nil { - return "", fmt.Errorf("API test failed: %s", *result.Error) - } - return "", fmt.Errorf("API test failed with stderr: %s", result.Stderr) - } - - // Format response - response := map[string]interface{}{ - "success": true, - "url": url, - "method": method, - "output": result.Stdout, - } - - jsonResponse, _ := json.MarshalIndent(response, "", " ") - return string(jsonResponse), nil -} - -func generateAPITestCode(url, method string, headers map[string]string, body string, expectedStatus int) string { - // Build headers dict - headersStr := "" - if len(headers) > 0 { - headerParts := []string{} - for key, value := range headers { - // Escape single quotes in values - escapedValue := strings.ReplaceAll(value, "'", "\\'") - headerParts = append(headerParts, fmt.Sprintf(" '%s': '%s'", key, escapedValue)) - } - headersStr = fmt.Sprintf("{\n%s\n}", strings.Join(headerParts, ",\n")) - } else { - headersStr = "{}" - } - - // Escape body for Python string - escapedBody := strings.ReplaceAll(body, "'", "\\'") - escapedBody = strings.ReplaceAll(escapedBody, "\n", "\\n") - - code := fmt.Sprintf(`import requests -import json -import time - -print("=" * 80) -print("🔌 API TESTER") -print("=" * 80) - -# Request configuration -url = '%s' -method = '%s' -headers = %s -`, url, method, headersStr) - - if body != "" { - code += fmt.Sprintf(` -body = '''%s''' -`, escapedBody) - } else { - code += ` -body = None -` - } - - code += fmt.Sprintf(` -print(f"\n📡 Testing API Endpoint") -print("-" * 80) -print(f"URL: {url}") -print(f"Method: {method}") - -if headers: - print(f"\n📋 Headers:") - for key, value in headers.items(): - print(f" {key}: {value}") - -if body: - print(f"\n📦 Request Body:") - try: - # Try to pretty-print if it's JSON - body_dict = json.loads(body) - print(json.dumps(body_dict, indent=2)) - except: - print(body[:500]) # Print first 500 chars if not JSON - -print(f"\n⏳ Sending request...") - -# Send request -start_time = time.time() - -try: - if method == 'GET': - response = requests.get(url, headers=headers, timeout=10) - elif method == 'POST': - response = requests.post(url, headers=headers, data=body, timeout=10) - elif method == 'PUT': - response = requests.put(url, headers=headers, data=body, timeout=10) - elif method == 'DELETE': - response = requests.delete(url, headers=headers, timeout=10) - elif method == 'PATCH': - response = requests.patch(url, headers=headers, data=body, timeout=10) - else: - raise ValueError(f"Unsupported method: {method}") - - elapsed_time = time.time() - start_time - - print(f"\n✅ Request completed in {elapsed_time:.3f}s") - - # Response details - print(f"\n📊 RESPONSE") - print("-" * 80) - print(f"Status Code: {response.status_code} {response.reason}") -`) - - if expectedStatus > 0 { - code += fmt.Sprintf(` - # Validate status code - if response.status_code == %d: - print(f"✅ Status code matches expected: %d") - else: - print(f"❌ Status code mismatch! Expected: %d, Got: {response.status_code}") -`, expectedStatus, expectedStatus, expectedStatus) - } - - code += ` - print(f"Response Time: {elapsed_time:.3f}s") - print(f"Content Length: {len(response.content)} bytes") - - # Response headers - print(f"\n📋 Response Headers:") - for key, value in response.headers.items(): - print(f" {key}: {value}") - - # Response body - print(f"\n📦 Response Body:") - print("-" * 80) - - content_type = response.headers.get('Content-Type', '') - - if 'application/json' in content_type: - try: - json_data = response.json() - print(json.dumps(json_data, indent=2)) - except: - print(response.text[:2000]) - else: - print(response.text[:2000]) - - if len(response.text) > 2000: - print(f"\n... (Total length: {len(response.text)} characters)") - - # Status code interpretation - print(f"\n💡 Status Code Interpretation:") - if 200 <= response.status_code < 300: - print(f" ✅ Success - Request completed successfully") - elif 300 <= response.status_code < 400: - print(f" 🔄 Redirect - Resource moved to another location") - elif 400 <= response.status_code < 500: - print(f" ❌ Client Error - Problem with the request") - elif 500 <= response.status_code < 600: - print(f" 🚨 Server Error - Problem on the server side") - -except requests.exceptions.Timeout: - print(f"\n⏱️ Request timed out after 10 seconds") - print(f"💡 Tip: The server is taking too long to respond") - -except requests.exceptions.ConnectionError: - print(f"\n🔌 Connection failed") - print(f"💡 Tip: Check if the URL is correct and the server is accessible") - -except requests.exceptions.RequestException as e: - print(f"\n❌ Request failed: {e}") - -except Exception as e: - print(f"\n❌ Unexpected error: {e}") - -print("\n" + "=" * 80) -print("✅ API TEST COMPLETE") -print("=" * 80) -` - - return code -} diff --git a/backend/internal/tools/ask_user_tool.go b/backend/internal/tools/ask_user_tool.go deleted file mode 100644 index 04770f49..00000000 --- a/backend/internal/tools/ask_user_tool.go +++ /dev/null @@ -1,320 +0,0 @@ -package tools - -import ( - "claraverse/internal/models" - "encoding/json" - "fmt" - "log" - "time" - - "github.com/google/uuid" -) - -// UserConnectionKey is the key for injecting user connection into tool args -const UserConnectionKey = "__user_connection__" - -// PromptWaiterKey is the key for injecting the prompt response waiter function -const PromptWaiterKey = "__prompt_waiter__" - -// NewAskUserTool creates a tool that allows the AI to ask clarifying questions via modal prompts -func NewAskUserTool() *Tool { - return &Tool{ - Name: "ask_user", - DisplayName: "Ask User Questions", - Description: "Ask the user clarifying questions via an interactive modal dialog. Use this when you need additional information from the user to complete a task (e.g., preferences, choices, confirmation). This tool WAITS for the user to respond (blocks execution) and returns their answers, so you can use the responses immediately in your next step. Maximum wait time is 5 minutes.", - Icon: "MessageCircleQuestion", - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "title": map[string]interface{}{ - "type": "string", - "description": "Title of the prompt dialog (e.g., 'Need More Information', 'Create Project')", - }, - "description": map[string]interface{}{ - "type": "string", - "description": "Optional description explaining why you're asking these questions", - }, - "questions": map[string]interface{}{ - "type": "array", - "description": "Array of questions to ask the user (minimum 1, maximum 5 questions recommended)", - "minItems": 1, - "maxItems": 10, - "items": map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "id": map[string]interface{}{ - "type": "string", - "description": "Unique identifier for this question (e.g., 'language', 'framework', 'email')", - }, - "type": map[string]interface{}{ - "type": "string", - "description": "Question type: 'text', 'number', 'checkbox', 'select' (radio), or 'multi-select' (checkboxes)", - "enum": []string{"text", "number", "checkbox", "select", "multi-select"}, - }, - "label": map[string]interface{}{ - "type": "string", - "description": "The question text to display to the user", - }, - "placeholder": map[string]interface{}{ - "type": "string", - "description": "Placeholder text for text/number inputs (optional)", - }, - "required": map[string]interface{}{ - "type": "boolean", - "description": "Whether the user must answer this question (default: false)", - "default": false, - }, - "options": map[string]interface{}{ - "type": "array", - "description": "Options for 'select' or 'multi-select' questions (required for those types)", - "items": map[string]interface{}{ - "type": "string", - }, - }, - "allow_other": map[string]interface{}{ - "type": "boolean", - "description": "For select/multi-select: allow 'Other' option with custom text input (default: false)", - "default": false, - }, - "validation": map[string]interface{}{ - "type": "object", - "description": "Validation rules for the question (optional)", - "properties": map[string]interface{}{ - "min": map[string]interface{}{ - "type": "number", - "description": "Minimum value for number type", - }, - "max": map[string]interface{}{ - "type": "number", - "description": "Maximum value for number type", - }, - "pattern": map[string]interface{}{ - "type": "string", - "description": "Regex pattern for text validation (e.g., email pattern)", - }, - "min_length": map[string]interface{}{ - "type": "integer", - "description": "Minimum length for text input", - }, - "max_length": map[string]interface{}{ - "type": "integer", - "description": "Maximum length for text input", - }, - }, - }, - }, - "required": []string{"id", "type", "label"}, - }, - }, - "allow_skip": map[string]interface{}{ - "type": "boolean", - "description": "Whether the user can skip/cancel the prompt (default: true). Set to false for critical questions.", - "default": true, - }, - }, - "required": []string{"title", "questions"}, - }, - Execute: executeAskUser, - Source: ToolSourceBuiltin, - Category: "interaction", - Keywords: []string{"ask", "question", "prompt", "user", "input", "clarify", "modal"}, - } -} - -func executeAskUser(args map[string]interface{}) (string, error) { - // Extract user connection (injected by chat service) - userConn, ok := args[UserConnectionKey].(*models.UserConnection) - if !ok || userConn == nil { - return "", fmt.Errorf("interactive prompts are not available in this context (user connection not found)") - } - - // Extract title - title, ok := args["title"].(string) - if !ok || title == "" { - return "", fmt.Errorf("title is required") - } - - // Extract description (optional) - description, _ := args["description"].(string) - - // Extract questions array - questionsRaw, ok := args["questions"].([]interface{}) - if !ok || len(questionsRaw) == 0 { - return "", fmt.Errorf("questions array is required and must not be empty") - } - - // Convert questions to InteractiveQuestion structs - questions := make([]models.InteractiveQuestion, 0, len(questionsRaw)) - for i, qRaw := range questionsRaw { - qMap, ok := qRaw.(map[string]interface{}) - if !ok { - return "", fmt.Errorf("question at index %d is not a valid object", i) - } - - // Extract required fields - id, _ := qMap["id"].(string) - qType, _ := qMap["type"].(string) - label, _ := qMap["label"].(string) - - if id == "" || qType == "" || label == "" { - return "", fmt.Errorf("question at index %d is missing required fields (id, type, or label)", i) - } - - // Validate question type - validTypes := map[string]bool{ - "text": true, - "number": true, - "checkbox": true, - "select": true, - "multi-select": true, - } - if !validTypes[qType] { - return "", fmt.Errorf("invalid question type '%s' at index %d. Must be: text, number, checkbox, select, or multi-select", qType, i) - } - - question := models.InteractiveQuestion{ - ID: id, - Type: qType, - Label: label, - } - - // Optional: placeholder - if placeholder, ok := qMap["placeholder"].(string); ok { - question.Placeholder = placeholder - } - - // Optional: required - if required, ok := qMap["required"].(bool); ok { - question.Required = required - } - - // Optional: options (required for select/multi-select) - if optionsRaw, ok := qMap["options"].([]interface{}); ok { - options := make([]string, 0, len(optionsRaw)) - for _, opt := range optionsRaw { - if optStr, ok := opt.(string); ok { - options = append(options, optStr) - } - } - question.Options = options - } else if qType == "select" || qType == "multi-select" { - return "", fmt.Errorf("question '%s' (type: %s) requires an 'options' array", id, qType) - } - - // Optional: allow_other - if allowOther, ok := qMap["allow_other"].(bool); ok { - question.AllowOther = allowOther - } - - // Optional: validation - if validationRaw, ok := qMap["validation"].(map[string]interface{}); ok { - validation := &models.QuestionValidation{} - - if min, ok := validationRaw["min"].(float64); ok { - validation.Min = &min - } - if max, ok := validationRaw["max"].(float64); ok { - validation.Max = &max - } - if pattern, ok := validationRaw["pattern"].(string); ok { - validation.Pattern = pattern - } - if minLength, ok := validationRaw["min_length"].(float64); ok { - minLenInt := int(minLength) - validation.MinLength = &minLenInt - } - if maxLength, ok := validationRaw["max_length"].(float64); ok { - maxLenInt := int(maxLength) - validation.MaxLength = &maxLenInt - } - - question.Validation = validation - } - - questions = append(questions, question) - } - - // Extract allow_skip (default: true) - allowSkip := true - if skipRaw, ok := args["allow_skip"].(bool); ok { - allowSkip = skipRaw - } - - // Generate prompt ID - promptID := uuid.New().String() - - // Create the prompt message - prompt := models.ServerMessage{ - Type: "interactive_prompt", - PromptID: promptID, - ConversationID: userConn.ConversationID, - Title: title, - Description: description, - Questions: questions, - AllowSkip: &allowSkip, - } - - // Extract prompt waiter function (injected by chat service) - waiterFunc, ok := args[PromptWaiterKey].(models.PromptWaiterFunc) - if !ok || waiterFunc == nil { - return "", fmt.Errorf("prompt waiter not available (internal error)") - } - - // Send the prompt to the user - success := userConn.SafeSend(prompt) - if !success { - log.Printf("❌ [ASK_USER] Failed to send interactive prompt (connection closed)") - return "", fmt.Errorf("failed to send prompt: connection closed") - } - - log.Printf("✅ [ASK_USER] Sent interactive prompt: %s (id: %s, questions: %d, allow_skip: %v)", - title, promptID, len(questions), allowSkip) - log.Printf("⏳ [ASK_USER] Waiting for user response...") - - // Wait for user response (5 minute timeout) - answers, skipped, err := waiterFunc(promptID, 5*time.Minute) - if err != nil { - log.Printf("❌ [ASK_USER] Error waiting for response: %v", err) - return "", fmt.Errorf("failed to receive user response: %w", err) - } - - // Check if user skipped - if skipped { - log.Printf("📋 [ASK_USER] User skipped the prompt") - return "User skipped the prompt without providing answers.", nil - } - - // Format the answers for the LLM - result := map[string]interface{}{ - "status": "completed", - "message": "User answered the prompt. Here are their responses:", - "answers": make(map[string]interface{}), - } - - // Convert answers to a format the LLM can understand - for questionID, answer := range answers { - // Find the question to get its label - var questionLabel string - for _, q := range questions { - if q.ID == questionID { - questionLabel = q.Label - break - } - } - - answerData := map[string]interface{}{ - "question": questionLabel, - "answer": answer.Value, - } - - if answer.IsOther { - answerData["is_custom_answer"] = true - } - - result["answers"].(map[string]interface{})[questionID] = answerData - } - - resultJSON, _ := json.MarshalIndent(result, "", " ") - log.Printf("✅ [ASK_USER] Returning user's answers to LLM") - return string(resultJSON), nil -} diff --git a/backend/internal/tools/brevo_tool.go b/backend/internal/tools/brevo_tool.go deleted file mode 100644 index 428f9bef..00000000 --- a/backend/internal/tools/brevo_tool.go +++ /dev/null @@ -1,325 +0,0 @@ -package tools - -import ( - "bytes" - "encoding/json" - "fmt" - "io" - "net/http" - "strings" - "time" -) - -// NewBrevoTool creates a Brevo (formerly SendInBlue) email sending tool -func NewBrevoTool() *Tool { - return &Tool{ - Name: "send_brevo_email", - DisplayName: "Send Email (Brevo)", - Description: `Send emails via Brevo (formerly SendInBlue) API. Supports transactional emails, marketing campaigns, and templates. - -Features: -- Send to single or multiple recipients (to, cc, bcc) -- HTML and plain text email bodies -- Custom sender name and reply-to address -- Template support with dynamic parameters -- Attachment support via URL - -Authentication is handled automatically via configured Brevo credentials. Do NOT ask users for API keys. -The sender email (from_email) can be configured in credentials as default, or overridden per-request.`, - Icon: "Mail", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"brevo", "sendinblue", "email", "send", "mail", "message", "notification", "newsletter", "transactional", "marketing"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "to": map[string]interface{}{ - "type": "string", - "description": "Recipient email address(es). For multiple recipients, separate with commas (e.g., 'user1@example.com, user2@example.com')", - }, - "from_email": map[string]interface{}{ - "type": "string", - "description": "Sender email address (optional). Must be a verified sender in Brevo. If not provided, uses the default from_email configured in credentials.", - }, - "from_name": map[string]interface{}{ - "type": "string", - "description": "Sender display name (optional, e.g., 'John Doe' or 'My Company')", - }, - "subject": map[string]interface{}{ - "type": "string", - "description": "Email subject line", - }, - "text_content": map[string]interface{}{ - "type": "string", - "description": "Plain text email body. Either text_content or html_content (or both) must be provided.", - }, - "html_content": map[string]interface{}{ - "type": "string", - "description": "HTML email body for rich formatting. Either text_content or html_content (or both) must be provided.", - }, - "cc": map[string]interface{}{ - "type": "string", - "description": "CC recipient(s). For multiple, separate with commas.", - }, - "bcc": map[string]interface{}{ - "type": "string", - "description": "BCC recipient(s). For multiple, separate with commas.", - }, - "reply_to": map[string]interface{}{ - "type": "string", - "description": "Reply-to email address (optional)", - }, - "template_id": map[string]interface{}{ - "type": "number", - "description": "Brevo template ID to use instead of html_content/text_content (optional)", - }, - "params": map[string]interface{}{ - "type": "object", - "description": "Template parameters as key-value pairs (optional, used with template_id)", - }, - "tags": map[string]interface{}{ - "type": "array", - "description": "Tags to categorize this email (optional)", - "items": map[string]interface{}{ - "type": "string", - }, - }, - }, - "required": []string{"to", "subject"}, - }, - Execute: executeBrevoEmail, - } -} - -// BrevoRecipient represents an email recipient -type BrevoRecipient struct { - Email string `json:"email"` - Name string `json:"name,omitempty"` -} - -// BrevoSender represents the email sender -type BrevoSender struct { - Email string `json:"email"` - Name string `json:"name,omitempty"` -} - -// BrevoRequest represents the Brevo API request -type BrevoRequest struct { - Sender BrevoSender `json:"sender"` - To []BrevoRecipient `json:"to"` - CC []BrevoRecipient `json:"cc,omitempty"` - BCC []BrevoRecipient `json:"bcc,omitempty"` - ReplyTo *BrevoRecipient `json:"replyTo,omitempty"` - Subject string `json:"subject"` - HTMLContent string `json:"htmlContent,omitempty"` - TextContent string `json:"textContent,omitempty"` - TemplateID int `json:"templateId,omitempty"` - Params map[string]interface{} `json:"params,omitempty"` - Tags []string `json:"tags,omitempty"` -} - -func executeBrevoEmail(args map[string]interface{}) (string, error) { - // Get all credential data first - credData, credErr := GetCredentialData(args, "brevo") - - // Resolve API key from credential - apiKey, err := ResolveAPIKey(args, "brevo", "api_key") - if err != nil { - return "", fmt.Errorf("failed to get Brevo API key: %w. Please configure Brevo credentials first.", err) - } - - // Validate API key format (Brevo keys start with "xkeysib-") - if !strings.HasPrefix(apiKey, "xkeysib-") { - return "", fmt.Errorf("invalid Brevo API key format (should start with 'xkeysib-')") - } - - // Extract required parameters - toStr, ok := args["to"].(string) - if !ok || toStr == "" { - return "", fmt.Errorf("'to' email address is required") - } - - // Get from_email - first check args, then fall back to credential data - fromEmail, _ := args["from_email"].(string) - if fromEmail == "" && credErr == nil && credData != nil { - if credFromEmail, ok := credData["from_email"].(string); ok { - fromEmail = credFromEmail - } - } - if fromEmail == "" { - return "", fmt.Errorf("'from_email' is required - either provide it in the request or configure a default in Brevo credentials") - } - - subject, ok := args["subject"].(string) - if !ok || subject == "" { - return "", fmt.Errorf("'subject' is required") - } - - // Extract content - textContent, _ := args["text_content"].(string) - htmlContent, _ := args["html_content"].(string) - templateID := 0 - if tid, ok := args["template_id"].(float64); ok { - templateID = int(tid) - } - - // Require either content or template - if textContent == "" && htmlContent == "" && templateID == 0 { - return "", fmt.Errorf("either 'text_content', 'html_content', or 'template_id' is required") - } - - // Parse recipient email addresses - toRecipients := parseBrevoEmailList(toStr) - if len(toRecipients) == 0 { - return "", fmt.Errorf("at least one valid 'to' email address is required") - } - - // Build sender - sender := BrevoSender{Email: fromEmail} - if fromName, ok := args["from_name"].(string); ok && fromName != "" { - sender.Name = fromName - } - - // Build request - request := BrevoRequest{ - Sender: sender, - To: toRecipients, - Subject: subject, - } - - // Add content or template - if templateID > 0 { - request.TemplateID = templateID - if params, ok := args["params"].(map[string]interface{}); ok { - request.Params = params - } - } else { - if htmlContent != "" { - request.HTMLContent = htmlContent - } - if textContent != "" { - request.TextContent = textContent - } - } - - // Parse CC recipients - if ccStr, ok := args["cc"].(string); ok && ccStr != "" { - request.CC = parseBrevoEmailList(ccStr) - } - - // Parse BCC recipients - if bccStr, ok := args["bcc"].(string); ok && bccStr != "" { - request.BCC = parseBrevoEmailList(bccStr) - } - - // Add reply-to if provided - if replyTo, ok := args["reply_to"].(string); ok && replyTo != "" { - request.ReplyTo = &BrevoRecipient{Email: replyTo} - } - - // Add tags if provided - if tags, ok := args["tags"].([]interface{}); ok { - for _, tag := range tags { - if tagStr, ok := tag.(string); ok { - request.Tags = append(request.Tags, tagStr) - } - } - } - - // Serialize request - jsonPayload, err := json.Marshal(request) - if err != nil { - return "", fmt.Errorf("failed to serialize request: %w", err) - } - - // Create HTTP client - client := &http.Client{ - Timeout: 30 * time.Second, - } - - // Create request to Brevo API - req, err := http.NewRequest("POST", "https://api.brevo.com/v3/smtp/email", bytes.NewBuffer(jsonPayload)) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("api-key", apiKey) - req.Header.Set("Accept", "application/json") - - // Execute request - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - // Read response body - respBody, err := io.ReadAll(resp.Body) - if err != nil { - return "", fmt.Errorf("failed to read response: %w", err) - } - - // Brevo returns 201 Created on success - success := resp.StatusCode == 201 - - // Build result - result := map[string]interface{}{ - "success": success, - "status_code": resp.StatusCode, - "email_sent": success, - "recipients": len(toRecipients), - "subject": subject, - "from": fromEmail, - } - - if len(request.CC) > 0 { - result["cc_count"] = len(request.CC) - } - if len(request.BCC) > 0 { - result["bcc_count"] = len(request.BCC) - } - if templateID > 0 { - result["template_id"] = templateID - } - - // Parse response - if len(respBody) > 0 { - var apiResp map[string]interface{} - if err := json.Unmarshal(respBody, &apiResp); err == nil { - if messageId, ok := apiResp["messageId"].(string); ok { - result["message_id"] = messageId - } - if !success { - result["error"] = apiResp - } - } else if !success { - result["error"] = string(respBody) - } - } - - if success { - result["message"] = fmt.Sprintf("Email sent successfully to %d recipient(s)", len(toRecipients)) - } - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} - -// parseBrevoEmailList parses a comma-separated list of email addresses -func parseBrevoEmailList(emailStr string) []BrevoRecipient { - var recipients []BrevoRecipient - parts := strings.Split(emailStr, ",") - for _, part := range parts { - email := strings.TrimSpace(part) - if email != "" && strings.Contains(email, "@") { - recipients = append(recipients, BrevoRecipient{Email: email}) - } - } - return recipients -} - diff --git a/backend/internal/tools/calendly_tool.go b/backend/internal/tools/calendly_tool.go deleted file mode 100644 index 14aabc1d..00000000 --- a/backend/internal/tools/calendly_tool.go +++ /dev/null @@ -1,323 +0,0 @@ -package tools - -import ( - "encoding/json" - "fmt" - "io" - "net/http" - "net/url" - "time" -) - -// NewCalendlyEventsTool creates a Calendly events listing tool -func NewCalendlyEventsTool() *Tool { - return &Tool{ - Name: "calendly_events", - DisplayName: "Calendly Events", - Description: `List scheduled events from Calendly. - -Returns event details including start/end times, invitees, and status. -Authentication is handled automatically via configured Calendly API key.`, - Icon: "Calendar", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"calendly", "events", "meetings", "schedule", "calendar"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system.", - }, - "status": map[string]interface{}{ - "type": "string", - "description": "Filter by status: active or canceled", - }, - "min_start_time": map[string]interface{}{ - "type": "string", - "description": "Filter events starting after this time (ISO 8601 format)", - }, - "max_start_time": map[string]interface{}{ - "type": "string", - "description": "Filter events starting before this time (ISO 8601 format)", - }, - "count": map[string]interface{}{ - "type": "number", - "description": "Number of events to return (max 100)", - }, - }, - "required": []string{}, - }, - Execute: executeCalendlyEvents, - } -} - -// NewCalendlyEventTypesTool creates a Calendly event types listing tool -func NewCalendlyEventTypesTool() *Tool { - return &Tool{ - Name: "calendly_event_types", - DisplayName: "Calendly Event Types", - Description: `List available event types from Calendly. - -Returns scheduling links and configuration for each event type. -Authentication is handled automatically via configured Calendly API key.`, - Icon: "Calendar", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"calendly", "event", "types", "scheduling", "links"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system.", - }, - "active": map[string]interface{}{ - "type": "boolean", - "description": "Filter by active status", - }, - "count": map[string]interface{}{ - "type": "number", - "description": "Number of event types to return (max 100)", - }, - }, - "required": []string{}, - }, - Execute: executeCalendlyEventTypes, - } -} - -// NewCalendlyInviteesTool creates a Calendly invitees listing tool -func NewCalendlyInviteesTool() *Tool { - return &Tool{ - Name: "calendly_invitees", - DisplayName: "Calendly Invitees", - Description: `List invitees for a specific Calendly event. - -Returns invitee details including name, email, and responses. -Authentication is handled automatically via configured Calendly API key.`, - Icon: "Users", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"calendly", "invitees", "attendees", "participants"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system.", - }, - "event_uri": map[string]interface{}{ - "type": "string", - "description": "The scheduled event URI to get invitees for", - }, - "status": map[string]interface{}{ - "type": "string", - "description": "Filter by status: active or canceled", - }, - "count": map[string]interface{}{ - "type": "number", - "description": "Number of invitees to return (max 100)", - }, - }, - "required": []string{"event_uri"}, - }, - Execute: executeCalendlyInvitees, - } -} - -func getCalendlyCurrentUser(apiKey string) (string, error) { - req, _ := http.NewRequest("GET", "https://api.calendly.com/users/me", nil) - req.Header.Set("Authorization", "Bearer "+apiKey) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", err - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - var result map[string]interface{} - json.Unmarshal(body, &result) - - if resp.StatusCode >= 400 { - return "", fmt.Errorf("failed to get current user") - } - - if resource, ok := result["resource"].(map[string]interface{}); ok { - if uri, ok := resource["uri"].(string); ok { - return uri, nil - } - } - return "", fmt.Errorf("user URI not found") -} - -func executeCalendlyEvents(args map[string]interface{}) (string, error) { - credData, err := GetCredentialData(args, "calendly") - if err != nil { - return "", fmt.Errorf("failed to get Calendly credentials: %w", err) - } - - apiKey, _ := credData["api_key"].(string) - if apiKey == "" { - return "", fmt.Errorf("Calendly API key not configured") - } - - userURI, err := getCalendlyCurrentUser(apiKey) - if err != nil { - return "", fmt.Errorf("failed to get current user: %w", err) - } - - queryParams := url.Values{} - queryParams.Set("user", userURI) - if status, ok := args["status"].(string); ok && status != "" { - queryParams.Set("status", status) - } - if minStart, ok := args["min_start_time"].(string); ok && minStart != "" { - queryParams.Set("min_start_time", minStart) - } - if maxStart, ok := args["max_start_time"].(string); ok && maxStart != "" { - queryParams.Set("max_start_time", maxStart) - } - if count, ok := args["count"].(float64); ok && count > 0 { - queryParams.Set("count", fmt.Sprintf("%d", int(count))) - } - - apiURL := "https://api.calendly.com/scheduled_events?" + queryParams.Encode() - req, _ := http.NewRequest("GET", apiURL, nil) - req.Header.Set("Authorization", "Bearer "+apiKey) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to send request: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - var result map[string]interface{} - json.Unmarshal(body, &result) - - if resp.StatusCode >= 400 { - errMsg := "unknown error" - if msg, ok := result["message"].(string); ok { - errMsg = msg - } - return "", fmt.Errorf("Calendly API error: %s", errMsg) - } - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} - -func executeCalendlyEventTypes(args map[string]interface{}) (string, error) { - credData, err := GetCredentialData(args, "calendly") - if err != nil { - return "", fmt.Errorf("failed to get Calendly credentials: %w", err) - } - - apiKey, _ := credData["api_key"].(string) - if apiKey == "" { - return "", fmt.Errorf("Calendly API key not configured") - } - - userURI, err := getCalendlyCurrentUser(apiKey) - if err != nil { - return "", fmt.Errorf("failed to get current user: %w", err) - } - - queryParams := url.Values{} - queryParams.Set("user", userURI) - if active, ok := args["active"].(bool); ok { - queryParams.Set("active", fmt.Sprintf("%t", active)) - } - if count, ok := args["count"].(float64); ok && count > 0 { - queryParams.Set("count", fmt.Sprintf("%d", int(count))) - } - - apiURL := "https://api.calendly.com/event_types?" + queryParams.Encode() - req, _ := http.NewRequest("GET", apiURL, nil) - req.Header.Set("Authorization", "Bearer "+apiKey) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to send request: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - var result map[string]interface{} - json.Unmarshal(body, &result) - - if resp.StatusCode >= 400 { - errMsg := "unknown error" - if msg, ok := result["message"].(string); ok { - errMsg = msg - } - return "", fmt.Errorf("Calendly API error: %s", errMsg) - } - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} - -func executeCalendlyInvitees(args map[string]interface{}) (string, error) { - credData, err := GetCredentialData(args, "calendly") - if err != nil { - return "", fmt.Errorf("failed to get Calendly credentials: %w", err) - } - - apiKey, _ := credData["api_key"].(string) - if apiKey == "" { - return "", fmt.Errorf("Calendly API key not configured") - } - - eventURI, _ := args["event_uri"].(string) - if eventURI == "" { - return "", fmt.Errorf("'event_uri' is required") - } - - queryParams := url.Values{} - if status, ok := args["status"].(string); ok && status != "" { - queryParams.Set("status", status) - } - if count, ok := args["count"].(float64); ok && count > 0 { - queryParams.Set("count", fmt.Sprintf("%d", int(count))) - } - - apiURL := eventURI + "/invitees" - if len(queryParams) > 0 { - apiURL += "?" + queryParams.Encode() - } - - req, _ := http.NewRequest("GET", apiURL, nil) - req.Header.Set("Authorization", "Bearer "+apiKey) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to send request: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - var result map[string]interface{} - json.Unmarshal(body, &result) - - if resp.StatusCode >= 400 { - errMsg := "unknown error" - if msg, ok := result["message"].(string); ok { - errMsg = msg - } - return "", fmt.Errorf("Calendly API error: %s", errMsg) - } - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} diff --git a/backend/internal/tools/clickup_tool.go b/backend/internal/tools/clickup_tool.go deleted file mode 100644 index 0304ebd5..00000000 --- a/backend/internal/tools/clickup_tool.go +++ /dev/null @@ -1,322 +0,0 @@ -package tools - -import ( - "bytes" - "encoding/json" - "fmt" - "io" - "net/http" - "time" -) - -// NewClickUpTasksTool creates a ClickUp tasks listing tool -func NewClickUpTasksTool() *Tool { - return &Tool{ - Name: "clickup_tasks", - DisplayName: "ClickUp Tasks", - Description: `List tasks from a ClickUp list. - -Returns task details including name, status, assignees, and due dates. -Authentication is handled automatically via configured ClickUp API key.`, - Icon: "CheckSquare", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"clickup", "tasks", "list", "project", "management"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system.", - }, - "list_id": map[string]interface{}{ - "type": "string", - "description": "The ClickUp list ID to get tasks from", - }, - "archived": map[string]interface{}{ - "type": "boolean", - "description": "Include archived tasks", - }, - "subtasks": map[string]interface{}{ - "type": "boolean", - "description": "Include subtasks", - }, - }, - "required": []string{"list_id"}, - }, - Execute: executeClickUpTasks, - } -} - -// NewClickUpCreateTaskTool creates a ClickUp task creation tool -func NewClickUpCreateTaskTool() *Tool { - return &Tool{ - Name: "clickup_create_task", - DisplayName: "Create ClickUp Task", - Description: `Create a new task in a ClickUp list. - -Supports setting name, description, status, priority, due date, assignees, and tags. -Authentication is handled automatically via configured ClickUp API key.`, - Icon: "Plus", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"clickup", "task", "create", "new", "project"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system.", - }, - "list_id": map[string]interface{}{ - "type": "string", - "description": "The ClickUp list ID to create task in", - }, - "name": map[string]interface{}{ - "type": "string", - "description": "The task name", - }, - "description": map[string]interface{}{ - "type": "string", - "description": "The task description (supports markdown)", - }, - "status": map[string]interface{}{ - "type": "string", - "description": "The task status", - }, - "priority": map[string]interface{}{ - "type": "number", - "description": "Priority: 1 (Urgent), 2 (High), 3 (Normal), 4 (Low)", - }, - "due_date": map[string]interface{}{ - "type": "number", - "description": "Due date as Unix timestamp in milliseconds", - }, - }, - "required": []string{"list_id", "name"}, - }, - Execute: executeClickUpCreateTask, - } -} - -// NewClickUpUpdateTaskTool creates a ClickUp task update tool -func NewClickUpUpdateTaskTool() *Tool { - return &Tool{ - Name: "clickup_update_task", - DisplayName: "Update ClickUp Task", - Description: `Update an existing ClickUp task. - -Can modify name, description, status, priority, due date, or archive status. -Authentication is handled automatically via configured ClickUp API key.`, - Icon: "Edit", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"clickup", "task", "update", "edit", "modify"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system.", - }, - "task_id": map[string]interface{}{ - "type": "string", - "description": "The ClickUp task ID to update", - }, - "name": map[string]interface{}{ - "type": "string", - "description": "The new task name", - }, - "description": map[string]interface{}{ - "type": "string", - "description": "The new task description", - }, - "status": map[string]interface{}{ - "type": "string", - "description": "The new task status", - }, - "priority": map[string]interface{}{ - "type": "number", - "description": "Priority: 1 (Urgent), 2 (High), 3 (Normal), 4 (Low)", - }, - }, - "required": []string{"task_id"}, - }, - Execute: executeClickUpUpdateTask, - } -} - -func executeClickUpTasks(args map[string]interface{}) (string, error) { - credData, err := GetCredentialData(args, "clickup") - if err != nil { - return "", fmt.Errorf("failed to get ClickUp credentials: %w", err) - } - - apiKey, _ := credData["api_key"].(string) - if apiKey == "" { - return "", fmt.Errorf("ClickUp API key not configured") - } - - listID, _ := args["list_id"].(string) - if listID == "" { - return "", fmt.Errorf("'list_id' is required") - } - - apiURL := fmt.Sprintf("https://api.clickup.com/api/v2/list/%s/task", listID) - if archived, ok := args["archived"].(bool); ok && archived { - apiURL += "?archived=true" - } - - req, _ := http.NewRequest("GET", apiURL, nil) - req.Header.Set("Authorization", apiKey) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to send request: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - var result map[string]interface{} - json.Unmarshal(body, &result) - - if resp.StatusCode >= 400 { - errMsg := "unknown error" - if msg, ok := result["err"].(string); ok { - errMsg = msg - } - return "", fmt.Errorf("ClickUp API error: %s", errMsg) - } - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} - -func executeClickUpCreateTask(args map[string]interface{}) (string, error) { - credData, err := GetCredentialData(args, "clickup") - if err != nil { - return "", fmt.Errorf("failed to get ClickUp credentials: %w", err) - } - - apiKey, _ := credData["api_key"].(string) - if apiKey == "" { - return "", fmt.Errorf("ClickUp API key not configured") - } - - listID, _ := args["list_id"].(string) - name, _ := args["name"].(string) - if listID == "" || name == "" { - return "", fmt.Errorf("'list_id' and 'name' are required") - } - - payload := map[string]interface{}{"name": name} - if desc, ok := args["description"].(string); ok && desc != "" { - payload["description"] = desc - } - if status, ok := args["status"].(string); ok && status != "" { - payload["status"] = status - } - if priority, ok := args["priority"].(float64); ok && priority > 0 { - payload["priority"] = int(priority) - } - if dueDate, ok := args["due_date"].(float64); ok && dueDate > 0 { - payload["due_date"] = int64(dueDate) - } - - jsonBody, _ := json.Marshal(payload) - apiURL := fmt.Sprintf("https://api.clickup.com/api/v2/list/%s/task", listID) - req, _ := http.NewRequest("POST", apiURL, bytes.NewBuffer(jsonBody)) - req.Header.Set("Authorization", apiKey) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to send request: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - var result map[string]interface{} - json.Unmarshal(body, &result) - - if resp.StatusCode >= 400 { - errMsg := "unknown error" - if msg, ok := result["err"].(string); ok { - errMsg = msg - } - return "", fmt.Errorf("ClickUp API error: %s", errMsg) - } - - output := map[string]interface{}{ - "success": true, - "task": result, - } - jsonResult, _ := json.MarshalIndent(output, "", " ") - return string(jsonResult), nil -} - -func executeClickUpUpdateTask(args map[string]interface{}) (string, error) { - credData, err := GetCredentialData(args, "clickup") - if err != nil { - return "", fmt.Errorf("failed to get ClickUp credentials: %w", err) - } - - apiKey, _ := credData["api_key"].(string) - if apiKey == "" { - return "", fmt.Errorf("ClickUp API key not configured") - } - - taskID, _ := args["task_id"].(string) - if taskID == "" { - return "", fmt.Errorf("'task_id' is required") - } - - payload := make(map[string]interface{}) - if name, ok := args["name"].(string); ok && name != "" { - payload["name"] = name - } - if desc, ok := args["description"].(string); ok && desc != "" { - payload["description"] = desc - } - if status, ok := args["status"].(string); ok && status != "" { - payload["status"] = status - } - if priority, ok := args["priority"].(float64); ok && priority > 0 { - payload["priority"] = int(priority) - } - - jsonBody, _ := json.Marshal(payload) - apiURL := fmt.Sprintf("https://api.clickup.com/api/v2/task/%s", taskID) - req, _ := http.NewRequest("PUT", apiURL, bytes.NewBuffer(jsonBody)) - req.Header.Set("Authorization", apiKey) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to send request: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - var result map[string]interface{} - json.Unmarshal(body, &result) - - if resp.StatusCode >= 400 { - errMsg := "unknown error" - if msg, ok := result["err"].(string); ok { - errMsg = msg - } - return "", fmt.Errorf("ClickUp API error: %s", errMsg) - } - - output := map[string]interface{}{ - "success": true, - "task": result, - } - jsonResult, _ := json.MarshalIndent(output, "", " ") - return string(jsonResult), nil -} diff --git a/backend/internal/tools/composio_gmail_tool.go b/backend/internal/tools/composio_gmail_tool.go deleted file mode 100644 index 3d1c8de4..00000000 --- a/backend/internal/tools/composio_gmail_tool.go +++ /dev/null @@ -1,1224 +0,0 @@ -package tools - -import ( - "bytes" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "net/url" - "os" - "regexp" - "strings" - "sync" - "time" -) - -// composioGmailRateLimiter implements per-user rate limiting for Composio Gmail API calls -type composioGmailRateLimiter struct { - requests map[string][]time.Time - mutex sync.RWMutex - maxCalls int - window time.Duration -} - -var globalGmailRateLimiter = &composioGmailRateLimiter{ - requests: make(map[string][]time.Time), - maxCalls: 50, - window: 1 * time.Minute, -} - -// checkGmailRateLimit checks rate limit using user ID from args -func checkGmailRateLimit(args map[string]interface{}) error { - userID, ok := args["__user_id__"].(string) - if !ok || userID == "" { - log.Printf("⚠️ [GMAIL] No user ID for rate limiting") - return nil - } - - globalGmailRateLimiter.mutex.Lock() - defer globalGmailRateLimiter.mutex.Unlock() - - now := time.Now() - windowStart := now.Add(-globalGmailRateLimiter.window) - - timestamps := globalGmailRateLimiter.requests[userID] - validTimestamps := []time.Time{} - for _, ts := range timestamps { - if ts.After(windowStart) { - validTimestamps = append(validTimestamps, ts) - } - } - - if len(validTimestamps) >= globalGmailRateLimiter.maxCalls { - return fmt.Errorf("rate limit exceeded: max %d requests per minute", globalGmailRateLimiter.maxCalls) - } - - validTimestamps = append(validTimestamps, now) - globalGmailRateLimiter.requests[userID] = validTimestamps - return nil -} - -// NewComposioGmailSendTool creates a tool for sending emails via Composio Gmail -func NewComposioGmailSendTool() *Tool { - return &Tool{ - Name: "gmail_send_email", - DisplayName: "Gmail - Send Email", - Description: `Send an email via Gmail using OAuth authentication. - -Features: -- Send to multiple recipients (To, Cc, Bcc) -- HTML or plain text body -- Subject and body -- OAuth authentication handled by Composio - -Use this to send emails from the authenticated user's Gmail account.`, - Icon: "Mail", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"gmail", "email", "send", "compose", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "recipient_email": map[string]interface{}{ - "type": "string", - "description": "Primary recipient email address", - }, - "subject": map[string]interface{}{ - "type": "string", - "description": "Email subject", - }, - "body": map[string]interface{}{ - "type": "string", - "description": "Email body (plain text or HTML)", - }, - "is_html": map[string]interface{}{ - "type": "boolean", - "description": "Set to true if body contains HTML (default: false)", - }, - "cc": map[string]interface{}{ - "type": "array", - "description": "Array of CC email addresses", - "items": map[string]interface{}{ - "type": "string", - }, - }, - "bcc": map[string]interface{}{ - "type": "array", - "description": "Array of BCC email addresses", - "items": map[string]interface{}{ - "type": "string", - }, - }, - }, - "required": []string{}, - }, - Execute: executeComposioGmailSend, - } -} - -func executeComposioGmailSend(args map[string]interface{}) (string, error) { - if err := checkGmailRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_gmail") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - // Build input - input := map[string]interface{}{ - "user_id": "me", - } - - if recipientEmail, ok := args["recipient_email"].(string); ok && recipientEmail != "" { - input["recipient_email"] = recipientEmail - } - if subject, ok := args["subject"].(string); ok { - input["subject"] = subject - } - if body, ok := args["body"].(string); ok { - input["body"] = body - } - if isHTML, ok := args["is_html"].(bool); ok { - input["is_html"] = isHTML - } - if cc, ok := args["cc"].([]interface{}); ok && len(cc) > 0 { - input["cc"] = cc - } - if bcc, ok := args["bcc"].([]interface{}); ok && len(bcc) > 0 { - input["bcc"] = bcc - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "gmail", - "input": input, - } - - return callComposioGmailAPI(composioAPIKey, entityID, "GMAIL_SEND_EMAIL", payload) -} - -// NewComposioGmailFetchTool creates a tool for fetching/searching emails -func NewComposioGmailFetchTool() *Tool { - return &Tool{ - Name: "gmail_fetch_emails", - DisplayName: "Gmail - Fetch Emails", - Description: `Fetch and search emails from Gmail. - -Features: -- Search with Gmail query syntax (e.g., "from:user@example.com", "is:unread") -- Filter by labels -- Pagination support -- Returns email metadata and content -- OAuth authentication handled by Composio - -Use this to list, search, and retrieve emails from Gmail inbox.`, - Icon: "Mail", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"gmail", "email", "fetch", "search", "list", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "query": map[string]interface{}{ - "type": "string", - "description": "Gmail search query (e.g., 'from:user@example.com is:unread')", - }, - "max_results": map[string]interface{}{ - "type": "integer", - "description": "Maximum number of emails to return (default: 10)", - }, - "label_ids": map[string]interface{}{ - "type": "array", - "description": "Filter by label IDs (e.g., ['INBOX', 'UNREAD'])", - "items": map[string]interface{}{ - "type": "string", - }, - }, - }, - "required": []string{}, - }, - Execute: executeComposioGmailFetch, - } -} - -func executeComposioGmailFetch(args map[string]interface{}) (string, error) { - if err := checkGmailRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_gmail") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - // Build input - input := map[string]interface{}{ - "user_id": "me", - "include_payload": true, - "verbose": true, - } - - if query, ok := args["query"].(string); ok && query != "" { - input["query"] = query - } - if maxResults, ok := args["max_results"].(float64); ok { - input["max_results"] = int(maxResults) - } else { - input["max_results"] = 10 - } - if labelIDs, ok := args["label_ids"].([]interface{}); ok && len(labelIDs) > 0 { - input["label_ids"] = labelIDs - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "gmail", - "input": input, - } - - result, err := callComposioGmailAPI(composioAPIKey, entityID, "GMAIL_FETCH_EMAILS", payload) - if err != nil { - return "", err - } - - // Parse and simplify the response for LLM consumption - return simplifyGmailFetchResponse(result) -} - -// NewComposioGmailGetMessageTool creates a tool for getting a specific email by ID -func NewComposioGmailGetMessageTool() *Tool { - return &Tool{ - Name: "gmail_get_message", - DisplayName: "Gmail - Get Message", - Description: `Get a specific email message by its ID. - -Features: -- Retrieve full email content and metadata -- Get headers, body, attachments info -- OAuth authentication handled by Composio - -Use this to fetch details of a specific email when you have its message ID.`, - Icon: "Mail", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"gmail", "email", "get", "fetch", "message", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "message_id": map[string]interface{}{ - "type": "string", - "description": "The Gmail message ID", - }, - }, - "required": []string{"message_id"}, - }, - Execute: executeComposioGmailGetMessage, - } -} - -func executeComposioGmailGetMessage(args map[string]interface{}) (string, error) { - if err := checkGmailRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_gmail") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - messageID, _ := args["message_id"].(string) - if messageID == "" { - return "", fmt.Errorf("'message_id' is required") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "gmail", - "input": map[string]interface{}{ - "message_id": messageID, - "user_id": "me", - "format": "full", - }, - } - - return callComposioGmailAPI(composioAPIKey, entityID, "GMAIL_FETCH_MESSAGE_BY_MESSAGE_ID", payload) -} - -// NewComposioGmailReplyTool creates a tool for replying to email threads -func NewComposioGmailReplyTool() *Tool { - return &Tool{ - Name: "gmail_reply_to_thread", - DisplayName: "Gmail - Reply to Thread", - Description: `Reply to an existing email thread. - -Features: -- Reply within existing conversation -- Maintains thread continuity -- Supports HTML or plain text -- OAuth authentication handled by Composio - -Use this to send replies to existing email conversations.`, - Icon: "Mail", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"gmail", "email", "reply", "thread", "conversation", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "thread_id": map[string]interface{}{ - "type": "string", - "description": "The Gmail thread ID to reply to", - }, - "message_body": map[string]interface{}{ - "type": "string", - "description": "Reply message body", - }, - "recipient_email": map[string]interface{}{ - "type": "string", - "description": "Recipient email (optional if replying to thread)", - }, - "is_html": map[string]interface{}{ - "type": "boolean", - "description": "Set to true if body contains HTML", - }, - }, - "required": []string{"thread_id"}, - }, - Execute: executeComposioGmailReply, - } -} - -func executeComposioGmailReply(args map[string]interface{}) (string, error) { - if err := checkGmailRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_gmail") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - threadID, _ := args["thread_id"].(string) - if threadID == "" { - return "", fmt.Errorf("'thread_id' is required") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - input := map[string]interface{}{ - "thread_id": threadID, - "user_id": "me", - } - - if messageBody, ok := args["message_body"].(string); ok && messageBody != "" { - input["message_body"] = messageBody - } - if recipientEmail, ok := args["recipient_email"].(string); ok && recipientEmail != "" { - input["recipient_email"] = recipientEmail - } - if isHTML, ok := args["is_html"].(bool); ok { - input["is_html"] = isHTML - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "gmail", - "input": input, - } - - return callComposioGmailAPI(composioAPIKey, entityID, "GMAIL_REPLY_TO_THREAD", payload) -} - -// NewComposioGmailCreateDraftTool creates a tool for creating email drafts -func NewComposioGmailCreateDraftTool() *Tool { - return &Tool{ - Name: "gmail_create_draft", - DisplayName: "Gmail - Create Draft", - Description: `Create an email draft in Gmail. - -Features: -- Create drafts to send later -- Supports To, Cc, Bcc -- HTML or plain text -- Can be edited before sending -- OAuth authentication handled by Composio - -Use this to create email drafts that can be reviewed and sent later.`, - Icon: "Mail", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"gmail", "email", "draft", "compose", "save", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "recipient_email": map[string]interface{}{ - "type": "string", - "description": "Primary recipient email address (optional)", - }, - "subject": map[string]interface{}{ - "type": "string", - "description": "Email subject", - }, - "body": map[string]interface{}{ - "type": "string", - "description": "Email body", - }, - "is_html": map[string]interface{}{ - "type": "boolean", - "description": "Set to true if body contains HTML", - }, - }, - "required": []string{}, - }, - Execute: executeComposioGmailCreateDraft, - } -} - -func executeComposioGmailCreateDraft(args map[string]interface{}) (string, error) { - if err := checkGmailRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_gmail") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - input := map[string]interface{}{ - "user_id": "me", - } - - if recipientEmail, ok := args["recipient_email"].(string); ok && recipientEmail != "" { - input["recipient_email"] = recipientEmail - } - if subject, ok := args["subject"].(string); ok { - input["subject"] = subject - } - if body, ok := args["body"].(string); ok { - input["body"] = body - } - if isHTML, ok := args["is_html"].(bool); ok { - input["is_html"] = isHTML - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "gmail", - "input": input, - } - - return callComposioGmailAPI(composioAPIKey, entityID, "GMAIL_CREATE_EMAIL_DRAFT", payload) -} - -// NewComposioGmailSendDraftTool creates a tool for sending existing drafts -func NewComposioGmailSendDraftTool() *Tool { - return &Tool{ - Name: "gmail_send_draft", - DisplayName: "Gmail - Send Draft", - Description: `Send an existing email draft. - -Features: -- Send previously created drafts -- Draft is deleted after sending -- OAuth authentication handled by Composio - -Use this to send drafts that were created earlier.`, - Icon: "Mail", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"gmail", "email", "draft", "send", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "draft_id": map[string]interface{}{ - "type": "string", - "description": "The Gmail draft ID to send", - }, - }, - "required": []string{"draft_id"}, - }, - Execute: executeComposioGmailSendDraft, - } -} - -func executeComposioGmailSendDraft(args map[string]interface{}) (string, error) { - if err := checkGmailRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_gmail") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - draftID, _ := args["draft_id"].(string) - if draftID == "" { - return "", fmt.Errorf("'draft_id' is required") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "gmail", - "input": map[string]interface{}{ - "draft_id": draftID, - "user_id": "me", - }, - } - - return callComposioGmailAPI(composioAPIKey, entityID, "GMAIL_SEND_DRAFT", payload) -} - -// NewComposioGmailListDraftsTool creates a tool for listing drafts -func NewComposioGmailListDraftsTool() *Tool { - return &Tool{ - Name: "gmail_list_drafts", - DisplayName: "Gmail - List Drafts", - Description: `List all email drafts in Gmail. - -Features: -- List all saved drafts -- Pagination support -- Returns draft IDs and metadata -- OAuth authentication handled by Composio - -Use this to view all saved email drafts.`, - Icon: "Mail", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"gmail", "email", "draft", "list", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "max_results": map[string]interface{}{ - "type": "integer", - "description": "Maximum number of drafts to return (default: 10)", - }, - }, - "required": []string{}, - }, - Execute: executeComposioGmailListDrafts, - } -} - -func executeComposioGmailListDrafts(args map[string]interface{}) (string, error) { - if err := checkGmailRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_gmail") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - input := map[string]interface{}{ - "user_id": "me", - "verbose": true, - } - - if maxResults, ok := args["max_results"].(float64); ok { - input["max_results"] = int(maxResults) - } else { - input["max_results"] = 10 - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "gmail", - "input": input, - } - - return callComposioGmailAPI(composioAPIKey, entityID, "GMAIL_LIST_DRAFTS", payload) -} - -// NewComposioGmailAddLabelTool creates a tool for managing email labels -func NewComposioGmailAddLabelTool() *Tool { - return &Tool{ - Name: "gmail_add_label", - DisplayName: "Gmail - Add/Remove Labels", - Description: `Add or remove labels from an email message. - -Features: -- Add labels to organize emails -- Remove labels from emails -- Use system labels (INBOX, UNREAD, STARRED, etc.) -- Use custom labels -- OAuth authentication handled by Composio - -Use this to organize emails with labels (categories/tags).`, - Icon: "Mail", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"gmail", "email", "label", "tag", "organize", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "message_id": map[string]interface{}{ - "type": "string", - "description": "The Gmail message ID", - }, - "add_label_ids": map[string]interface{}{ - "type": "array", - "description": "Array of label IDs to add (e.g., ['INBOX', 'STARRED'])", - "items": map[string]interface{}{ - "type": "string", - }, - }, - "remove_label_ids": map[string]interface{}{ - "type": "array", - "description": "Array of label IDs to remove (e.g., ['UNREAD'])", - "items": map[string]interface{}{ - "type": "string", - }, - }, - }, - "required": []string{"message_id"}, - }, - Execute: executeComposioGmailAddLabel, - } -} - -func executeComposioGmailAddLabel(args map[string]interface{}) (string, error) { - if err := checkGmailRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_gmail") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - messageID, _ := args["message_id"].(string) - if messageID == "" { - return "", fmt.Errorf("'message_id' is required") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - input := map[string]interface{}{ - "message_id": messageID, - "user_id": "me", - } - - if addLabelIDs, ok := args["add_label_ids"].([]interface{}); ok && len(addLabelIDs) > 0 { - input["add_label_ids"] = addLabelIDs - } - if removeLabelIDs, ok := args["remove_label_ids"].([]interface{}); ok && len(removeLabelIDs) > 0 { - input["remove_label_ids"] = removeLabelIDs - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "gmail", - "input": input, - } - - return callComposioGmailAPI(composioAPIKey, entityID, "GMAIL_ADD_LABEL_TO_EMAIL", payload) -} - -// NewComposioGmailListLabelsTool creates a tool for listing all labels -func NewComposioGmailListLabelsTool() *Tool { - return &Tool{ - Name: "gmail_list_labels", - DisplayName: "Gmail - List Labels", - Description: `List all Gmail labels (system and custom). - -Features: -- List all available labels -- Includes system labels (INBOX, SENT, TRASH, etc.) -- Includes user-created custom labels -- Returns label IDs and names -- OAuth authentication handled by Composio - -Use this to discover available labels for organizing emails.`, - Icon: "Mail", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"gmail", "email", "label", "list", "categories", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - }, - "required": []string{}, - }, - Execute: executeComposioGmailListLabels, - } -} - -func executeComposioGmailListLabels(args map[string]interface{}) (string, error) { - if err := checkGmailRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_gmail") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "gmail", - "input": map[string]interface{}{ - "user_id": "me", - }, - } - - return callComposioGmailAPI(composioAPIKey, entityID, "GMAIL_LIST_LABELS", payload) -} - -// NewComposioGmailTrashTool creates a tool for moving emails to trash -func NewComposioGmailTrashTool() *Tool { - return &Tool{ - Name: "gmail_move_to_trash", - DisplayName: "Gmail - Move to Trash", - Description: `Move an email message to trash. - -Features: -- Moves message to Trash (not permanent deletion) -- Can be recovered from Trash -- OAuth authentication handled by Composio - -Use this to delete emails (they go to Trash first).`, - Icon: "Mail", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"gmail", "email", "trash", "delete", "remove", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "message_id": map[string]interface{}{ - "type": "string", - "description": "The Gmail message ID to trash", - }, - }, - "required": []string{"message_id"}, - }, - Execute: executeComposioGmailTrash, - } -} - -func executeComposioGmailTrash(args map[string]interface{}) (string, error) { - if err := checkGmailRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_gmail") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - messageID, _ := args["message_id"].(string) - if messageID == "" { - return "", fmt.Errorf("'message_id' is required") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "gmail", - "input": map[string]interface{}{ - "message_id": messageID, - "user_id": "me", - }, - } - - return callComposioGmailAPI(composioAPIKey, entityID, "GMAIL_MOVE_TO_TRASH", payload) -} - -// callComposioGmailAPI makes a v2 API call to Composio for Gmail actions -func callComposioGmailAPI(apiKey string, entityID string, action string, payload map[string]interface{}) (string, error) { - // Get connected account ID - connectedAccountID, err := getGmailConnectedAccountID(apiKey, entityID, "gmail") - if err != nil { - return "", fmt.Errorf("failed to get connected account: %w", err) - } - - url := "https://backend.composio.dev/api/v2/actions/" + action + "/execute" - - v2Payload := map[string]interface{}{ - "connectedAccountId": connectedAccountID, - "input": payload["input"], - } - - jsonData, err := json.Marshal(v2Payload) - if err != nil { - return "", fmt.Errorf("failed to marshal request: %w", err) - } - - log.Printf("🔍 [GMAIL] Action: %s, ConnectedAccount: %s", action, maskSensitiveID(connectedAccountID)) - - req, err := http.NewRequest("POST", url, bytes.NewBuffer(jsonData)) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("x-api-key", apiKey) - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to send request: %w", err) - } - defer resp.Body.Close() - - // ✅ SECURITY FIX: Parse and log rate limit headers - parseGmailRateLimitHeaders(resp.Header, action) - - respBody, _ := io.ReadAll(resp.Body) - - if resp.StatusCode >= 400 { - log.Printf("❌ [GMAIL] API error (status %d) for action %s", resp.StatusCode, action) - log.Printf("❌ [GMAIL] Composio error response: %s", string(respBody)) - log.Printf("❌ [GMAIL] Request payload: %s", string(jsonData)) - - // Handle rate limiting with specific error - if resp.StatusCode == 429 { - retryAfter := resp.Header.Get("Retry-After") - if retryAfter != "" { - log.Printf("⚠️ [GMAIL] Rate limited, retry after: %s seconds", retryAfter) - return "", fmt.Errorf("rate limit exceeded, retry after %s seconds", retryAfter) - } - return "", fmt.Errorf("rate limit exceeded, please try again later") - } - - if resp.StatusCode >= 500 { - return "", fmt.Errorf("external service error (status %d)", resp.StatusCode) - } - return "", fmt.Errorf("invalid request (status %d): check parameters and permissions", resp.StatusCode) - } - - var apiResponse map[string]interface{} - if err := json.Unmarshal(respBody, &apiResponse); err != nil { - return string(respBody), nil - } - - result, _ := json.MarshalIndent(apiResponse, "", " ") - return string(result), nil -} - -// getGmailConnectedAccountID retrieves the connected account ID from Composio v3 API -func getGmailConnectedAccountID(apiKey string, userID string, appName string) (string, error) { - baseURL := "https://backend.composio.dev/api/v3/connected_accounts" - params := url.Values{} - params.Add("user_ids", userID) - fullURL := baseURL + "?" + params.Encode() - - req, err := http.NewRequest("GET", fullURL, nil) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("x-api-key", apiKey) - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to fetch connected accounts: %w", err) - } - defer resp.Body.Close() - - respBody, _ := io.ReadAll(resp.Body) - - if resp.StatusCode >= 400 { - return "", fmt.Errorf("Composio API error (status %d): %s", resp.StatusCode, string(respBody)) - } - - // Parse v3 response with proper structure including deprecated.uuid - var response struct { - Items []struct { - ID string `json:"id"` - Toolkit struct { - Slug string `json:"slug"` - } `json:"toolkit"` - Deprecated struct { - UUID string `json:"uuid"` - } `json:"deprecated"` - } `json:"items"` - } - if err := json.Unmarshal(respBody, &response); err != nil { - return "", fmt.Errorf("failed to parse response: %w", err) - } - - // Find the connected account for this app - for _, account := range response.Items { - if account.Toolkit.Slug == appName { - // v2 execution endpoint needs the old UUID, not the new nano ID - // Check if deprecated.uuid exists (for v2 compatibility) - if account.Deprecated.UUID != "" { - return account.Deprecated.UUID, nil - } - // Fall back to nano ID if UUID not available - return account.ID, nil - } - } - - return "", fmt.Errorf("no %s connection found for user. Please connect your Gmail account first", appName) -} - -// stripHTMLAndClean removes HTML tags and cleans up whitespace from text -func stripHTMLAndClean(html string) string { - // Remove HTML tags using regex - re := regexp.MustCompile(`<[^>]*>`) - text := re.ReplaceAllString(html, "") - - // Decode HTML entities like  , &, etc. - text = strings.ReplaceAll(text, " ", " ") - text = strings.ReplaceAll(text, "&", "&") - text = strings.ReplaceAll(text, "<", "<") - text = strings.ReplaceAll(text, ">", ">") - text = strings.ReplaceAll(text, """, "\"") - text = strings.ReplaceAll(text, "'", "'") - text = strings.ReplaceAll(text, "'", "'") - text = strings.ReplaceAll(text, "\u00a0", " ") // Non-breaking space - text = strings.ReplaceAll(text, "\u200b", "") // Zero-width space - text = strings.ReplaceAll(text, "\u200c", "") // Zero-width non-joiner - text = strings.ReplaceAll(text, "\u200d", "") // Zero-width joiner - text = strings.ReplaceAll(text, "\ufeff", "") // Zero-width no-break space - text = strings.ReplaceAll(text, "\r", "") // Remove carriage returns - text = strings.ReplaceAll(text, "\u003e", " ") // Remove greater-than symbol - text = strings.ReplaceAll(text, "\u003c", " ") // Remove less-than symbol - text = strings.ReplaceAll(text, "\u0026", " ") // Remove ampersand symbol - text = strings.ReplaceAll(text, "\u00ab", " ") // Remove left-pointing double angle quotation mark - text = strings.ReplaceAll(text, "\u00bb", " ") // Remove right-pointing double angle quotation mark - text = strings.ReplaceAll(text, "\u0026", "") // Remove ampersand symbol - - - // Remove excessive whitespace - lines := strings.Split(text, "\n") - var cleanedLines []string - for _, line := range lines { - line = strings.TrimSpace(line) - if line != "" { - cleanedLines = append(cleanedLines, line) - } - } - - text = strings.Join(cleanedLines, "\n") - - // Collapse multiple spaces into one - re = regexp.MustCompile(`\s+`) - text = re.ReplaceAllString(text, " ") - - // Final trim - text = strings.TrimSpace(text) - - return text -} - -// simplifyGmailFetchResponse parses the raw Composio Gmail response and returns a simplified, LLM-friendly format -func simplifyGmailFetchResponse(rawResponse string) (string, error) { - var response map[string]interface{} - if err := json.Unmarshal([]byte(rawResponse), &response); err != nil { - // If parsing fails, return raw response - return rawResponse, nil - } - - // Extract the data.messages array - data, ok := response["data"].(map[string]interface{}) - if !ok { - return rawResponse, nil - } - - messages, ok := data["messages"].([]interface{}) - if !ok || len(messages) == 0 { - return "No emails found matching your criteria.", nil - } - - // Build simplified response - simplified := make([]map[string]interface{}, 0, len(messages)) - - for _, msg := range messages { - msgMap, ok := msg.(map[string]interface{}) - if !ok { - continue - } - - simplifiedMsg := make(map[string]interface{}) - - // Extract essential fields - if messageID, ok := msgMap["messageId"].(string); ok { - simplifiedMsg["message_id"] = messageID - } - if threadID, ok := msgMap["threadId"].(string); ok { - simplifiedMsg["thread_id"] = threadID - } - if subject, ok := msgMap["subject"].(string); ok { - simplifiedMsg["subject"] = subject - } - if from, ok := msgMap["from"].(string); ok { - simplifiedMsg["from"] = from - } - if date, ok := msgMap["date"].(string); ok { - simplifiedMsg["date"] = date - } - if snippet, ok := msgMap["snippet"].(string); ok { - simplifiedMsg["snippet"] = snippet - } - - // Extract message text (prefer full text over snippet) - // Strip HTML tags and clean up whitespace - if messageText, ok := msgMap["messageText"].(string); ok && messageText != "" { - simplifiedMsg["message"] = stripHTMLAndClean(messageText) - } else if snippet, ok := msgMap["snippet"].(string); ok { - simplifiedMsg["message"] = snippet - } - - // Include labels only if they contain useful info (skip internal IDs) - if labels, ok := msgMap["labelIds"].([]interface{}); ok { - readableLabels := []string{} - for _, label := range labels { - if labelStr, ok := label.(string); ok { - // Only include readable labels (INBOX, UNREAD, IMPORTANT, etc.) - if labelStr == "INBOX" || labelStr == "UNREAD" || labelStr == "IMPORTANT" || - labelStr == "STARRED" || labelStr == "SENT" || labelStr == "DRAFT" { - readableLabels = append(readableLabels, labelStr) - } - } - } - if len(readableLabels) > 0 { - simplifiedMsg["labels"] = readableLabels - } - } - - simplified = append(simplified, simplifiedMsg) - } - - // Format as JSON for LLM - result, err := json.MarshalIndent(map[string]interface{}{ - "count": len(simplified), - "messages": simplified, - }, "", " ") - - if err != nil { - return rawResponse, nil - } - - return string(result), nil -} - -// parseGmailRateLimitHeaders parses and logs rate limit headers from Gmail API responses -func parseGmailRateLimitHeaders(headers http.Header, action string) { - limit := headers.Get("X-RateLimit-Limit") - remaining := headers.Get("X-RateLimit-Remaining") - reset := headers.Get("X-RateLimit-Reset") - - if limit != "" || remaining != "" || reset != "" { - log.Printf("📊 [GMAIL] Rate limits for %s - Limit: %s, Remaining: %s, Reset: %s", - action, limit, remaining, reset) - - // Warning if approaching rate limit - if remaining != "" && limit != "" { - remainingInt := 0 - limitInt := 0 - fmt.Sscanf(remaining, "%d", &remainingInt) - fmt.Sscanf(limit, "%d", &limitInt) - - if limitInt > 0 { - percentRemaining := float64(remainingInt) / float64(limitInt) * 100 - if percentRemaining < 20 { - log.Printf("⚠️ [GMAIL] Rate limit warning: only %.1f%% remaining (%d/%d)", - percentRemaining, remainingInt, limitInt) - } - } - } - } -} diff --git a/backend/internal/tools/composio_googlesheets_tool.go b/backend/internal/tools/composio_googlesheets_tool.go deleted file mode 100644 index e63f5807..00000000 --- a/backend/internal/tools/composio_googlesheets_tool.go +++ /dev/null @@ -1,1401 +0,0 @@ -package tools - -import ( - "bytes" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "net/url" - "os" - "sync" - "time" -) - -// maskSensitiveID masks a sensitive ID for safe logging (e.g., "acc_abc123xyz" -> "acc_...xyz") -func maskSensitiveID(id string) string { - if len(id) <= 8 { - return "***" - } - return id[:4] + "..." + id[len(id)-4:] -} - -// composioRateLimiter implements per-user rate limiting for Composio API calls -type composioRateLimiter struct { - requests map[string][]time.Time // userID -> timestamps - mutex sync.RWMutex - maxCalls int // max calls per window - window time.Duration // time window -} - -var globalComposioRateLimiter = &composioRateLimiter{ - requests: make(map[string][]time.Time), - maxCalls: 50, // 50 calls per minute per user - window: 1 * time.Minute, -} - -// checkRateLimit checks if user has exceeded rate limit -func (rl *composioRateLimiter) checkRateLimit(userID string) error { - rl.mutex.Lock() - defer rl.mutex.Unlock() - - now := time.Now() - windowStart := now.Add(-rl.window) - - // Get user's request history - timestamps := rl.requests[userID] - - // Remove timestamps outside window - validTimestamps := []time.Time{} - for _, ts := range timestamps { - if ts.After(windowStart) { - validTimestamps = append(validTimestamps, ts) - } - } - - // Check if limit exceeded - if len(validTimestamps) >= rl.maxCalls { - return fmt.Errorf("rate limit exceeded: max %d requests per minute", rl.maxCalls) - } - - // Add current timestamp - validTimestamps = append(validTimestamps, now) - rl.requests[userID] = validTimestamps - - return nil -} - -// checkComposioRateLimit checks rate limit using user ID from args -func checkComposioRateLimit(args map[string]interface{}) error { - // Extract user ID from args (injected by chat service) - userID, ok := args["__user_id__"].(string) - if !ok || userID == "" { - // If no user ID, allow but log warning - log.Printf("⚠️ [COMPOSIO] No user ID for rate limiting") - return nil - } - - return globalComposioRateLimiter.checkRateLimit(userID) -} - -// NewComposioGoogleSheetsReadTool creates a tool for reading Google Sheets via Composio -func NewComposioGoogleSheetsReadTool() *Tool { - return &Tool{ - Name: "googlesheets_read", - DisplayName: "Google Sheets - Read Range", - Description: `Read data from a Google Sheets range via Composio. - -Features: -- Read any range from a spreadsheet (e.g., "Sheet1!A1:D10") -- Returns data as 2D array -- Supports named sheets and ranges -- OAuth authentication handled by Composio - -Use this to fetch data from Google Sheets for processing, analysis, or automation workflows.`, - Icon: "FileSpreadsheet", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"google", "sheets", "spreadsheet", "read", "data", "excel", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "spreadsheet_id": map[string]interface{}{ - "type": "string", - "description": "Google Sheets spreadsheet ID (from the URL)", - }, - "range": map[string]interface{}{ - "type": "string", - "description": "Range to read (e.g., 'Sheet1!A1:D10' or 'Sheet1!A:D')", - }, - }, - "required": []string{"spreadsheet_id", "range"}, - }, - Execute: executeComposioGoogleSheetsRead, - } -} - -// NewComposioGoogleSheetsWriteTool creates a tool for writing to Google Sheets via Composio -func NewComposioGoogleSheetsWriteTool() *Tool { - return &Tool{ - Name: "googlesheets_write", - DisplayName: "Google Sheets - Write Range", - Description: `Write data to a Google Sheets range via Composio. - -Features: -- Write data to specific sheet (overwrites existing data) -- Supports 2D arrays for multiple rows/columns -- Can write formulas and formatted strings (uses USER_ENTERED mode) -- OAuth authentication handled by Composio - -Use this to update Google Sheets with calculated results, API responses, or processed data. - -Note: The range parameter should include the sheet name (e.g., 'Sheet1!A1:D10'). The sheet name will be automatically extracted.`, - Icon: "FileSpreadsheet", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"google", "sheets", "spreadsheet", "write", "update", "data", "excel", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "spreadsheet_id": map[string]interface{}{ - "type": "string", - "description": "Google Sheets spreadsheet ID (from the URL)", - }, - "range": map[string]interface{}{ - "type": "string", - "description": "Sheet name and range to write (e.g., 'Sheet1!A1:D10'). Sheet name is required.", - }, - "values": map[string]interface{}{ - "type": "array", - "description": "2D array of values to write [[row1], [row2], ...] or JSON string", - "items": map[string]interface{}{ - "type": "array", - "items": map[string]interface{}{}, // Allow any type (string, number, boolean, etc.) - }, - }, - }, - "required": []string{"spreadsheet_id", "range", "values"}, - }, - Execute: executeComposioGoogleSheetsWrite, - } -} - -// NewComposioGoogleSheetsAppendTool creates a tool for appending to Google Sheets via Composio -func NewComposioGoogleSheetsAppendTool() *Tool { - return &Tool{ - Name: "googlesheets_append", - DisplayName: "Google Sheets - Append Rows", - Description: `Append rows to a Google Sheets spreadsheet via Composio. - -Features: -- Appends rows to the end of the specified range -- Automatically finds the next empty row -- Supports multiple rows in one operation -- Uses USER_ENTERED mode (formulas are evaluated) -- OAuth authentication handled by Composio - -Use this to add new data without overwriting existing content (logs, form responses, etc.).`, - Icon: "FileSpreadsheet", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"google", "sheets", "spreadsheet", "append", "add", "insert", "data", "excel", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "spreadsheet_id": map[string]interface{}{ - "type": "string", - "description": "Google Sheets spreadsheet ID (from the URL)", - }, - "range": map[string]interface{}{ - "type": "string", - "description": "Sheet name and column range to append to (e.g., 'Sheet1!A:D' or 'Sheet1')", - }, - "values": map[string]interface{}{ - "type": "array", - "description": "2D array of values to append [[row1], [row2], ...] or JSON string", - "items": map[string]interface{}{ - "type": "array", - "items": map[string]interface{}{}, // Allow any type (string, number, boolean, etc.) - }, - }, - }, - "required": []string{"spreadsheet_id", "range", "values"}, - }, - Execute: executeComposioGoogleSheetsAppend, - } -} - -func executeComposioGoogleSheetsRead(args map[string]interface{}) (string, error) { - // ✅ RATE LIMITING - Check per-user rate limit - if err := checkComposioRateLimit(args); err != nil { - return "", err - } - - // Get Composio credentials - credData, err := GetCredentialData(args, "composio_googlesheets") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - // Extract parameters - spreadsheetID, _ := args["spreadsheet_id"].(string) - rangeSpec, _ := args["range"].(string) - - if spreadsheetID == "" { - return "", fmt.Errorf("'spreadsheet_id' is required") - } - if rangeSpec == "" { - return "", fmt.Errorf("'range' is required") - } - - // Call Composio API - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - // Use exact parameter names from Composio docs - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "googlesheets", - "input": map[string]interface{}{ - "spreadsheet_id": spreadsheetID, - "ranges": []string{rangeSpec}, - }, - } - - return callComposioAPI(composioAPIKey, "GOOGLESHEETS_BATCH_GET", payload) -} - -func executeComposioGoogleSheetsWrite(args map[string]interface{}) (string, error) { - // ✅ RATE LIMITING - if err := checkComposioRateLimit(args); err != nil { - return "", err - } - - // Get Composio credentials - credData, err := GetCredentialData(args, "composio_googlesheets") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - // Extract parameters - spreadsheetID, _ := args["spreadsheet_id"].(string) - rangeSpec, _ := args["range"].(string) - values := args["values"] - - if spreadsheetID == "" { - return "", fmt.Errorf("'spreadsheet_id' is required") - } - if rangeSpec == "" { - return "", fmt.Errorf("'range' is required") - } - if values == nil { - return "", fmt.Errorf("'values' is required") - } - - // Parse values if it's a JSON string - var valuesArray [][]interface{} - switch v := values.(type) { - case string: - if err := json.Unmarshal([]byte(v), &valuesArray); err != nil { - return "", fmt.Errorf("failed to parse values JSON: %w", err) - } - case []interface{}: - // Convert to 2D array - for _, row := range v { - if rowArr, ok := row.([]interface{}); ok { - valuesArray = append(valuesArray, rowArr) - } else { - // Single value row - valuesArray = append(valuesArray, []interface{}{row}) - } - } - default: - return "", fmt.Errorf("values must be array or JSON string") - } - - // Extract sheet name from range (e.g., "Sheet1!A1:D10" -> "Sheet1") - sheetName := "Sheet1" - for i := 0; i < len(rangeSpec); i++ { - if rangeSpec[i] == '!' { - sheetName = rangeSpec[:i] - break - } - } - - // Call Composio API - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - // Use exact parameter names from Composio docs for GOOGLESHEETS_BATCH_UPDATE - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "googlesheets", - "input": map[string]interface{}{ - "spreadsheet_id": spreadsheetID, - "sheet_name": sheetName, - "values": valuesArray, - "valueInputOption": "USER_ENTERED", // Default value from docs - }, - } - - return callComposioAPI(composioAPIKey, "GOOGLESHEETS_BATCH_UPDATE", payload) -} - -func executeComposioGoogleSheetsAppend(args map[string]interface{}) (string, error) { - // ✅ RATE LIMITING - if err := checkComposioRateLimit(args); err != nil { - return "", err - } - - // Get Composio credentials - credData, err := GetCredentialData(args, "composio_googlesheets") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - // Extract parameters - spreadsheetID, _ := args["spreadsheet_id"].(string) - rangeSpec, _ := args["range"].(string) - values := args["values"] - - if spreadsheetID == "" { - return "", fmt.Errorf("'spreadsheet_id' is required") - } - if rangeSpec == "" { - return "", fmt.Errorf("'range' is required") - } - if values == nil { - return "", fmt.Errorf("'values' is required") - } - - // Parse values if it's a JSON string - var valuesArray [][]interface{} - switch v := values.(type) { - case string: - if err := json.Unmarshal([]byte(v), &valuesArray); err != nil { - return "", fmt.Errorf("failed to parse values JSON: %w", err) - } - case []interface{}: - // Convert to 2D array - for _, row := range v { - if rowArr, ok := row.([]interface{}); ok { - valuesArray = append(valuesArray, rowArr) - } else { - // Single value row - valuesArray = append(valuesArray, []interface{}{row}) - } - } - default: - return "", fmt.Errorf("values must be array or JSON string") - } - - // Call Composio API - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - // Use exact parameter names from Composio docs for GOOGLESHEETS_SPREADSHEETS_VALUES_APPEND - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "googlesheets", - "input": map[string]interface{}{ - "spreadsheetId": spreadsheetID, - "range": rangeSpec, - "valueInputOption": "USER_ENTERED", // Required by docs - "values": valuesArray, - }, - } - - return callComposioAPI(composioAPIKey, "GOOGLESHEETS_SPREADSHEETS_VALUES_APPEND", payload) -} - -// NewComposioGoogleSheetsCreateTool creates a tool for creating new Google Sheets via Composio -func NewComposioGoogleSheetsCreateTool() *Tool { - return &Tool{ - Name: "googlesheets_create", - DisplayName: "Google Sheets - Create Spreadsheet", - Description: `Create a new Google Sheets spreadsheet via Composio. - -Features: -- Creates a new spreadsheet in Google Drive -- Can specify custom title or use default -- Returns spreadsheet ID and URL -- OAuth authentication handled by Composio - -Use this to create new spreadsheets for data storage, reports, or automation workflows.`, - Icon: "FileSpreadsheet", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"google", "sheets", "spreadsheet", "create", "new", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "title": map[string]interface{}{ - "type": "string", - "description": "Title for the new spreadsheet (optional, defaults to 'Untitled spreadsheet')", - }, - }, - "required": []string{}, - }, - Execute: executeComposioGoogleSheetsCreate, - } -} - -func executeComposioGoogleSheetsCreate(args map[string]interface{}) (string, error) { - // ✅ RATE LIMITING - if err := checkComposioRateLimit(args); err != nil { - return "", err - } - - // Get Composio credentials - credData, err := GetCredentialData(args, "composio_googlesheets") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - // Extract optional title parameter - title, _ := args["title"].(string) - - // Call Composio API - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - // Build payload based on whether title is provided - input := map[string]interface{}{} - if title != "" { - input["title"] = title - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "googlesheets", - "input": input, - } - - return callComposioAPI(composioAPIKey, "GOOGLESHEETS_CREATE_GOOGLE_SHEET1", payload) -} - -// NewComposioGoogleSheetsInfoTool creates a tool for getting spreadsheet metadata via Composio -func NewComposioGoogleSheetsInfoTool() *Tool { - return &Tool{ - Name: "googlesheets_get_info", - DisplayName: "Google Sheets - Get Spreadsheet Info", - Description: `Get comprehensive metadata for a Google Sheets spreadsheet via Composio. - -Features: -- Returns spreadsheet title, locale, timezone -- Lists all sheets/worksheets with their properties -- Gets sheet dimensions and tab colors -- OAuth authentication handled by Composio - -Use this to discover sheet names, understand spreadsheet structure, or validate spreadsheet existence.`, - Icon: "FileSpreadsheet", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"google", "sheets", "spreadsheet", "info", "metadata", "sheets list", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "spreadsheet_id": map[string]interface{}{ - "type": "string", - "description": "Google Sheets spreadsheet ID (from the URL)", - }, - }, - "required": []string{"spreadsheet_id"}, - }, - Execute: executeComposioGoogleSheetsInfo, - } -} - -func executeComposioGoogleSheetsInfo(args map[string]interface{}) (string, error) { - // ✅ RATE LIMITING - if err := checkComposioRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_googlesheets") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - spreadsheetID, _ := args["spreadsheet_id"].(string) - if spreadsheetID == "" { - return "", fmt.Errorf("'spreadsheet_id' is required") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "googlesheets", - "input": map[string]interface{}{ - "spreadsheet_id": spreadsheetID, - }, - } - - return callComposioAPI(composioAPIKey, "GOOGLESHEETS_GET_SPREADSHEET_INFO", payload) -} - -// NewComposioGoogleSheetsListSheetsTool creates a tool for listing sheet names via Composio -func NewComposioGoogleSheetsListSheetsTool() *Tool { - return &Tool{ - Name: "googlesheets_list_sheets", - DisplayName: "Google Sheets - List Sheet Names", - Description: `List all worksheet names in a Google Spreadsheet via Composio. - -Features: -- Returns array of all sheet/tab names in order -- Fast, lightweight operation (no cell data) -- Useful before reading/writing to specific sheets -- OAuth authentication handled by Composio - -Use this to discover available sheets or validate sheet existence before operations.`, - Icon: "FileSpreadsheet", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"google", "sheets", "spreadsheet", "list", "tabs", "worksheets", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "spreadsheet_id": map[string]interface{}{ - "type": "string", - "description": "Google Sheets spreadsheet ID (from the URL)", - }, - }, - "required": []string{"spreadsheet_id"}, - }, - Execute: executeComposioGoogleSheetsListSheets, - } -} - -func executeComposioGoogleSheetsListSheets(args map[string]interface{}) (string, error) { - // ✅ RATE LIMITING - if err := checkComposioRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_googlesheets") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - spreadsheetID, _ := args["spreadsheet_id"].(string) - if spreadsheetID == "" { - return "", fmt.Errorf("'spreadsheet_id' is required") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "googlesheets", - "input": map[string]interface{}{ - "spreadsheet_id": spreadsheetID, - }, - } - - return callComposioAPI(composioAPIKey, "GOOGLESHEETS_GET_SHEET_NAMES", payload) -} - -// NewComposioGoogleSheetsSearchTool creates a tool for searching spreadsheets via Composio -func NewComposioGoogleSheetsSearchTool() *Tool { - return &Tool{ - Name: "googlesheets_search", - DisplayName: "Google Sheets - Search Spreadsheets", - Description: `Search for Google Spreadsheets using filters via Composio. - -Features: -- Search by name, content, or metadata -- Filter by creation/modification date -- Find shared or starred spreadsheets -- Returns spreadsheet IDs and metadata -- OAuth authentication handled by Composio - -Use this to find spreadsheets by name when you don't have the ID, or discover available sheets.`, - Icon: "FileSpreadsheet", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"google", "sheets", "spreadsheet", "search", "find", "discover", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "query": map[string]interface{}{ - "type": "string", - "description": "Search query (searches in name and content)", - }, - "max_results": map[string]interface{}{ - "type": "integer", - "description": "Maximum number of results to return (default: 10)", - }, - }, - "required": []string{}, - }, - Execute: executeComposioGoogleSheetsSearch, - } -} - -func executeComposioGoogleSheetsSearch(args map[string]interface{}) (string, error) { - // ✅ RATE LIMITING - if err := checkComposioRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_googlesheets") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - // Build input parameters - input := map[string]interface{}{} - - if query, ok := args["query"].(string); ok && query != "" { - input["query"] = query - } - - if maxResults, ok := args["max_results"].(float64); ok { - input["max_results"] = int(maxResults) - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "googlesheets", - "input": input, - } - - return callComposioAPI(composioAPIKey, "GOOGLESHEETS_SEARCH_SPREADSHEETS", payload) -} - -// NewComposioGoogleSheetsClearTool creates a tool for clearing cell values via Composio -func NewComposioGoogleSheetsClearTool() *Tool { - return &Tool{ - Name: "googlesheets_clear", - DisplayName: "Google Sheets - Clear Values", - Description: `Clear cell content from a range in Google Sheets via Composio. - -Features: -- Clears cell values but preserves formatting -- Preserves cell notes/comments -- Clears formulas and data -- Supports A1 notation ranges -- OAuth authentication handled by Composio - -Use this to clear data from specific ranges while keeping cell formatting intact.`, - Icon: "FileSpreadsheet", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"google", "sheets", "spreadsheet", "clear", "delete", "erase", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "spreadsheet_id": map[string]interface{}{ - "type": "string", - "description": "Google Sheets spreadsheet ID (from the URL)", - }, - "range": map[string]interface{}{ - "type": "string", - "description": "Range to clear (e.g., 'Sheet1!A1:D10' or 'Sheet1!A:D')", - }, - }, - "required": []string{"spreadsheet_id", "range"}, - }, - Execute: executeComposioGoogleSheetsClear, - } -} - -func executeComposioGoogleSheetsClear(args map[string]interface{}) (string, error) { - // ✅ RATE LIMITING - if err := checkComposioRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_googlesheets") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - spreadsheetID, _ := args["spreadsheet_id"].(string) - rangeSpec, _ := args["range"].(string) - - if spreadsheetID == "" { - return "", fmt.Errorf("'spreadsheet_id' is required") - } - if rangeSpec == "" { - return "", fmt.Errorf("'range' is required") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "googlesheets", - "input": map[string]interface{}{ - "spreadsheet_id": spreadsheetID, - "range": rangeSpec, - }, - } - - return callComposioAPI(composioAPIKey, "GOOGLESHEETS_CLEAR_VALUES", payload) -} - -// NewComposioGoogleSheetsAddSheetTool creates a tool for adding new sheets via Composio -func NewComposioGoogleSheetsAddSheetTool() *Tool { - return &Tool{ - Name: "googlesheets_add_sheet", - DisplayName: "Google Sheets - Add Sheet", - Description: `Add a new worksheet/tab to an existing Google Spreadsheet via Composio. - -Features: -- Creates new sheet within existing spreadsheet -- Can specify sheet title -- Can set initial row/column count -- Can set tab color -- OAuth authentication handled by Composio - -Use this to add new tabs/worksheets to organize data in existing spreadsheets.`, - Icon: "FileSpreadsheet", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"google", "sheets", "spreadsheet", "add", "create", "tab", "worksheet", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "spreadsheet_id": map[string]interface{}{ - "type": "string", - "description": "Google Sheets spreadsheet ID (from the URL)", - }, - "title": map[string]interface{}{ - "type": "string", - "description": "Title for the new sheet (default: 'Sheet{N}')", - }, - }, - "required": []string{"spreadsheet_id"}, - }, - Execute: executeComposioGoogleSheetsAddSheet, - } -} - -func executeComposioGoogleSheetsAddSheet(args map[string]interface{}) (string, error) { - // ✅ RATE LIMITING - if err := checkComposioRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_googlesheets") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - spreadsheetID, _ := args["spreadsheet_id"].(string) - if spreadsheetID == "" { - return "", fmt.Errorf("'spreadsheet_id' is required") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - // Build input with optional title - input := map[string]interface{}{ - "spreadsheetId": spreadsheetID, - } - - // Add optional properties - properties := map[string]interface{}{} - if title, ok := args["title"].(string); ok && title != "" { - properties["title"] = title - } - - if len(properties) > 0 { - input["properties"] = properties - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "googlesheets", - "input": input, - } - - return callComposioAPI(composioAPIKey, "GOOGLESHEETS_ADD_SHEET", payload) -} - -// NewComposioGoogleSheetsDeleteSheetTool creates a tool for deleting sheets via Composio -func NewComposioGoogleSheetsDeleteSheetTool() *Tool { - return &Tool{ - Name: "googlesheets_delete_sheet", - DisplayName: "Google Sheets - Delete Sheet", - Description: `Delete a worksheet/tab from a Google Spreadsheet via Composio. - -Features: -- Permanently removes a sheet from spreadsheet -- Requires sheet ID (numeric ID, not name) -- Cannot delete the last remaining sheet -- OAuth authentication handled by Composio - -Use this to remove unwanted worksheets. Get sheet ID from 'googlesheets_get_info' first. - -WARNING: This action is permanent and cannot be undone!`, - Icon: "FileSpreadsheet", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"google", "sheets", "spreadsheet", "delete", "remove", "tab", "worksheet", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "spreadsheet_id": map[string]interface{}{ - "type": "string", - "description": "Google Sheets spreadsheet ID (from the URL)", - }, - "sheet_id": map[string]interface{}{ - "type": "integer", - "description": "Numeric ID of the sheet to delete (get from googlesheets_get_info)", - }, - }, - "required": []string{"spreadsheet_id", "sheet_id"}, - }, - Execute: executeComposioGoogleSheetsDeleteSheet, - } -} - -func executeComposioGoogleSheetsDeleteSheet(args map[string]interface{}) (string, error) { - // ✅ RATE LIMITING - if err := checkComposioRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_googlesheets") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - spreadsheetID, _ := args["spreadsheet_id"].(string) - if spreadsheetID == "" { - return "", fmt.Errorf("'spreadsheet_id' is required") - } - - // Handle both float64 and int types for sheet_id - var sheetID int - switch v := args["sheet_id"].(type) { - case float64: - sheetID = int(v) - case int: - sheetID = v - default: - return "", fmt.Errorf("'sheet_id' must be a number") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "googlesheets", - "input": map[string]interface{}{ - "spreadsheetId": spreadsheetID, - "sheet_id": sheetID, - }, - } - - return callComposioAPI(composioAPIKey, "GOOGLESHEETS_DELETE_SHEET", payload) -} - -// NewComposioGoogleSheetsFindReplaceTool creates a tool for find and replace via Composio -func NewComposioGoogleSheetsFindReplaceTool() *Tool { - return &Tool{ - Name: "googlesheets_find_replace", - DisplayName: "Google Sheets - Find and Replace", - Description: `Find and replace text in a Google Spreadsheet via Composio. - -Features: -- Find and replace across entire spreadsheet or specific sheets -- Case-sensitive or case-insensitive matching -- Match entire cell or partial content -- Supports regex patterns -- OAuth authentication handled by Composio - -Use this to bulk update values, fix errors, or update formulas across your spreadsheet.`, - Icon: "FileSpreadsheet", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"google", "sheets", "spreadsheet", "find", "replace", "search", "update", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "spreadsheet_id": map[string]interface{}{ - "type": "string", - "description": "Google Sheets spreadsheet ID (from the URL)", - }, - "find": map[string]interface{}{ - "type": "string", - "description": "Text or pattern to find", - }, - "replace": map[string]interface{}{ - "type": "string", - "description": "Text to replace with", - }, - "sheet_id": map[string]interface{}{ - "type": "integer", - "description": "Optional: Numeric sheet ID to limit search (omit for all sheets)", - }, - "match_case": map[string]interface{}{ - "type": "boolean", - "description": "Whether to match case (default: false)", - }, - }, - "required": []string{"spreadsheet_id", "find", "replace"}, - }, - Execute: executeComposioGoogleSheetsFindReplace, - } -} - -func executeComposioGoogleSheetsFindReplace(args map[string]interface{}) (string, error) { - // ✅ RATE LIMITING - if err := checkComposioRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_googlesheets") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - spreadsheetID, _ := args["spreadsheet_id"].(string) - find, _ := args["find"].(string) - replace, _ := args["replace"].(string) - - if spreadsheetID == "" { - return "", fmt.Errorf("'spreadsheet_id' is required") - } - if find == "" { - return "", fmt.Errorf("'find' is required") - } - if replace == "" { - return "", fmt.Errorf("'replace' is required") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - // Build input - input := map[string]interface{}{ - "spreadsheetId": spreadsheetID, - "find": find, - "replace": replace, - } - - // Add optional parameters - if sheetID, ok := args["sheet_id"].(float64); ok { - input["sheetId"] = int(sheetID) - } - if matchCase, ok := args["match_case"].(bool); ok { - input["matchCase"] = matchCase - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "googlesheets", - "input": input, - } - - return callComposioAPI(composioAPIKey, "GOOGLESHEETS_FIND_REPLACE", payload) -} - -// NewComposioGoogleSheetsUpsertRowsTool creates a tool for upserting rows via Composio -func NewComposioGoogleSheetsUpsertRowsTool() *Tool { - return &Tool{ - Name: "googlesheets_upsert_rows", - DisplayName: "Google Sheets - Upsert Rows", - Description: `Smart update/insert rows by matching a key column via Composio. - -Features: -- Updates existing rows by matching key column -- Appends new rows if key not found -- Auto-adds missing columns to sheet -- Supports partial column updates -- Column order doesn't matter (auto-maps by header) -- Prevents duplicates -- OAuth authentication handled by Composio - -Use this for CRM syncs, inventory updates, or any scenario where you want to update existing records or create new ones based on a unique identifier. - -Example: Update contacts by email, inventory by SKU, leads by Lead ID, etc.`, - Icon: "FileSpreadsheet", - Source: ToolSourceComposio, - Category: "integration", - Keywords: []string{"google", "sheets", "spreadsheet", "upsert", "update", "insert", "merge", "sync", "composio"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "spreadsheet_id": map[string]interface{}{ - "type": "string", - "description": "Google Sheets spreadsheet ID (from the URL)", - }, - "sheet_name": map[string]interface{}{ - "type": "string", - "description": "Name of the sheet/tab to upsert into", - }, - "key_column": map[string]interface{}{ - "type": "string", - "description": "Column name to match on (e.g., 'Email', 'SKU', 'Lead ID')", - }, - "rows": map[string]interface{}{ - "type": "array", - "description": "Array of row data arrays [[row1], [row2], ...]", - "items": map[string]interface{}{ - "type": "array", - "items": map[string]interface{}{}, // Allow any type - }, - }, - "headers": map[string]interface{}{ - "type": "array", - "description": "Optional: Array of column headers (if not provided, uses first row of sheet)", - "items": map[string]interface{}{ - "type": "string", - }, - }, - }, - "required": []string{"spreadsheet_id", "sheet_name", "rows"}, - }, - Execute: executeComposioGoogleSheetsUpsertRows, - } -} - -func executeComposioGoogleSheetsUpsertRows(args map[string]interface{}) (string, error) { - // ✅ RATE LIMITING - if err := checkComposioRateLimit(args); err != nil { - return "", err - } - - credData, err := GetCredentialData(args, "composio_googlesheets") - if err != nil { - return "", fmt.Errorf("failed to get Composio credentials: %w", err) - } - - entityID, ok := credData["composio_entity_id"].(string) - if !ok || entityID == "" { - return "", fmt.Errorf("composio_entity_id not found in credentials") - } - - spreadsheetID, _ := args["spreadsheet_id"].(string) - sheetName, _ := args["sheet_name"].(string) - rows := args["rows"] - - if spreadsheetID == "" { - return "", fmt.Errorf("'spreadsheet_id' is required") - } - if sheetName == "" { - return "", fmt.Errorf("'sheet_name' is required") - } - if rows == nil { - return "", fmt.Errorf("'rows' is required") - } - - composioAPIKey := os.Getenv("COMPOSIO_API_KEY") - if composioAPIKey == "" { - return "", fmt.Errorf("COMPOSIO_API_KEY environment variable not set") - } - - // Build input - input := map[string]interface{}{ - "spreadsheetId": spreadsheetID, - "sheetName": sheetName, - "rows": rows, - } - - // Add optional parameters - if keyColumn, ok := args["key_column"].(string); ok && keyColumn != "" { - input["keyColumn"] = keyColumn - } - if headers, ok := args["headers"].([]interface{}); ok && len(headers) > 0 { - input["headers"] = headers - } - - payload := map[string]interface{}{ - "entityId": entityID, - "appName": "googlesheets", - "input": input, - } - - return callComposioAPI(composioAPIKey, "GOOGLESHEETS_UPSERT_ROWS", payload) -} - -// getConnectedAccountID retrieves the connected account ID from Composio v3 API -func getConnectedAccountID(apiKey string, userID string, appName string) (string, error) { - // Query v3 API to get connected accounts for this user (URL-safe to prevent injection) - baseURL := "https://backend.composio.dev/api/v3/connected_accounts" - params := url.Values{} - params.Add("user_ids", userID) - fullURL := baseURL + "?" + params.Encode() - - req, err := http.NewRequest("GET", fullURL, nil) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("x-api-key", apiKey) - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to send request: %w", err) - } - defer resp.Body.Close() - - respBody, _ := io.ReadAll(resp.Body) - - if resp.StatusCode >= 400 { - return "", fmt.Errorf("Composio API error (status %d): %s", resp.StatusCode, string(respBody)) - } - - // Parse v3 response - var response struct { - Items []struct { - ID string `json:"id"` - Toolkit struct { - Slug string `json:"slug"` - } `json:"toolkit"` - Deprecated struct { - UUID string `json:"uuid"` - } `json:"deprecated"` - } `json:"items"` - } - - if err := json.Unmarshal(respBody, &response); err != nil { - return "", fmt.Errorf("failed to parse response: %w", err) - } - - // Find the connected account for this app - for _, account := range response.Items { - if account.Toolkit.Slug == appName { - // v2 execution endpoint needs the old UUID, not the new nano ID - // Check if deprecated.uuid exists (for v2 compatibility) - if account.Deprecated.UUID != "" { - return account.Deprecated.UUID, nil - } - // Fall back to nano ID if UUID not available - return account.ID, nil - } - } - - return "", fmt.Errorf("no connected account found for app '%s' and user '%s'", appName, userID) -} - -// callComposioAPI makes a request to Composio's v3 API -func callComposioAPI(apiKey string, action string, payload map[string]interface{}) (string, error) { - // v2 execution endpoint still works with v3 connected accounts - url := "https://backend.composio.dev/api/v2/actions/" + action + "/execute" - - // Get params from payload - entityID, _ := payload["entityId"].(string) - appName, _ := payload["appName"].(string) - input, _ := payload["input"].(map[string]interface{}) - - // For v3, we need to find the connected account ID - connectedAccountID, err := getConnectedAccountID(apiKey, entityID, appName) - if err != nil { - return "", fmt.Errorf("failed to get connected account ID: %w", err) - } - - // Build v2 payload (v2 execution endpoint uses connectedAccountId with camelCase) - v2Payload := map[string]interface{}{ - "connectedAccountId": connectedAccountID, - "input": input, - } - - jsonData, err := json.Marshal(v2Payload) - if err != nil { - return "", fmt.Errorf("failed to marshal request: %w", err) - } - - // ✅ SECURE LOGGING - Only log non-sensitive metadata - log.Printf("🔍 [COMPOSIO] Action: %s, ConnectedAccount: %s", action, maskSensitiveID(connectedAccountID)) - - req, err := http.NewRequest("POST", url, bytes.NewBuffer(jsonData)) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("x-api-key", apiKey) - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to send request: %w", err) - } - defer resp.Body.Close() - - // ✅ SECURITY FIX: Parse and log rate limit headers - parseRateLimitHeaders(resp.Header, action) - - respBody, _ := io.ReadAll(resp.Body) - - if resp.StatusCode >= 400 { - // ✅ SECURE ERROR HANDLING - Log full details server-side, sanitize for user - log.Printf("❌ [COMPOSIO] API error (status %d) for action %s", resp.StatusCode, action) - - // Handle rate limiting with specific error - if resp.StatusCode == 429 { - retryAfter := resp.Header.Get("Retry-After") - if retryAfter != "" { - log.Printf("⚠️ [COMPOSIO] Rate limited, retry after: %s seconds", retryAfter) - return "", fmt.Errorf("rate limit exceeded, retry after %s seconds", retryAfter) - } - return "", fmt.Errorf("rate limit exceeded, please try again later") - } - - // Don't expose internal Composio error details to users - if resp.StatusCode >= 500 { - return "", fmt.Errorf("external service error (status %d)", resp.StatusCode) - } - // Client errors (4xx) can be slightly more specific - return "", fmt.Errorf("invalid request (status %d): check spreadsheet ID and permissions", resp.StatusCode) - } - - // Parse response - var apiResponse map[string]interface{} - if err := json.Unmarshal(respBody, &apiResponse); err != nil { - return string(respBody), nil - } - - // Return formatted response - result, _ := json.MarshalIndent(apiResponse, "", " ") - return string(result), nil -} - -// parseRateLimitHeaders parses and logs rate limit headers from Composio API responses -func parseRateLimitHeaders(headers http.Header, action string) { - limit := headers.Get("X-RateLimit-Limit") - remaining := headers.Get("X-RateLimit-Remaining") - reset := headers.Get("X-RateLimit-Reset") - - if limit != "" || remaining != "" || reset != "" { - log.Printf("📊 [COMPOSIO] Rate limits for %s - Limit: %s, Remaining: %s, Reset: %s", - action, limit, remaining, reset) - - // Warning if approaching rate limit - if remaining != "" && limit != "" { - remainingInt := 0 - limitInt := 0 - fmt.Sscanf(remaining, "%d", &remainingInt) - fmt.Sscanf(limit, "%d", &limitInt) - - if limitInt > 0 { - percentRemaining := float64(remainingInt) / float64(limitInt) * 100 - if percentRemaining < 20 { - log.Printf("⚠️ [COMPOSIO] Rate limit warning: only %.1f%% remaining (%d/%d)", - percentRemaining, remainingInt, limitInt) - } - } - } - } -} diff --git a/backend/internal/tools/credential_helper.go b/backend/internal/tools/credential_helper.go deleted file mode 100644 index 20345aaf..00000000 --- a/backend/internal/tools/credential_helper.go +++ /dev/null @@ -1,138 +0,0 @@ -package tools - -import ( - "fmt" - - "claraverse/internal/models" -) - -// CredentialResolver is a function type for resolving credentials at runtime -// This is injected into tool args to provide access to credentials without -// exposing the credential service directly to tools -type CredentialResolver func(credentialID string) (*models.DecryptedCredential, error) - -// ContextKey for passing credential resolver through args -const CredentialResolverKey = "__credential_resolver__" -const UserIDKey = "__user_id__" - -// Note: CreateCredentialResolver is defined in services/credential_service.go -// to avoid import cycles (services imports tools, so tools cannot import services) - -// ResolveWebhookURL resolves a webhook URL from either direct URL or credential ID -// Priority: 1. Direct webhook_url parameter, 2. credential_id lookup -func ResolveWebhookURL(args map[string]interface{}, integrationType string) (string, error) { - // First, check for direct webhook_url - if webhookURL, ok := args["webhook_url"].(string); ok && webhookURL != "" { - return webhookURL, nil - } - - // Check for credential_id - credentialID, hasCredID := args["credential_id"].(string) - if !hasCredID || credentialID == "" { - return "", fmt.Errorf("either webhook_url or credential_id is required") - } - - // Get credential resolver from args - resolver, ok := args[CredentialResolverKey].(CredentialResolver) - if !ok || resolver == nil { - return "", fmt.Errorf("credential resolver not available") - } - - // Resolve the credential - cred, err := resolver(credentialID) - if err != nil { - return "", fmt.Errorf("failed to resolve credential: %w", err) - } - - // Verify integration type matches - if cred.IntegrationType != integrationType { - return "", fmt.Errorf("credential type mismatch: expected %s, got %s", integrationType, cred.IntegrationType) - } - - // Extract webhook URL from credential data - webhookURL, ok := cred.Data["webhook_url"].(string) - if !ok || webhookURL == "" { - // Try alternate key names - if url, ok := cred.Data["url"].(string); ok && url != "" { - webhookURL = url - } else { - return "", fmt.Errorf("credential does not contain a valid webhook URL") - } - } - - return webhookURL, nil -} - -// ResolveAPIKey resolves an API key from either direct parameter or credential ID -func ResolveAPIKey(args map[string]interface{}, integrationType string, keyFieldName string) (string, error) { - // First, check for direct API key - if apiKey, ok := args[keyFieldName].(string); ok && apiKey != "" { - return apiKey, nil - } - - // Check for credential_id - credentialID, hasCredID := args["credential_id"].(string) - if !hasCredID || credentialID == "" { - return "", fmt.Errorf("either %s or credential_id is required", keyFieldName) - } - - // Get credential resolver from args - resolver, ok := args[CredentialResolverKey].(CredentialResolver) - if !ok || resolver == nil { - return "", fmt.Errorf("credential resolver not available") - } - - // Resolve the credential - cred, err := resolver(credentialID) - if err != nil { - return "", fmt.Errorf("failed to resolve credential: %w", err) - } - - // Verify integration type matches - if cred.IntegrationType != integrationType { - return "", fmt.Errorf("credential type mismatch: expected %s, got %s", integrationType, cred.IntegrationType) - } - - // Extract API key from credential data - apiKey, ok := cred.Data[keyFieldName].(string) - if !ok || apiKey == "" { - // Try alternate key names - if key, ok := cred.Data["api_key"].(string); ok && key != "" { - apiKey = key - } else if key, ok := cred.Data["token"].(string); ok && key != "" { - apiKey = key - } else { - return "", fmt.Errorf("credential does not contain a valid API key") - } - } - - return apiKey, nil -} - -// GetCredentialData retrieves all data from a credential by ID -func GetCredentialData(args map[string]interface{}, integrationType string) (map[string]interface{}, error) { - // Check for credential_id - credentialID, hasCredID := args["credential_id"].(string) - if !hasCredID || credentialID == "" { - return nil, fmt.Errorf("credential_id is required") - } - - // Get credential resolver from args - resolver, ok := args[CredentialResolverKey].(CredentialResolver) - if !ok || resolver == nil { - return nil, fmt.Errorf("credential resolver not available") - } - - // Resolve the credential - cred, err := resolver(credentialID) - if err != nil { - return nil, fmt.Errorf("failed to resolve credential: %w", err) - } - - // Verify integration type matches (if provided) - if integrationType != "" && cred.IntegrationType != integrationType { - return nil, fmt.Errorf("credential type mismatch: expected %s, got %s", integrationType, cred.IntegrationType) - } - - return cred.Data, nil -} diff --git a/backend/internal/tools/data_analyst_tool.go b/backend/internal/tools/data_analyst_tool.go deleted file mode 100644 index 4451afa8..00000000 --- a/backend/internal/tools/data_analyst_tool.go +++ /dev/null @@ -1,500 +0,0 @@ -package tools - -import ( - "context" - "encoding/json" - "fmt" - "log" - "regexp" - "strings" - - "claraverse/internal/e2b" -) - -// stripDataLoadingCalls removes pd.read_csv(), pd.read_excel(), pd.read_json() calls from user code -// This prevents LLMs from trying to load files by filename (which don't exist in the sandbox) -func stripDataLoadingCalls(code string) string { - // Patterns to match various forms of data loading - patterns := []string{ - // Match: df = pd.read_csv('filename.csv') or similar with double quotes - `(?m)^\s*\w+\s*=\s*pd\.read_csv\s*\([^)]+\)\s*$`, - `(?m)^\s*\w+\s*=\s*pd\.read_excel\s*\([^)]+\)\s*$`, - `(?m)^\s*\w+\s*=\s*pd\.read_json\s*\([^)]+\)\s*$`, - `(?m)^\s*\w+\s*=\s*pd\.read_table\s*\([^)]+\)\s*$`, - // Match inline read calls (not assigned) - `(?m)^\s*pd\.read_csv\s*\([^)]+\)\s*$`, - `(?m)^\s*pd\.read_excel\s*\([^)]+\)\s*$`, - `(?m)^\s*pd\.read_json\s*\([^)]+\)\s*$`, - // Match with pandas prefix - `(?m)^\s*\w+\s*=\s*pandas\.read_csv\s*\([^)]+\)\s*$`, - `(?m)^\s*\w+\s*=\s*pandas\.read_excel\s*\([^)]+\)\s*$`, - } - - result := code - stripped := false - for _, pattern := range patterns { - re := regexp.MustCompile(pattern) - if re.MatchString(result) { - stripped = true - result = re.ReplaceAllString(result, "# [AUTO-REMOVED: Data is pre-loaded as 'df']") - } - } - - if stripped { - log.Printf("🔧 [DATA-ANALYST] Stripped data loading calls from user code") - } - - return result -} - -// NewDataAnalystTool creates a new AI Data Analyst tool -func NewDataAnalystTool() *Tool { - return &Tool{ - Name: "analyze_data", - DisplayName: "AI Data Analyst", - Description: `Analyze data with Python. Full access to pandas, numpy, matplotlib, and seaborn. - -⚠️ IMPORTANT: Data is AUTOMATICALLY loaded as 'df' (pandas DataFrame). -DO NOT use pd.read_csv(), pd.read_excel(), or pd.read_json() - it will fail! -Just use 'df' directly in your code. - -Example usage: -- df.head() - view data -- df.describe() - statistics -- sns.barplot(data=df, x='category', y='sales') - bar chart -- df.plot() - line plot - -Chart types you can create: -- Bar: sns.barplot(data=df, x='col1', y='col2') -- Pie: plt.pie(df.groupby('cat')['val'].sum(), labels=..., autopct='%1.1f%%') -- Scatter: sns.scatterplot(data=df, x='col1', y='col2', hue='col3') -- Heatmap: sns.heatmap(df.corr(), annot=True, cmap='coolwarm') -- Histogram: sns.histplot(df['col'], bins=30) -- Box: sns.boxplot(data=df, x='category', y='value') - -Always use plt.show() after each plot. Add titles and labels for clarity.`, - Icon: "ChartBar", - Source: ToolSourceBuiltin, - Category: "computation", - Keywords: []string{"analyze", "data", "python", "pandas", "visualization", "chart", "graph", "statistics", "csv", "dataframe", "plot", "analytics"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "file_id": map[string]interface{}{ - "type": "string", - "description": "Upload ID of the CSV/Excel file to analyze (from file upload). Use this for uploaded files.", - }, - "csv_data": map[string]interface{}{ - "type": "string", - "description": "CSV data as a string (with headers). Use this for small inline data. Either file_id or csv_data is required.", - }, - "python_code": map[string]interface{}{ - "type": "string", - "description": `Custom Python code for visualization. - -⚠️ CRITICAL: 'df' is ALREADY LOADED as a pandas DataFrame! -DO NOT use pd.read_csv(), pd.read_excel(), or pd.read_json()! -The file is NOT accessible by filename in the sandbox. - -Just write code that uses 'df' directly: -- sns.barplot(data=df, x='category', y='sales') -- plt.pie(df.groupby('cat')['val'].sum(), labels=df['cat'].unique()) -- sns.heatmap(df.corr(), annot=True) - -Always end with plt.show()`, - }, - "analysis_type": map[string]interface{}{ - "type": "string", - "description": "Predefined analysis type (used only if python_code is not provided)", - "enum": []string{"summary", "correlation", "trend", "distribution", "outliers", "full"}, - "default": "summary", - }, - "columns": map[string]interface{}{ - "type": "array", - "description": "Optional: Specific columns to analyze (if empty, analyzes all columns)", - "items": map[string]interface{}{ - "type": "string", - }, - }, - }, - }, - Execute: executeDataAnalyst, - } -} - -func executeDataAnalyst(args map[string]interface{}) (string, error) { - var csvData []byte - var filename string - - // Try file_id first (uploaded files) - if fileID, ok := args["file_id"].(string); ok && fileID != "" { - content, name, err := GetUploadedFile(fileID) - if err != nil { - return "", fmt.Errorf("failed to get uploaded file: %w", err) - } - csvData = content - filename = name - } else if csvDataStr, ok := args["csv_data"].(string); ok && csvDataStr != "" { - // Fallback to direct CSV data for small inline data - csvData = []byte(csvDataStr) - filename = "data.csv" - } else { - return "", fmt.Errorf("either file_id or csv_data is required") - } - - // Create file map with CSV data - files := map[string][]byte{ - filename: csvData, - } - - var pythonCode string - - // Check for custom python_code first (LLM-generated visualizations) - if customCode, ok := args["python_code"].(string); ok && customCode != "" { - // Strip any pd.read_* calls - the LLM shouldn't load files, we pre-load them - cleanedCode := stripDataLoadingCalls(customCode) - - // Use LLM-generated code with pre-loaded data - pythonCode = fmt.Sprintf(`import pandas as pd -import numpy as np -import matplotlib.pyplot as plt -import seaborn as sns - -# Set plot style -plt.style.use('seaborn-v0_8-darkgrid') -sns.set_palette("husl") - -# Load data -df = pd.read_csv('%s') - -print("=" * 80) -print("DATA ANALYSIS") -print("=" * 80) -print(f"\nDataset: %s") -print(f"Shape: {df.shape[0]} rows × {df.shape[1]} columns") -print(f"Columns: {list(df.columns)}") -print() - -# Execute custom analysis code -%s - -print("\n" + "=" * 80) -print("✅ ANALYSIS COMPLETE") -print("=" * 80) -`, filename, filename, cleanedCode) - } else { - // Fallback to predefined analysis types - analysisType := "summary" - if at, ok := args["analysis_type"].(string); ok { - analysisType = at - } - - var columns []string - if colsRaw, ok := args["columns"].([]interface{}); ok { - for _, col := range colsRaw { - columns = append(columns, fmt.Sprintf("%v", col)) - } - } - - pythonCode = generateDataAnalysisCode([]string{filename}, analysisType, columns) - } - - // Execute code with longer timeout for custom visualizations - e2bService := e2b.GetE2BExecutorService() - result, err := e2bService.ExecuteWithFiles(context.Background(), pythonCode, files, 120) - if err != nil { - return "", fmt.Errorf("failed to execute analysis: %w", err) - } - - if !result.Success { - errorMsg := result.Stderr - if result.Error != nil { - errorMsg = *result.Error - } - - // Check for FileNotFoundError and provide helpful message - if strings.Contains(errorMsg, "FileNotFoundError") || strings.Contains(errorMsg, "No such file or directory") { - return "", fmt.Errorf(`analysis failed: %s - -💡 HINT: The data is already pre-loaded as 'df' (pandas DataFrame). -Do NOT use pd.read_csv(), pd.read_excel(), or pd.read_json() - those files don't exist in the sandbox! -Just use 'df' directly in your code. Example: sns.barplot(data=df, x='category', y='sales')`, errorMsg) - } - - return "", fmt.Errorf("analysis failed: %s", errorMsg) - } - - // Format response - response := map[string]interface{}{ - "success": true, - "analysis": result.Stdout, - "plots": result.Plots, - "plot_count": len(result.Plots), - "filename": filename, - } - - jsonResponse, _ := json.MarshalIndent(response, "", " ") - return string(jsonResponse), nil -} - -func generateDataAnalysisCode(fileNames []string, analysisType string, columns []string) string { - // Determine the primary file - primaryFile := fileNames[0] - - // Column filter - colFilter := "" - if len(columns) > 0 { - colsStr := "'" + strings.Join(columns, "', '") + "'" - colFilter = fmt.Sprintf("\ndf = df[[%s]]", colsStr) - } - - // Base code - code := fmt.Sprintf(`import pandas as pd -import matplotlib.pyplot as plt -import numpy as np -import seaborn as sns - -# Set plot style -plt.style.use('seaborn-v0_8-darkgrid') -sns.set_palette("husl") - -# Load data -df = pd.read_csv('%s')%s - -print("=" * 80) -print("DATA ANALYSIS REPORT") -print("=" * 80) -print(f"\nDataset: %s") -print(f"Shape: {df.shape[0]} rows × {df.shape[1]} columns") -print() -`, primaryFile, colFilter, primaryFile) - - switch analysisType { - case "summary": - code += ` -# Summary Statistics -print("\n📊 SUMMARY STATISTICS") -print("-" * 80) -print(df.describe()) - -# Data types -print("\n📋 DATA TYPES") -print("-" * 80) -print(df.dtypes) - -# Missing values -print("\n⚠️ MISSING VALUES") -print("-" * 80) -missing = df.isnull().sum() -if missing.sum() > 0: - print(missing[missing > 0]) -else: - print("No missing values!") -` - - case "correlation": - code += ` -# Correlation Analysis -print("\n🔗 CORRELATION ANALYSIS") -print("-" * 80) - -numeric_cols = df.select_dtypes(include=[np.number]).columns -if len(numeric_cols) > 1: - corr = df[numeric_cols].corr() - print(corr) - - # Correlation heatmap - plt.figure(figsize=(10, 8)) - sns.heatmap(corr, annot=True, cmap='coolwarm', center=0, fmt='.2f') - plt.title('Correlation Matrix') - plt.tight_layout() - plt.show() -else: - print("Not enough numeric columns for correlation analysis") -` - - case "trend": - code += ` -# Trend Analysis -print("\n📈 TREND ANALYSIS") -print("-" * 80) - -numeric_cols = df.select_dtypes(include=[np.number]).columns -if len(numeric_cols) > 0: - # Line plot for numeric columns - fig, axes = plt.subplots(len(numeric_cols), 1, figsize=(12, 4 * len(numeric_cols))) - if len(numeric_cols) == 1: - axes = [axes] - - for ax, col in zip(axes, numeric_cols): - df[col].plot(ax=ax, linewidth=2) - ax.set_title(f'{col} - Trend Over Time') - ax.set_xlabel('Index') - ax.set_ylabel(col) - ax.grid(True, alpha=0.3) - - plt.tight_layout() - plt.show() - - print(f"Analyzed trends for {len(numeric_cols)} numeric columns") -else: - print("No numeric columns for trend analysis") -` - - case "distribution": - code += ` -# Distribution Analysis -print("\n📊 DISTRIBUTION ANALYSIS") -print("-" * 80) - -numeric_cols = df.select_dtypes(include=[np.number]).columns -if len(numeric_cols) > 0: - # Create subplots - n_cols = min(3, len(numeric_cols)) - n_rows = (len(numeric_cols) + n_cols - 1) // n_cols - - fig, axes = plt.subplots(n_rows, n_cols, figsize=(5 * n_cols, 4 * n_rows)) - if len(numeric_cols) == 1: - axes = [axes] - else: - axes = axes.flatten() if n_rows > 1 else axes - - for ax, col in zip(axes, numeric_cols): - df[col].hist(ax=ax, bins=30, edgecolor='black', alpha=0.7) - ax.set_title(f'{col} Distribution') - ax.set_xlabel(col) - ax.set_ylabel('Frequency') - ax.grid(True, alpha=0.3) - - # Hide empty subplots - for i in range(len(numeric_cols), len(axes)): - axes[i].set_visible(False) - - plt.tight_layout() - plt.show() - - # Print statistics - for col in numeric_cols: - print(f"\n{col}:") - print(f" Mean: {df[col].mean():.2f}") - print(f" Median: {df[col].median():.2f}") - print(f" Std Dev: {df[col].std():.2f}") - print(f" Min: {df[col].min():.2f}, Max: {df[col].max():.2f}") -else: - print("No numeric columns for distribution analysis") -` - - case "outliers": - code += ` -# Outlier Detection -print("\n🚨 OUTLIER DETECTION") -print("-" * 80) - -numeric_cols = df.select_dtypes(include=[np.number]).columns -if len(numeric_cols) > 0: - for col in numeric_cols: - Q1 = df[col].quantile(0.25) - Q3 = df[col].quantile(0.75) - IQR = Q3 - Q1 - lower_bound = Q1 - 1.5 * IQR - upper_bound = Q3 + 1.5 * IQR - - outliers = df[(df[col] < lower_bound) | (df[col] > upper_bound)] - - print(f"\n{col}:") - print(f" Lower bound: {lower_bound:.2f}") - print(f" Upper bound: {upper_bound:.2f}") - print(f" Outliers found: {len(outliers)}") - - if len(outliers) > 0: - print(f" Outlier values: {sorted(outliers[col].unique())[:10]}") - - # Box plot - fig, axes = plt.subplots(len(numeric_cols), 1, figsize=(10, 3 * len(numeric_cols))) - if len(numeric_cols) == 1: - axes = [axes] - - for ax, col in zip(axes, numeric_cols): - df.boxplot(column=col, ax=ax, vert=False) - ax.set_title(f'{col} - Box Plot (Outlier Detection)') - ax.grid(True, alpha=0.3) - - plt.tight_layout() - plt.show() -else: - print("No numeric columns for outlier detection") -` - - case "full": - code += ` -# Full Analysis -print("\n📊 SUMMARY STATISTICS") -print("-" * 80) -print(df.describe()) - -print("\n📋 DATA TYPES") -print("-" * 80) -print(df.dtypes) - -print("\n⚠️ MISSING VALUES") -print("-" * 80) -missing = df.isnull().sum() -if missing.sum() > 0: - print(missing[missing > 0]) -else: - print("✅ No missing values!") - -numeric_cols = df.select_dtypes(include=[np.number]).columns - -if len(numeric_cols) > 1: - # Correlation heatmap - print("\n🔗 CORRELATION ANALYSIS") - print("-" * 80) - corr = df[numeric_cols].corr() - print(corr) - - plt.figure(figsize=(10, 8)) - sns.heatmap(corr, annot=True, cmap='coolwarm', center=0, fmt='.2f', - square=True, linewidths=0.5) - plt.title('Correlation Matrix', fontsize=16, fontweight='bold') - plt.tight_layout() - plt.show() - -if len(numeric_cols) > 0: - # Distribution plots - print("\n📊 DISTRIBUTION ANALYSIS") - print("-" * 80) - - n_cols = min(3, len(numeric_cols)) - n_rows = (len(numeric_cols) + n_cols - 1) // n_cols - - fig, axes = plt.subplots(n_rows, n_cols, figsize=(5 * n_cols, 4 * n_rows)) - if len(numeric_cols) == 1: - axes = [axes] - else: - axes = axes.flatten() if n_rows > 1 else axes - - for ax, col in zip(axes, numeric_cols): - df[col].hist(ax=ax, bins=30, edgecolor='black', alpha=0.7, color='skyblue') - ax.set_title(f'{col} Distribution', fontweight='bold') - ax.set_xlabel(col) - ax.set_ylabel('Frequency') - ax.grid(True, alpha=0.3) - - for i in range(len(numeric_cols), len(axes)): - axes[i].set_visible(False) - - plt.tight_layout() - plt.show() - - for col in numeric_cols: - print(f"{col}: μ={df[col].mean():.2f}, σ={df[col].std():.2f}") - -print("\n" + "=" * 80) -print("✅ ANALYSIS COMPLETE") -print("=" * 80) -` - } - - return code -} diff --git a/backend/internal/tools/describe_image_tool.go b/backend/internal/tools/describe_image_tool.go deleted file mode 100644 index 6f18b310..00000000 --- a/backend/internal/tools/describe_image_tool.go +++ /dev/null @@ -1,372 +0,0 @@ -package tools - -import ( - "claraverse/internal/filecache" - "claraverse/internal/vision" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "os" - "path/filepath" - "strings" - "time" -) - -// NewDescribeImageTool creates the describe_image tool for AI image analysis -func NewDescribeImageTool() *Tool { - return &Tool{ - Name: "describe_image", - DisplayName: "Describe Image", - Description: `Analyzes an image using AI vision and returns a detailed text description. - -Use this tool when the user asks you to: -- Describe what's in an image -- Analyze the content of a picture -- Answer questions about an image -- Identify objects, people, or text in an image - -Parameters: -- image_url: A direct URL to an image on the web (e.g., "https://example.com/image.jpg"). Supports http/https URLs. -- image_id: The image handle (e.g., "img-1") from the available images list. Use this for generated or previously referenced images. -- file_id: Alternative - use the direct file ID from an upload response -- question: Optional specific question about the image -- detail: "brief" for 1-2 sentences, "detailed" for comprehensive description - -You must provide one of: image_url, image_id, OR file_id. Use image_url for web images, image_id for generated/edited images, file_id for uploaded files.`, - Icon: "Image", - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "image_url": map[string]interface{}{ - "type": "string", - "description": "A direct URL to an image on the web (e.g., 'https://example.com/image.jpg'). Use this to analyze images from the internet.", - }, - "image_id": map[string]interface{}{ - "type": "string", - "description": "The image handle (e.g., 'img-1') from the available images list. Preferred for generated or edited images.", - }, - "file_id": map[string]interface{}{ - "type": "string", - "description": "Alternative: The direct file ID of the uploaded image. Use image_id when available.", - }, - "question": map[string]interface{}{ - "type": "string", - "description": "Optional specific question about the image (e.g., 'What color is the car?', 'How many people are in this photo?')", - }, - "detail": map[string]interface{}{ - "type": "string", - "enum": []string{"brief", "detailed"}, - "description": "Level of detail: 'brief' for 1-2 sentences, 'detailed' for comprehensive description. Default is 'detailed'", - }, - }, - "required": []string{}, - }, - Execute: executeDescribeImage, - Source: ToolSourceBuiltin, - Category: "data_sources", - Keywords: []string{"image", "describe", "analyze", "vision", "picture", "photo", "screenshot", "diagram", "chart", "url"}, - } -} - -// Constants for URL image fetching -const ( - describeImageMaxSize = 20 * 1024 * 1024 // 20MB for images - describeImageTimeout = 30 * time.Second -) - -func executeDescribeImage(args map[string]interface{}) (string, error) { - // Extract image_url, image_id (handle like "img-1") or file_id (direct UUID) - imageURL, hasImageURL := args["image_url"].(string) - imageID, hasImageID := args["image_id"].(string) - fileID, hasFileID := args["file_id"].(string) - - if (!hasImageURL || imageURL == "") && (!hasImageID || imageID == "") && (!hasFileID || fileID == "") { - return "", fmt.Errorf("one of image_url, image_id, or file_id is required. Use image_url for web images, image_id (e.g., 'img-1') for generated images, or file_id for uploaded files") - } - - // Extract optional question parameter - question := "" - if q, ok := args["question"].(string); ok { - question = q - } - - // Extract detail level (default to "detailed") - detail := "detailed" - if d, ok := args["detail"].(string); ok && (d == "brief" || d == "detailed") { - detail = d - } - - // Extract user context (injected by tool executor) - userID, _ := args["__user_id__"].(string) - convID, _ := args["__conversation_id__"].(string) - - // Variables to hold image data and metadata - var imageData []byte - var mimeType string - var filename string - var resolvedFileID string - var sourceURL string - - // If image_url is provided, fetch the image directly from the web - if hasImageURL && imageURL != "" { - log.Printf("🖼️ [DESCRIBE-IMAGE] Fetching image from URL: %s", imageURL) - - data, mime, fname, err := fetchImageFromURL(imageURL) - if err != nil { - log.Printf("❌ [DESCRIBE-IMAGE] Failed to fetch image from URL: %v", err) - return "", fmt.Errorf("failed to fetch image from URL: %v", err) - } - - imageData = data - mimeType = mime - filename = fname - sourceURL = imageURL - resolvedFileID = "url-image" - - log.Printf("✅ [DESCRIBE-IMAGE] Fetched image from URL: %s (%d bytes, %s)", filename, len(imageData), mimeType) - } else { - // Get file cache service for image_id or file_id - fileCacheService := filecache.GetService() - var file *filecache.CachedFile - - // If image_id is provided, resolve it via the registry - if hasImageID && imageID != "" { - // Get image registry (injected by chat_service) - registry, ok := args[ImageRegistryKey].(ImageRegistryInterface) - if !ok || registry == nil { - // Registry not available - try to use image_id as file_id fallback - log.Printf("⚠️ [DESCRIBE-IMAGE] Image registry not available, treating image_id as file_id") - resolvedFileID = imageID - } else { - // Look up the image by handle - entry := registry.GetByHandle(convID, imageID) - if entry == nil { - // Provide helpful error message with available handles - handles := registry.ListHandles(convID) - if len(handles) == 0 { - return "", fmt.Errorf("image '%s' not found. No images are available in this conversation. Please upload an image first or use file_id for direct file access", imageID) - } - return "", fmt.Errorf("image '%s' not found. Available images: %s", imageID, strings.Join(handles, ", ")) - } - resolvedFileID = entry.FileID - log.Printf("🖼️ [DESCRIBE-IMAGE] Resolved image_id '%s' to file_id '%s'", imageID, resolvedFileID) - } - } else { - // Use file_id directly - resolvedFileID = fileID - } - - log.Printf("🖼️ [DESCRIBE-IMAGE] Analyzing image file_id=%s detail=%s (user=%s, conv=%s)", resolvedFileID, detail, userID, convID) - - // Get file from cache with proper validation - if userID != "" && convID != "" { - var err error - file, err = fileCacheService.GetByUserAndConversation(resolvedFileID, userID, convID) - if err != nil { - // Try with just user validation - file, err = fileCacheService.GetByUser(resolvedFileID, userID) - if err != nil { - // Try without validation for workflow context - file, _ = fileCacheService.Get(resolvedFileID) - if file != nil && file.UserID != "" && file.UserID != userID { - log.Printf("🚫 [DESCRIBE-IMAGE] Access denied: file %s belongs to different user", resolvedFileID) - return "", fmt.Errorf("access denied: you don't have permission to access this file") - } - } - } - } else if userID != "" { - var err error - file, err = fileCacheService.GetByUser(resolvedFileID, userID) - if err != nil { - file, _ = fileCacheService.Get(resolvedFileID) - } - } else { - file, _ = fileCacheService.Get(resolvedFileID) - } - - if file == nil { - log.Printf("❌ [DESCRIBE-IMAGE] File not found: %s", resolvedFileID) - if hasImageID && imageID != "" { - return "", fmt.Errorf("image '%s' has expired or is no longer available. Images are cached for 30 minutes. Please upload or generate the image again", imageID) - } - return "", fmt.Errorf("image file not found or has expired. Files are only available for 30 minutes after upload") - } - - // Validate it's an image - if !strings.HasPrefix(file.MimeType, "image/") { - log.Printf("⚠️ [DESCRIBE-IMAGE] File is not an image: %s (%s)", resolvedFileID, file.MimeType) - return "", fmt.Errorf("file is not an image (type: %s). Use read_document for documents or read_data_file for data files", file.MimeType) - } - - // Read image data from disk - if file.FilePath == "" { - return "", fmt.Errorf("image file path not available") - } - - var err error - imageData, err = os.ReadFile(file.FilePath) - if err != nil { - log.Printf("❌ [DESCRIBE-IMAGE] Failed to read image from disk: %v (path: %s)", err, file.FilePath) - return "", fmt.Errorf("image file has expired or been deleted. Please upload or generate the image again") - } - - mimeType = file.MimeType - filename = file.Filename - } - - // Get the vision service - visionService := vision.GetService() - if visionService == nil { - return "", fmt.Errorf("vision service not available. Please configure a vision-capable model (e.g., GPT-4o)") - } - - // Build the request - req := &vision.DescribeImageRequest{ - ImageData: imageData, - MimeType: mimeType, - Question: question, - Detail: detail, - } - - // Call vision service - result, err := visionService.DescribeImage(req) - if err != nil { - log.Printf("❌ [DESCRIBE-IMAGE] Vision analysis failed: %v", err) - return "", fmt.Errorf("failed to analyze image: %v", err) - } - - // Build response - response := map[string]interface{}{ - "success": true, - "filename": filename, - "mime_type": mimeType, - "description": result.Description, - "model": result.Model, - "provider": result.Provider, - } - - // Include source-specific fields - if sourceURL != "" { - response["source_url"] = sourceURL - } else { - response["file_id"] = resolvedFileID - } - - // Include image_id if it was used - if hasImageID && imageID != "" { - response["image_id"] = imageID - } - - if question != "" { - response["question"] = question - } - - responseJSON, err := json.Marshal(response) - if err != nil { - return "", fmt.Errorf("failed to marshal response: %w", err) - } - - log.Printf("✅ [DESCRIBE-IMAGE] Successfully described image %s using %s", filename, result.Model) - - return string(responseJSON), nil -} - -// fetchImageFromURL downloads an image from a URL and returns the data, mime type, and filename -func fetchImageFromURL(urlStr string) ([]byte, string, string, error) { - // Validate URL using the existing validation function from download_file_tool - parsedURL, err := validateDownloadURL(urlStr) - if err != nil { - return nil, "", "", fmt.Errorf("invalid URL: %v", err) - } - - // Create HTTP client with timeout - client := &http.Client{ - Timeout: describeImageTimeout, - CheckRedirect: func(req *http.Request, via []*http.Request) error { - if len(via) >= 5 { - return fmt.Errorf("too many redirects") - } - // Validate redirect URL - if _, err := validateDownloadURL(req.URL.String()); err != nil { - return fmt.Errorf("redirect blocked: %v", err) - } - return nil - }, - } - - // Create request - req, err := http.NewRequest("GET", parsedURL.String(), nil) - if err != nil { - return nil, "", "", fmt.Errorf("failed to create request: %v", err) - } - - // Set a reasonable User-Agent - req.Header.Set("User-Agent", "ClaraVerse/1.0 (Image Analyzer)") - - // Make request - resp, err := client.Do(req) - if err != nil { - return nil, "", "", fmt.Errorf("failed to fetch image: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != http.StatusOK { - return nil, "", "", fmt.Errorf("failed to fetch image: HTTP %d", resp.StatusCode) - } - - // Get content type - contentType := resp.Header.Get("Content-Type") - if contentType == "" { - contentType = "application/octet-stream" - } - // Strip charset suffix if present - if idx := strings.Index(contentType, ";"); idx != -1 { - contentType = strings.TrimSpace(contentType[:idx]) - } - - // Validate it's an image - if !strings.HasPrefix(contentType, "image/") { - // Try to detect from URL extension - ext := strings.ToLower(filepath.Ext(parsedURL.Path)) - switch ext { - case ".jpg", ".jpeg": - contentType = "image/jpeg" - case ".png": - contentType = "image/png" - case ".gif": - contentType = "image/gif" - case ".webp": - contentType = "image/webp" - case ".svg": - contentType = "image/svg+xml" - case ".bmp": - contentType = "image/bmp" - default: - return nil, "", "", fmt.Errorf("URL does not point to an image (content-type: %s)", contentType) - } - } - - // Check content length if available - if resp.ContentLength > describeImageMaxSize { - return nil, "", "", fmt.Errorf("image too large: %d bytes (max %d bytes)", resp.ContentLength, describeImageMaxSize) - } - - // Read body with size limit - limitedReader := io.LimitReader(resp.Body, describeImageMaxSize+1) - content, err := io.ReadAll(limitedReader) - if err != nil { - return nil, "", "", fmt.Errorf("failed to read image: %v", err) - } - - if int64(len(content)) > describeImageMaxSize { - return nil, "", "", fmt.Errorf("image too large: max %d bytes", describeImageMaxSize) - } - - // Extract filename from URL or Content-Disposition - filename := extractFilename(parsedURL, contentType, resp.Header.Get("Content-Disposition")) - filename = sanitizeFilename(filename) - - return content, contentType, filename, nil -} diff --git a/backend/internal/tools/discord_tool.go b/backend/internal/tools/discord_tool.go deleted file mode 100644 index 32161b07..00000000 --- a/backend/internal/tools/discord_tool.go +++ /dev/null @@ -1,431 +0,0 @@ -package tools - -import ( - "bytes" - "encoding/base64" - "encoding/json" - "fmt" - "io" - "mime/multipart" - "net/http" - "os" - "strings" - "time" -) - -// NewDiscordTool creates a Discord webhook messaging tool -func NewDiscordTool() *Tool { - return &Tool{ - Name: "send_discord_message", - DisplayName: "Send Discord Message", - Description: "Send a message to Discord via webhook. Message content is limited to 2000 characters max (Discord API limit). Just provide the message content - webhook authentication is handled automatically via configured credentials. Do NOT ask the user for webhook URLs. Supports embeds for rich formatting.", - Icon: "MessageCircle", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"discord", "message", "chat", "notify", "webhook", "channel", "bot", "notification"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "webhook_url": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Resolved from credentials. Do not ask user for this.", - }, - "content": map[string]interface{}{ - "type": "string", - "description": "Message content (max 2000 characters). This is the main text message.", - }, - "username": map[string]interface{}{ - "type": "string", - "description": "Override the default webhook username (optional)", - }, - "avatar_url": map[string]interface{}{ - "type": "string", - "description": "Override the default webhook avatar URL (optional)", - }, - "embed_title": map[string]interface{}{ - "type": "string", - "description": "Title for an embed (optional, for rich formatting)", - }, - "embed_description": map[string]interface{}{ - "type": "string", - "description": "Description for an embed (optional, max 4096 characters)", - }, - "embed_color": map[string]interface{}{ - "type": "number", - "description": "Embed color as decimal (optional, e.g., 5814783 for blue)", - }, - "image_data": map[string]interface{}{ - "type": "string", - "description": "Base64 encoded image data to attach (optional). Can include data URI prefix or raw base64.", - }, - "image_filename": map[string]interface{}{ - "type": "string", - "description": "Filename for the attached image (optional, defaults to 'chart.png')", - }, - "file_url": map[string]interface{}{ - "type": "string", - "description": "URL to download a file to attach (optional). Supports both absolute URLs and relative paths starting with /api/files/. For relative paths, the backend URL will be automatically resolved.", - }, - "file_name": map[string]interface{}{ - "type": "string", - "description": "Filename for the attached file from URL (optional, will be inferred from URL if not provided)", - }, - }, - "required": []string{}, - }, - Execute: executeDiscordMessage, - } -} - -func executeDiscordMessage(args map[string]interface{}) (string, error) { - // Resolve webhook URL from credential or direct parameter - webhookURL, err := ResolveWebhookURL(args, "discord") - if err != nil { - // Fallback: check for direct webhook_url if credential resolution failed - if url, ok := args["webhook_url"].(string); ok && url != "" { - webhookURL = url - } else { - return "", fmt.Errorf("failed to get webhook URL: %w", err) - } - } - - // Validate Discord webhook URL - if !strings.Contains(webhookURL, "discord.com/api/webhooks/") && !strings.Contains(webhookURL, "discordapp.com/api/webhooks/") { - return "", fmt.Errorf("invalid Discord webhook URL") - } - - // Extract content (optional now since we might just send an image) - content, _ := args["content"].(string) - - // Truncate content if too long (Discord limit is 2000) - if len(content) > 2000 { - content = content[:1997] + "..." - } - - // Check for image data - imageData, hasImage := args["image_data"].(string) - imageFilename := "chart.png" - if fn, ok := args["image_filename"].(string); ok && fn != "" { - imageFilename = fn - } - - // Check for file URL - fileURL, hasFileURL := args["file_url"].(string) - var fileData []byte - fileName := "" - if fn, ok := args["file_name"].(string); ok && fn != "" { - fileName = fn - } - - // Fetch file from URL if provided - if hasFileURL && fileURL != "" { - var fetchErr error - fileData, fileName, fetchErr = fetchFileFromURL(fileURL, fileName) - if fetchErr != nil { - return "", fmt.Errorf("failed to fetch file from URL: %w", fetchErr) - } - } - - // Build Discord webhook payload - payload := map[string]interface{}{} - if content != "" { - payload["content"] = content - } - - // Optional username override - if username, ok := args["username"].(string); ok && username != "" { - payload["username"] = username - } - - // Optional avatar override - if avatarURL, ok := args["avatar_url"].(string); ok && avatarURL != "" { - payload["avatar_url"] = avatarURL - } - - // Build embed if any embed fields provided - embed := make(map[string]interface{}) - hasEmbed := false - - if embedTitle, ok := args["embed_title"].(string); ok && embedTitle != "" { - embed["title"] = embedTitle - hasEmbed = true - } - - if embedDesc, ok := args["embed_description"].(string); ok && embedDesc != "" { - // Truncate embed description if too long (Discord limit is 4096) - if len(embedDesc) > 4096 { - embedDesc = embedDesc[:4093] + "..." - } - embed["description"] = embedDesc - hasEmbed = true - } - - if embedColor, ok := args["embed_color"].(float64); ok { - embed["color"] = int(embedColor) - hasEmbed = true - } - - // If we have an image, reference it in the embed - if hasImage && imageData != "" { - if !hasEmbed { - embed["title"] = "Generated Chart" - hasEmbed = true - } - // Reference the attached image in the embed - embed["image"] = map[string]interface{}{ - "url": "attachment://" + imageFilename, - } - } - - if hasEmbed { - embed["timestamp"] = time.Now().UTC().Format(time.RFC3339) - payload["embeds"] = []map[string]interface{}{embed} - } - - // Require at least content, image, file, or embed - hasFile := len(fileData) > 0 - if content == "" && !hasImage && !hasFile && !hasEmbed { - return "", fmt.Errorf("either content, image_data, file_url, or embed is required") - } - - // Create HTTP client - client := &http.Client{ - Timeout: 60 * time.Second, - } - - var req *http.Request - - if hasFile { - // Send with multipart/form-data for file attachment - req, err = createMultipartRequestWithFile(webhookURL, payload, fileData, fileName) - } else if hasImage && imageData != "" { - // Send with multipart/form-data for image attachment - req, err = createMultipartRequest(webhookURL, payload, imageData, imageFilename) - } else { - // Send as JSON (no image) - jsonPayload, jsonErr := json.Marshal(payload) - if jsonErr != nil { - return "", fmt.Errorf("failed to serialize payload: %w", jsonErr) - } - req, err = http.NewRequest("POST", webhookURL, bytes.NewBuffer(jsonPayload)) - if err == nil { - req.Header.Set("Content-Type", "application/json") - } - } - - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - // Execute request - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - // Read response body - respBody, err := io.ReadAll(resp.Body) - if err != nil { - return "", fmt.Errorf("failed to read response: %w", err) - } - - // Build result - success := resp.StatusCode >= 200 && resp.StatusCode < 300 - result := map[string]interface{}{ - "success": success, - "status_code": resp.StatusCode, - "status": resp.Status, - "message_sent": success, - } - - if content != "" { - result["content_length"] = len(content) - } - if hasFile { - result["file_attached"] = true - result["file_name"] = fileName - result["file_size"] = len(fileData) - } - if hasImage { - result["image_attached"] = true - result["image_filename"] = imageFilename - } - - // Include response body if there's an error - if !success && len(respBody) > 0 { - result["error"] = string(respBody) - } - - // Add success message - if success { - if hasFile { - result["message"] = fmt.Sprintf("Discord message with file '%s' sent successfully", fileName) - } else if hasImage { - result["message"] = "Discord message with image sent successfully" - } else { - result["message"] = "Discord message sent successfully" - } - } - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} - -// fetchFileFromURL fetches a file from a URL (supports relative paths for internal files) -func fetchFileFromURL(fileURL, providedFileName string) ([]byte, string, error) { - // Resolve relative URLs to absolute URLs - actualURL := fileURL - if strings.HasPrefix(fileURL, "/api/") { - // Use BACKEND_URL env var for internal API calls - backendURL := os.Getenv("BACKEND_URL") - if backendURL == "" { - backendURL = "http://localhost:3001" - } - actualURL = backendURL + fileURL - } - - // Create HTTP client - client := &http.Client{ - Timeout: 30 * time.Second, - } - - resp, err := client.Get(actualURL) - if err != nil { - return nil, "", fmt.Errorf("failed to fetch file: %w", err) - } - defer resp.Body.Close() - - if resp.StatusCode != http.StatusOK { - return nil, "", fmt.Errorf("failed to fetch file: status %d", resp.StatusCode) - } - - // Read the file content - data, err := io.ReadAll(resp.Body) - if err != nil { - return nil, "", fmt.Errorf("failed to read file: %w", err) - } - - // Determine filename - fileName := providedFileName - if fileName == "" { - // Try to get from Content-Disposition header - if cd := resp.Header.Get("Content-Disposition"); cd != "" { - if strings.Contains(cd, "filename=") { - parts := strings.Split(cd, "filename=") - if len(parts) > 1 { - fileName = strings.Trim(parts[1], "\"' ") - } - } - } - // Fallback: extract from URL - if fileName == "" { - parts := strings.Split(strings.Split(fileURL, "?")[0], "/") - if len(parts) > 0 { - fileName = parts[len(parts)-1] - } - } - // Final fallback - if fileName == "" { - fileName = "attachment" - } - } - - return data, fileName, nil -} - -// createMultipartRequestWithFile creates a multipart request with JSON payload and raw file data -func createMultipartRequestWithFile(webhookURL string, payload map[string]interface{}, fileData []byte, filename string) (*http.Request, error) { - // Create multipart form - var body bytes.Buffer - writer := multipart.NewWriter(&body) - - // Add payload_json field - payloadJSON, err := json.Marshal(payload) - if err != nil { - return nil, fmt.Errorf("failed to serialize payload: %w", err) - } - - if err := writer.WriteField("payload_json", string(payloadJSON)); err != nil { - return nil, fmt.Errorf("failed to write payload field: %w", err) - } - - // Add file attachment - part, err := writer.CreateFormFile("files[0]", filename) - if err != nil { - return nil, fmt.Errorf("failed to create form file: %w", err) - } - - if _, err := part.Write(fileData); err != nil { - return nil, fmt.Errorf("failed to write file data: %w", err) - } - - if err := writer.Close(); err != nil { - return nil, fmt.Errorf("failed to close multipart writer: %w", err) - } - - // Create request - req, err := http.NewRequest("POST", webhookURL, &body) - if err != nil { - return nil, err - } - - req.Header.Set("Content-Type", writer.FormDataContentType()) - return req, nil -} - -// createMultipartRequest creates a multipart request with JSON payload and image attachment -func createMultipartRequest(webhookURL string, payload map[string]interface{}, imageData, filename string) (*http.Request, error) { - // Decode base64 image data - // Remove data URI prefix if present - imageData = strings.TrimPrefix(imageData, "data:image/png;base64,") - imageData = strings.TrimPrefix(imageData, "data:image/jpeg;base64,") - imageData = strings.TrimPrefix(imageData, "data:image/jpg;base64,") - imageData = strings.TrimPrefix(imageData, "data:image/gif;base64,") - - imageBytes, err := base64.StdEncoding.DecodeString(imageData) - if err != nil { - return nil, fmt.Errorf("failed to decode base64 image: %w", err) - } - - // Create multipart form - var body bytes.Buffer - writer := multipart.NewWriter(&body) - - // Add payload_json field - payloadJSON, err := json.Marshal(payload) - if err != nil { - return nil, fmt.Errorf("failed to serialize payload: %w", err) - } - - if err := writer.WriteField("payload_json", string(payloadJSON)); err != nil { - return nil, fmt.Errorf("failed to write payload field: %w", err) - } - - // Add file attachment - part, err := writer.CreateFormFile("files[0]", filename) - if err != nil { - return nil, fmt.Errorf("failed to create form file: %w", err) - } - - if _, err := part.Write(imageBytes); err != nil { - return nil, fmt.Errorf("failed to write image data: %w", err) - } - - if err := writer.Close(); err != nil { - return nil, fmt.Errorf("failed to close multipart writer: %w", err) - } - - // Create request - req, err := http.NewRequest("POST", webhookURL, &body) - if err != nil { - return nil, err - } - - req.Header.Set("Content-Type", writer.FormDataContentType()) - return req, nil -} diff --git a/backend/internal/tools/document_tool.go b/backend/internal/tools/document_tool.go deleted file mode 100644 index b99bf090..00000000 --- a/backend/internal/tools/document_tool.go +++ /dev/null @@ -1,139 +0,0 @@ -package tools - -import ( - "claraverse/internal/document" - "claraverse/internal/securefile" - "encoding/json" - "fmt" - "log" - "os" -) - -// NewDocumentTool creates the create_document tool -func NewDocumentTool() *Tool { - return &Tool{ - Name: "create_document", - DisplayName: "Create Document", - Description: `Creates a professional PDF document from custom HTML content. Full creative control with HTML/CSS for maximum design flexibility. - -Perfect for: -- Professional reports with custom branding -- Invoices and receipts with styled layouts -- Legal documents with precise formatting -- Technical documentation with code blocks -- Certificates and formal documents -- Creative documents with custom designs - -You can use any HTML/CSS - inline styles, flexbox/grid layouts, custom fonts, colors, gradients, tables, images (base64 or URLs), etc. The document is rendered as a standard A4 portrait PDF and stored for 30 days with an access code for download. - -**Page Break Control:** -- Use CSS 'page-break-before', 'page-break-after', or 'page-break-inside' to control page breaks -- Add 'page-break-inside: avoid' to prevent elements from being split across pages -- Use 'page-break-after: always' to force a new page after an element -- Tables, images, and code blocks should have 'page-break-inside: avoid' to prevent awkward cuts -- Example:
Content that stays together
`, - Icon: "FileText", - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "html": map[string]interface{}{ - "type": "string", - "description": "The document content as HTML. Can include inline CSS (

Title Slide

"}, - {"html": "

Topic 1

Content here

"}, - {"html": "

Topic 2

More content

"}, - {"html": "

Topic 3

"}, - {"html": "

Thank You

"} - ] -} - -**REQUIREMENTS:** -- MUST create 5-15 pages (NOT just 1-2!) -- Each page MUST be a complete HTML document starting with and ending with -- Include with

Slide 1

\"}, {\"html\": \"...

Slide 2

\"}, ...]", - "items": map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "html": map[string]interface{}{ - "type": "string", - "description": "COMPLETE HTML DOCUMENT for this slide. MUST start with and end with . MUST include and tags. Put styles in . Example: \"

My Title

\"", - }, - }, - "required": []string{"html"}, - }, - }, - }, - "required": []string{"title", "pages"}, - }, - Execute: executeCreatePresentation, - Source: ToolSourceBuiltin, - Category: "output", - Keywords: []string{"presentation", "slides", "powerpoint", "ppt", "slideshow", "pdf", "create", "generate", "export", "custom", "design"}, - } -} - -func executeCreatePresentation(args map[string]interface{}) (string, error) { - // Extract title (required) - title, ok := args["title"].(string) - if !ok || title == "" { - return "", fmt.Errorf("title is required") - } - - // Extract pages (required) - pagesRaw, ok := args["pages"].([]interface{}) - if !ok || len(pagesRaw) == 0 { - return "", fmt.Errorf("pages array is required and must not be empty") - } - - // Validate minimum page count (HARD REQUIREMENT: at least 3 pages) - if len(pagesRaw) < 3 { - return "", fmt.Errorf("presentations must have at least 3 pages (you provided %d). A complete presentation needs: (1) a title/cover page, (2) one or more content pages, and (3) a conclusion/thank you page. Please provide at least 3 page objects in the pages array, each with an 'html' field. Example: pages: [{\"html\": \"
Cover
\"}, {\"html\": \"
Content 1
\"}, {\"html\": \"
Content 2
\"}, {\"html\": \"
Thank You
\"}]", len(pagesRaw)) - } - - // Warn about short presentations (RECOMMENDATION: 5-15 pages) - if len(pagesRaw) < 5 { - log.Printf("⚠️ [PRESENTATION-TOOL] Short presentation detected (%d pages). Typical presentations should have 5-15 pages for better content coverage.", len(pagesRaw)) - } - - // Parse pages - slides := make([]presentation.Slide, 0, len(pagesRaw)) - for i, pageRaw := range pagesRaw { - pageMap, ok := pageRaw.(map[string]interface{}) - if !ok { - return "", fmt.Errorf("page %d is not a valid object", i) - } - - // HTML content (required) - htmlContent, ok := pageMap["html"].(string) - if !ok || htmlContent == "" { - return "", fmt.Errorf("page %d missing required 'html' field. Each page object must have an 'html' field containing the HTML content for that slide. Example: {\"html\": \"

Slide Title

\"}", i+1) - } - - // Validate HTML content is not just whitespace - trimmed := strings.TrimSpace(htmlContent) - if trimmed == "" { - return "", fmt.Errorf("page %d has empty HTML content (only whitespace). Please provide actual HTML content for this slide. Example: {\"html\": \"

Content Here

\"}", i+1) - } - - // Validate HTML is a complete document (must have and tags) - lowerHTML := strings.ToLower(trimmed) - if !strings.Contains(lowerHTML, " tag. You provided: %q. Each slide MUST be a standalone HTML document. Example: {\"html\": \"

Title

\"}", i+1, trimmed) - } - if !strings.Contains(lowerHTML, "") { - return "", fmt.Errorf("page %d must be a COMPLETE HTML document ending with tag. Your HTML is missing the closing tag. Example: {\"html\": \"Content\"}", i+1) - } - if !strings.Contains(lowerHTML, "") { - return "", fmt.Errorf("page %d must include tags. Complete HTML structure required: ...Your content here", i+1) - } - - slides = append(slides, presentation.Slide{ - HTML: htmlContent, - }) - } - - // Extract injected user context - userID, _ := args["__user_id__"].(string) - if userID == "" { - userID = "system" - } - conversationID, _ := args["__conversation_id__"].(string) - - // Clean up internal parameters - delete(args, "__user_id__") - delete(args, "__conversation_id__") - - log.Printf("🎯 [PRESENTATION-TOOL] Creating custom HTML presentation: %s (%d pages, 16:9 landscape PDF)", - title, len(slides)) - - // Build config - config := presentation.PresentationConfig{ - Title: title, - Slides: slides, - } - - // Get presentation service and generate PDF - presService := presentation.GetService() - tempResult, err := presService.GeneratePresentation(config, userID, conversationID) - if err != nil { - log.Printf("❌ [PRESENTATION-TOOL] Failed to generate presentation: %v", err) - return "", fmt.Errorf("failed to generate presentation: %w", err) - } - - // Read the generated file - fileContent, err := os.ReadFile(tempResult.FilePath) - if err != nil { - log.Printf("❌ [PRESENTATION-TOOL] Failed to read generated file: %v", err) - return "", fmt.Errorf("failed to read generated file: %w", err) - } - - // Store in secure file service - secureFileService := securefile.GetService() - secureResult, err := secureFileService.CreateFile(userID, fileContent, tempResult.Filename, "application/pdf") - if err != nil { - log.Printf("❌ [PRESENTATION-TOOL] Failed to store presentation securely: %v", err) - return "", fmt.Errorf("failed to store presentation: %w", err) - } - - // Cleanup temporary file - if err := os.Remove(tempResult.FilePath); err != nil { - log.Printf("⚠️ [PRESENTATION-TOOL] Failed to cleanup temp file: %v", err) - } - - // Format response - response := map[string]interface{}{ - "success": true, - "file_id": secureResult.ID, - "filename": secureResult.Filename, - "download_url": secureResult.DownloadURL, - "access_code": secureResult.AccessCode, - "size": secureResult.Size, - "file_type": "pdf", - "page_count": len(slides), - "aspect_ratio": "16:9 landscape", - "page_size": "10.67\" x 6\" (widescreen)", - "expires_at": secureResult.ExpiresAt.Format("2006-01-02"), - "message": fmt.Sprintf("Custom HTML presentation '%s' created successfully with %d pages in 16:9 landscape format. Download link (valid for 30 days): %s", secureResult.Filename, len(slides), secureResult.DownloadURL), - } - - responseJSON, _ := json.Marshal(response) - - log.Printf("✅ [PRESENTATION-TOOL] Custom HTML presentation generated: %s (%d bytes, %d pages, expires: %s)", - secureResult.Filename, secureResult.Size, len(slides), secureResult.ExpiresAt.Format("2006-01-02")) - - return string(responseJSON), nil -} - diff --git a/backend/internal/tools/python_runner_tool.go b/backend/internal/tools/python_runner_tool.go deleted file mode 100644 index 848da4c3..00000000 --- a/backend/internal/tools/python_runner_tool.go +++ /dev/null @@ -1,156 +0,0 @@ -package tools - -import ( - "context" - "encoding/json" - "fmt" - - "claraverse/internal/e2b" -) - -// NewPythonRunnerTool creates a new Python Code Runner tool -func NewPythonRunnerTool() *Tool { - return &Tool{ - Name: "run_python", - DisplayName: "Python Code Runner", - Description: `Execute Python code with custom pip dependencies. Install packages on-the-fly and retrieve generated files. Max 5 minutes execution time. - -⚠️ CRITICAL: This tool CANNOT access user-uploaded files! -- If user uploaded a file (CSV, Excel, JSON, etc.), use 'analyze_data' tool instead -- Files uploaded by users are NOT accessible by filename in this sandbox -- This tool runs in an isolated environment with NO access to local files - -USE THIS TOOL FOR: -- Running Python scripts that need specific pip packages (torch, transformers, etc.) -- Generating NEW files (model weights, processed data, images, PDFs) -- Quick computations, API calls, web scraping -- Processing data from URLs (not local files) -- Code that doesn't need user-uploaded input files - -DO NOT USE FOR: -- Analyzing user-uploaded CSV/Excel/JSON files → use 'analyze_data' instead -- Reading local files by filename → they don't exist in sandbox -- Any task requiring access to files the user shared in chat - -Remember: Install dependencies and run code in the same session - no persistence between calls.`, - Icon: "Terminal", - Source: ToolSourceBuiltin, - Category: "computation", - Keywords: []string{"python", "code", "execute", "run", "script", "programming", "processing", "compute", "pip", "packages", "dependencies"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "code": map[string]interface{}{ - "type": "string", - "description": "Python code to execute", - }, - "dependencies": map[string]interface{}{ - "type": "array", - "description": "Pip packages to install before execution (e.g., ['torch', 'transformers', 'requests'])", - "items": map[string]interface{}{ - "type": "string", - }, - }, - "output_files": map[string]interface{}{ - "type": "array", - "description": "File paths to retrieve after execution (e.g., ['model.pt', 'output.csv'])", - "items": map[string]interface{}{ - "type": "string", - }, - }, - }, - "required": []string{"code"}, - }, - Execute: executePythonRunner, - } -} - -func executePythonRunner(args map[string]interface{}) (string, error) { - // Extract code (required) - code, ok := args["code"].(string) - if !ok || code == "" { - return "", fmt.Errorf("code is required") - } - - // Extract dependencies (optional) - var dependencies []string - if depsRaw, ok := args["dependencies"].([]interface{}); ok { - for _, dep := range depsRaw { - if depStr, ok := dep.(string); ok { - dependencies = append(dependencies, depStr) - } - } - } - - // Extract output files (optional) - var outputFiles []string - if filesRaw, ok := args["output_files"].([]interface{}); ok { - for _, file := range filesRaw { - if fileStr, ok := file.(string); ok { - outputFiles = append(outputFiles, fileStr) - } - } - } - - // Build request - req := e2b.AdvancedExecuteRequest{ - Code: code, - Timeout: 300, // 5 minutes for complex tasks like Playwright, ML training, etc. - Dependencies: dependencies, - OutputFiles: outputFiles, - } - - // Execute - e2bService := e2b.GetE2BExecutorService() - result, err := e2bService.ExecuteAdvanced(context.Background(), req) - if err != nil { - return "", fmt.Errorf("failed to execute code: %w", err) - } - - if !result.Success { - errorMsg := "execution failed" - if result.Error != nil { - errorMsg = *result.Error - } - if result.Stderr != "" { - errorMsg += "\nStderr: " + result.Stderr - } - return "", fmt.Errorf("%s", errorMsg) - } - - // Build response - response := map[string]interface{}{ - "success": true, - "stdout": result.Stdout, - } - - // Include stderr if present - if result.Stderr != "" { - response["stderr"] = result.Stderr - } - - // Include install output if dependencies were installed - if result.InstallOutput != "" { - response["install_output"] = result.InstallOutput - } - - // Include plots if any - if len(result.Plots) > 0 { - response["plots"] = result.Plots - response["plot_count"] = len(result.Plots) - } - - // Include files if any were retrieved - if len(result.Files) > 0 { - response["files"] = result.Files - response["file_count"] = len(result.Files) - } - - // Include execution time - if result.ExecutionTime != nil { - response["execution_time"] = *result.ExecutionTime - } - - jsonResponse, _ := json.MarshalIndent(response, "", " ") - return string(jsonResponse), nil -} diff --git a/backend/internal/tools/read_datafile_tool.go b/backend/internal/tools/read_datafile_tool.go deleted file mode 100644 index 0eafe4cb..00000000 --- a/backend/internal/tools/read_datafile_tool.go +++ /dev/null @@ -1,246 +0,0 @@ -package tools - -import ( - "claraverse/internal/filecache" - "encoding/csv" - "encoding/json" - "fmt" - "log" - "os" - "strings" -) - -// NewReadDataFileTool creates the read_data_file tool for reading CSV/JSON/text files -func NewReadDataFileTool() *Tool { - return &Tool{ - Name: "read_data_file", - DisplayName: "Read Data File", - Description: "Reads the content of CSV, JSON, Excel, or text files that were uploaded. Returns the file content in the specified format.", - Icon: "FileSpreadsheet", - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "file_id": map[string]interface{}{ - "type": "string", - "description": "The file ID of the uploaded data file (from the upload response)", - }, - "format": map[string]interface{}{ - "type": "string", - "enum": []string{"raw", "json", "csv_rows"}, - "description": "Output format: 'raw' returns text as-is, 'json' parses JSON files, 'csv_rows' returns CSV as array of rows. Default is 'raw'", - }, - "max_rows": map[string]interface{}{ - "type": "integer", - "description": "Maximum number of rows to return for CSV files (default 100, max 500)", - }, - }, - "required": []string{"file_id"}, - }, - Execute: executeReadDataFile, - Source: ToolSourceBuiltin, - Category: "data_sources", - Keywords: []string{"csv", "json", "excel", "text", "data", "read", "file", "spreadsheet", "txt", "content"}, - } -} - -func executeReadDataFile(args map[string]interface{}) (string, error) { - // Extract file_id parameter - fileID, ok := args["file_id"].(string) - if !ok || fileID == "" { - return "", fmt.Errorf("file_id parameter is required and must be a string") - } - - // Extract format parameter (default to "raw") - format := "raw" - if f, ok := args["format"].(string); ok && f != "" { - format = f - } - - // Extract max_rows parameter (default 100, max 500) - maxRows := 100 - if mr, ok := args["max_rows"].(float64); ok { - maxRows = int(mr) - if maxRows > 500 { - maxRows = 500 - } - if maxRows < 1 { - maxRows = 1 - } - } - - // Extract user context (injected by tool executor) - userID, _ := args["__user_id__"].(string) - conversationID, _ := args["__conversation_id__"].(string) - - // Clean up internal parameters - delete(args, "__user_id__") - delete(args, "__conversation_id__") - - log.Printf("📊 [READ-DATA-FILE] Reading data file_id=%s format=%s (user=%s)", fileID, format, userID) - - // Get file cache service - fileCacheService := filecache.GetService() - - // Try to get file with ownership validation - var file *filecache.CachedFile - var err error - - if userID != "" && conversationID != "" { - file, err = fileCacheService.GetByUserAndConversation(fileID, userID, conversationID) - if err != nil { - // Try just by user (conversation might not match in workflow context) - file, _ = fileCacheService.Get(fileID) - if file != nil && file.UserID != userID { - log.Printf("🚫 [READ-DATA-FILE] Access denied: file %s belongs to different user", fileID) - return "", fmt.Errorf("access denied: you don't have permission to read this file") - } - } - } else { - // Fallback: get file without strict ownership check (for workflow context) - file, _ = fileCacheService.Get(fileID) - } - - if file == nil { - log.Printf("❌ [READ-DATA-FILE] File not found: %s", fileID) - return "", fmt.Errorf("file not found or has expired. Files are only available for 30 minutes after upload") - } - - // Validate file type is a data file - supportedTypes := map[string]string{ - "text/csv": "csv", - "application/json": "json", - "text/plain": "text", - "application/vnd.ms-excel": "excel", - "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet": "excel", - "text/tab-separated-values": "csv", - } - - fileType, supported := supportedTypes[file.MimeType] - if !supported { - // Check by extension if mime type not recognized - if strings.HasSuffix(strings.ToLower(file.Filename), ".csv") { - fileType = "csv" - } else if strings.HasSuffix(strings.ToLower(file.Filename), ".json") { - fileType = "json" - } else if strings.HasSuffix(strings.ToLower(file.Filename), ".txt") { - fileType = "text" - } else { - log.Printf("⚠️ [READ-DATA-FILE] Unsupported file type: %s", file.MimeType) - return "", fmt.Errorf("unsupported file type: %s. Use read_document for PDF/DOCX/PPTX files", file.MimeType) - } - } - - // Read file content - var content string - if file.FilePath != "" { - // File stored on disk (images, CSV, etc.) - data, err := os.ReadFile(file.FilePath) - if err != nil { - log.Printf("❌ [READ-DATA-FILE] Failed to read file from disk: %v", err) - return "", fmt.Errorf("failed to read file: %v", err) - } - content = string(data) - } else if file.ExtractedText != nil { - // File stored in memory (PDFs) - content = file.ExtractedText.String() - } else { - log.Printf("⚠️ [READ-DATA-FILE] No content available for file: %s", fileID) - return "", fmt.Errorf("no content available for this file") - } - - // Process based on format and file type - var response map[string]interface{} - - switch format { - case "json": - if fileType == "json" { - // Parse and return JSON - var jsonData interface{} - if err := json.Unmarshal([]byte(content), &jsonData); err != nil { - return "", fmt.Errorf("failed to parse JSON: %v", err) - } - response = map[string]interface{}{ - "success": true, - "file_id": file.FileID, - "filename": file.Filename, - "mime_type": file.MimeType, - "size": file.Size, - "format": "json", - "data": jsonData, - } - } else { - return "", fmt.Errorf("json format only supported for JSON files, this is a %s file", fileType) - } - - case "csv_rows": - if fileType == "csv" { - // Parse CSV and return as rows - reader := csv.NewReader(strings.NewReader(content)) - records, err := reader.ReadAll() - if err != nil { - return "", fmt.Errorf("failed to parse CSV: %v", err) - } - - // Limit rows - totalRows := len(records) - if len(records) > maxRows { - records = records[:maxRows] - } - - // Extract headers and data - var headers []string - var rows [][]string - if len(records) > 0 { - headers = records[0] - if len(records) > 1 { - rows = records[1:] - } - } - - response = map[string]interface{}{ - "success": true, - "file_id": file.FileID, - "filename": file.Filename, - "mime_type": file.MimeType, - "size": file.Size, - "format": "csv_rows", - "headers": headers, - "rows": rows, - "total_rows": totalRows - 1, // Exclude header - "returned_rows": len(rows), - "truncated": totalRows > maxRows, - } - } else { - return "", fmt.Errorf("csv_rows format only supported for CSV files, this is a %s file", fileType) - } - - default: // "raw" - // Return raw content - // Truncate if too large for LLM context (100KB limit) - truncated := false - if len(content) > 100000 { - content = content[:100000] - truncated = true - } - - response = map[string]interface{}{ - "success": true, - "file_id": file.FileID, - "filename": file.Filename, - "mime_type": file.MimeType, - "size": file.Size, - "format": "raw", - "content": content, - "truncated": truncated, - } - } - - responseJSON, err := json.Marshal(response) - if err != nil { - return "", fmt.Errorf("failed to marshal response: %w", err) - } - - log.Printf("✅ [READ-DATA-FILE] Successfully read data file %s (format=%s)", file.Filename, format) - - return string(responseJSON), nil -} diff --git a/backend/internal/tools/read_document_tool.go b/backend/internal/tools/read_document_tool.go deleted file mode 100644 index 366dc722..00000000 --- a/backend/internal/tools/read_document_tool.go +++ /dev/null @@ -1,124 +0,0 @@ -package tools - -import ( - "claraverse/internal/filecache" - "encoding/json" - "fmt" - "log" -) - -// NewReadDocumentTool creates the read_document tool for reading PDF/DOCX/PPTX files -func NewReadDocumentTool() *Tool { - return &Tool{ - Name: "read_document", - DisplayName: "Read Document", - Description: "Extracts and returns the text content from an uploaded PDF, DOCX, or PPTX document. Use this to read document contents that were uploaded by the user.", - Icon: "FileText", - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "file_id": map[string]interface{}{ - "type": "string", - "description": "The file ID of the uploaded document (from the upload response)", - }, - }, - "required": []string{"file_id"}, - }, - Execute: executeReadDocument, - Source: ToolSourceBuiltin, - Category: "data_sources", - Keywords: []string{"document", "pdf", "docx", "pptx", "read", "extract", "text", "file", "content", "word", "powerpoint"}, - } -} - -func executeReadDocument(args map[string]interface{}) (string, error) { - // Extract file_id parameter - fileID, ok := args["file_id"].(string) - if !ok || fileID == "" { - return "", fmt.Errorf("file_id parameter is required and must be a string") - } - - // Extract user context (injected by tool executor) - userID, _ := args["__user_id__"].(string) - conversationID, _ := args["__conversation_id__"].(string) - - // Clean up internal parameters - delete(args, "__user_id__") - delete(args, "__conversation_id__") - - log.Printf("📄 [READ-DOCUMENT] Reading document file_id=%s (user=%s)", fileID, userID) - - // Get file cache service - fileCacheService := filecache.GetService() - - // Try to get file with ownership validation first - var file *filecache.CachedFile - var err error - - if userID != "" && conversationID != "" { - file, err = fileCacheService.GetByUserAndConversation(fileID, userID, conversationID) - if err != nil { - // Try just by user (conversation might not match in workflow context) - file, _ = fileCacheService.Get(fileID) - if file != nil && file.UserID != userID { - log.Printf("🚫 [READ-DOCUMENT] Access denied: file %s belongs to different user", fileID) - return "", fmt.Errorf("access denied: you don't have permission to read this file") - } - } - } else { - // Fallback: get file without strict ownership check (for workflow context) - file, _ = fileCacheService.Get(fileID) - } - - if file == nil { - log.Printf("❌ [READ-DOCUMENT] File not found: %s", fileID) - return "", fmt.Errorf("file not found or has expired. Documents are only available for 30 minutes after upload") - } - - // Validate file type is a document - supportedTypes := map[string]bool{ - "application/pdf": true, - "application/vnd.openxmlformats-officedocument.wordprocessingml.document": true, // .docx - "application/vnd.openxmlformats-officedocument.presentationml.presentation": true, // .pptx - "application/msword": true, // .doc - "application/vnd.ms-powerpoint": true, // .ppt - } - - if !supportedTypes[file.MimeType] { - log.Printf("⚠️ [READ-DOCUMENT] Unsupported file type: %s", file.MimeType) - return "", fmt.Errorf("unsupported file type: %s. Use read_data_file for CSV/JSON/text files", file.MimeType) - } - - // Get extracted text - var textContent string - if file.ExtractedText != nil { - textContent = file.ExtractedText.String() - } - - if textContent == "" { - log.Printf("⚠️ [READ-DOCUMENT] No text content extracted from file: %s", fileID) - return "", fmt.Errorf("no text content could be extracted from this document") - } - - // Build response - response := map[string]interface{}{ - "success": true, - "file_id": file.FileID, - "filename": file.Filename, - "mime_type": file.MimeType, - "size": file.Size, - "page_count": file.PageCount, - "word_count": file.WordCount, - "content": textContent, - } - - responseJSON, err := json.Marshal(response) - if err != nil { - return "", fmt.Errorf("failed to marshal response: %w", err) - } - - log.Printf("✅ [READ-DOCUMENT] Successfully read document %s: %d pages, %d words", - file.Filename, file.PageCount, file.WordCount) - - return string(responseJSON), nil -} diff --git a/backend/internal/tools/redis_tool.go b/backend/internal/tools/redis_tool.go deleted file mode 100644 index 7e5a6ca8..00000000 --- a/backend/internal/tools/redis_tool.go +++ /dev/null @@ -1,775 +0,0 @@ -package tools - -import ( - "context" - "encoding/json" - "fmt" - "time" - - "github.com/redis/go-redis/v9" -) - -// NewRedisReadTool creates a tool for reading from Redis -func NewRedisReadTool() *Tool { - return &Tool{ - Name: "redis_read", - DisplayName: "Redis Read", - Description: "Read data from Redis. Supports GET, MGET, HGET, HGETALL, LRANGE, SMEMBERS, ZRANGE, KEYS, EXISTS, TTL, and TYPE operations.", - Icon: "Database", - Source: ToolSourceBuiltin, - Category: "database", - Keywords: []string{"redis", "cache", "database", "key-value", "read", "get", "memory"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "action": map[string]interface{}{ - "type": "string", - "enum": []string{"get", "mget", "hget", "hgetall", "lrange", "smembers", "zrange", "keys", "exists", "ttl", "type"}, - "description": "The read operation to perform", - }, - "key": map[string]interface{}{ - "type": "string", - "description": "The Redis key to read", - }, - "keys": map[string]interface{}{ - "type": "array", - "description": "Array of keys (for MGET operation)", - "items": map[string]interface{}{ - "type": "string", - }, - }, - "field": map[string]interface{}{ - "type": "string", - "description": "Hash field name (for HGET operation)", - }, - "start": map[string]interface{}{ - "type": "integer", - "description": "Start index for LRANGE/ZRANGE (default: 0)", - }, - "stop": map[string]interface{}{ - "type": "integer", - "description": "Stop index for LRANGE/ZRANGE (default: -1 for all)", - }, - "pattern": map[string]interface{}{ - "type": "string", - "description": "Pattern for KEYS operation (e.g., 'user:*'). Use with caution on large databases.", - }, - }, - "required": []string{"action"}, - }, - Execute: executeRedisRead, - } -} - -// NewRedisWriteTool creates a tool for writing to Redis (no delete operations) -func NewRedisWriteTool() *Tool { - return &Tool{ - Name: "redis_write", - DisplayName: "Redis Write", - Description: "Write data to Redis. Supports SET, MSET, HSET, LPUSH, RPUSH, SADD, ZADD, INCR, and EXPIRE operations. Delete operations are not permitted for safety.", - Icon: "DatabaseBackup", - Source: ToolSourceBuiltin, - Category: "database", - Keywords: []string{"redis", "cache", "database", "key-value", "write", "set", "memory", "store"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "action": map[string]interface{}{ - "type": "string", - "enum": []string{"set", "mset", "hset", "lpush", "rpush", "sadd", "zadd", "incr", "incrby", "expire", "setnx", "setex"}, - "description": "The write operation to perform (delete is not permitted)", - }, - "key": map[string]interface{}{ - "type": "string", - "description": "The Redis key", - }, - "value": map[string]interface{}{ - "description": "The value to set (for SET, SETEX, SETNX operations). Can be string, number, or object (will be JSON encoded).", - }, - "values": map[string]interface{}{ - "type": "object", - "description": "Key-value pairs for MSET operation", - }, - "field": map[string]interface{}{ - "type": "string", - "description": "Hash field name (for HSET operation)", - }, - "field_values": map[string]interface{}{ - "type": "object", - "description": "Multiple field-value pairs for HSET (alternative to single field/value)", - }, - "members": map[string]interface{}{ - "type": "array", - "description": "Array of members for LPUSH, RPUSH, SADD operations", - "items": map[string]interface{}{ - "type": "string", - }, - }, - "scored_members": map[string]interface{}{ - "type": "array", - "description": "Array of {score, member} objects for ZADD operation", - "items": map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "score": map[string]interface{}{ - "type": "number", - }, - "member": map[string]interface{}{ - "type": "string", - }, - }, - }, - }, - "ttl_seconds": map[string]interface{}{ - "type": "integer", - "description": "Time-to-live in seconds (for SETEX, EXPIRE operations)", - }, - "increment": map[string]interface{}{ - "type": "integer", - "description": "Increment value (for INCRBY operation)", - }, - }, - "required": []string{"action"}, - }, - Execute: executeRedisWrite, - } -} - -func getRedisClient(args map[string]interface{}) (*redis.Client, error) { - // Get credential data - credData, err := GetCredentialData(args, "redis") - if err != nil { - return nil, fmt.Errorf("failed to get Redis credentials: %w", err) - } - - host, _ := credData["host"].(string) - if host == "" { - host = "localhost" - } - - port, _ := credData["port"].(string) - if port == "" { - // Try as number - if portNum, ok := credData["port"].(float64); ok { - port = fmt.Sprintf("%.0f", portNum) - } else { - port = "6379" - } - } - - password, _ := credData["password"].(string) - - db := 0 - if dbNum, ok := credData["database"].(float64); ok { - db = int(dbNum) - } else if dbNum, ok := credData["db"].(float64); ok { - db = int(dbNum) - } - - // Check for connection_string (alternative format) - if connStr, ok := credData["connection_string"].(string); ok && connStr != "" { - opt, err := redis.ParseURL(connStr) - if err != nil { - return nil, fmt.Errorf("invalid Redis connection string: %w", err) - } - return redis.NewClient(opt), nil - } - - client := redis.NewClient(&redis.Options{ - Addr: fmt.Sprintf("%s:%s", host, port), - Password: password, - DB: db, - }) - - return client, nil -} - -func executeRedisRead(args map[string]interface{}) (string, error) { - client, err := getRedisClient(args) - if err != nil { - return "", err - } - defer client.Close() - - ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second) - defer cancel() - - // Verify connection - if err := client.Ping(ctx).Err(); err != nil { - return "", fmt.Errorf("failed to connect to Redis: %w", err) - } - - action, _ := args["action"].(string) - key, _ := args["key"].(string) - - var result interface{} - - switch action { - case "get": - if key == "" { - return "", fmt.Errorf("key is required for GET operation") - } - val, err := client.Get(ctx, key).Result() - if err == redis.Nil { - result = map[string]interface{}{ - "exists": false, - "value": nil, - } - } else if err != nil { - return "", fmt.Errorf("GET failed: %w", err) - } else { - // Try to parse as JSON - var jsonVal interface{} - if json.Unmarshal([]byte(val), &jsonVal) == nil { - result = map[string]interface{}{ - "exists": true, - "value": jsonVal, - } - } else { - result = map[string]interface{}{ - "exists": true, - "value": val, - } - } - } - - case "mget": - keys, ok := args["keys"].([]interface{}) - if !ok || len(keys) == 0 { - return "", fmt.Errorf("keys array is required for MGET operation") - } - strKeys := make([]string, len(keys)) - for i, k := range keys { - strKeys[i] = fmt.Sprintf("%v", k) - } - - vals, err := client.MGet(ctx, strKeys...).Result() - if err != nil { - return "", fmt.Errorf("MGET failed: %w", err) - } - - resultMap := make(map[string]interface{}) - for i, k := range strKeys { - if vals[i] != nil { - resultMap[k] = vals[i] - } else { - resultMap[k] = nil - } - } - result = resultMap - - case "hget": - if key == "" { - return "", fmt.Errorf("key is required for HGET operation") - } - field, _ := args["field"].(string) - if field == "" { - return "", fmt.Errorf("field is required for HGET operation") - } - val, err := client.HGet(ctx, key, field).Result() - if err == redis.Nil { - result = map[string]interface{}{ - "exists": false, - "value": nil, - } - } else if err != nil { - return "", fmt.Errorf("HGET failed: %w", err) - } else { - result = map[string]interface{}{ - "exists": true, - "value": val, - } - } - - case "hgetall": - if key == "" { - return "", fmt.Errorf("key is required for HGETALL operation") - } - val, err := client.HGetAll(ctx, key).Result() - if err != nil { - return "", fmt.Errorf("HGETALL failed: %w", err) - } - result = map[string]interface{}{ - "exists": len(val) > 0, - "fields": val, - } - - case "lrange": - if key == "" { - return "", fmt.Errorf("key is required for LRANGE operation") - } - start := int64(0) - stop := int64(-1) - if s, ok := args["start"].(float64); ok { - start = int64(s) - } - if s, ok := args["stop"].(float64); ok { - stop = int64(s) - } - vals, err := client.LRange(ctx, key, start, stop).Result() - if err != nil { - return "", fmt.Errorf("LRANGE failed: %w", err) - } - result = map[string]interface{}{ - "count": len(vals), - "elements": vals, - } - - case "smembers": - if key == "" { - return "", fmt.Errorf("key is required for SMEMBERS operation") - } - vals, err := client.SMembers(ctx, key).Result() - if err != nil { - return "", fmt.Errorf("SMEMBERS failed: %w", err) - } - result = map[string]interface{}{ - "count": len(vals), - "members": vals, - } - - case "zrange": - if key == "" { - return "", fmt.Errorf("key is required for ZRANGE operation") - } - start := int64(0) - stop := int64(-1) - if s, ok := args["start"].(float64); ok { - start = int64(s) - } - if s, ok := args["stop"].(float64); ok { - stop = int64(s) - } - vals, err := client.ZRangeWithScores(ctx, key, start, stop).Result() - if err != nil { - return "", fmt.Errorf("ZRANGE failed: %w", err) - } - members := make([]map[string]interface{}, len(vals)) - for i, z := range vals { - members[i] = map[string]interface{}{ - "member": z.Member, - "score": z.Score, - } - } - result = map[string]interface{}{ - "count": len(vals), - "members": members, - } - - case "keys": - pattern, _ := args["pattern"].(string) - if pattern == "" { - return "", fmt.Errorf("pattern is required for KEYS operation") - } - // Use SCAN instead of KEYS for safety on large databases - var cursor uint64 - var allKeys []string - for { - keys, nextCursor, err := client.Scan(ctx, cursor, pattern, 100).Result() - if err != nil { - return "", fmt.Errorf("SCAN failed: %w", err) - } - allKeys = append(allKeys, keys...) - cursor = nextCursor - if cursor == 0 { - break - } - // Limit to 1000 keys for safety - if len(allKeys) >= 1000 { - allKeys = allKeys[:1000] - break - } - } - result = map[string]interface{}{ - "count": len(allKeys), - "keys": allKeys, - } - - case "exists": - if key == "" { - return "", fmt.Errorf("key is required for EXISTS operation") - } - exists, err := client.Exists(ctx, key).Result() - if err != nil { - return "", fmt.Errorf("EXISTS failed: %w", err) - } - result = map[string]interface{}{ - "exists": exists > 0, - } - - case "ttl": - if key == "" { - return "", fmt.Errorf("key is required for TTL operation") - } - ttl, err := client.TTL(ctx, key).Result() - if err != nil { - return "", fmt.Errorf("TTL failed: %w", err) - } - result = map[string]interface{}{ - "ttl_seconds": int64(ttl.Seconds()), - "has_expiry": ttl > 0, - "no_expiry": ttl == -1, - "key_not_found": ttl == -2, - } - - case "type": - if key == "" { - return "", fmt.Errorf("key is required for TYPE operation") - } - keyType, err := client.Type(ctx, key).Result() - if err != nil { - return "", fmt.Errorf("TYPE failed: %w", err) - } - result = map[string]interface{}{ - "type": keyType, - } - - default: - return "", fmt.Errorf("unsupported read action: %s", action) - } - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} - -func executeRedisWrite(args map[string]interface{}) (string, error) { - client, err := getRedisClient(args) - if err != nil { - return "", err - } - defer client.Close() - - ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second) - defer cancel() - - // Verify connection - if err := client.Ping(ctx).Err(); err != nil { - return "", fmt.Errorf("failed to connect to Redis: %w", err) - } - - action, _ := args["action"].(string) - key, _ := args["key"].(string) - - var result interface{} - - switch action { - case "set": - if key == "" { - return "", fmt.Errorf("key is required for SET operation") - } - value := args["value"] - if value == nil { - return "", fmt.Errorf("value is required for SET operation") - } - // Serialize complex values to JSON - var strVal string - switch v := value.(type) { - case string: - strVal = v - default: - jsonBytes, _ := json.Marshal(v) - strVal = string(jsonBytes) - } - - err := client.Set(ctx, key, strVal, 0).Err() - if err != nil { - return "", fmt.Errorf("SET failed: %w", err) - } - result = map[string]interface{}{ - "success": true, - "key": key, - } - - case "setex": - if key == "" { - return "", fmt.Errorf("key is required for SETEX operation") - } - value := args["value"] - if value == nil { - return "", fmt.Errorf("value is required for SETEX operation") - } - ttl, ok := args["ttl_seconds"].(float64) - if !ok || ttl <= 0 { - return "", fmt.Errorf("positive ttl_seconds is required for SETEX operation") - } - - var strVal string - switch v := value.(type) { - case string: - strVal = v - default: - jsonBytes, _ := json.Marshal(v) - strVal = string(jsonBytes) - } - - err := client.SetEx(ctx, key, strVal, time.Duration(ttl)*time.Second).Err() - if err != nil { - return "", fmt.Errorf("SETEX failed: %w", err) - } - result = map[string]interface{}{ - "success": true, - "key": key, - "ttl_seconds": int64(ttl), - } - - case "setnx": - if key == "" { - return "", fmt.Errorf("key is required for SETNX operation") - } - value := args["value"] - if value == nil { - return "", fmt.Errorf("value is required for SETNX operation") - } - - var strVal string - switch v := value.(type) { - case string: - strVal = v - default: - jsonBytes, _ := json.Marshal(v) - strVal = string(jsonBytes) - } - - wasSet, err := client.SetNX(ctx, key, strVal, 0).Result() - if err != nil { - return "", fmt.Errorf("SETNX failed: %w", err) - } - result = map[string]interface{}{ - "success": true, - "was_set": wasSet, - "key": key, - } - - case "mset": - values, ok := args["values"].(map[string]interface{}) - if !ok || len(values) == 0 { - return "", fmt.Errorf("values object is required for MSET operation") - } - - pairs := make([]interface{}, 0, len(values)*2) - for k, v := range values { - var strVal string - switch val := v.(type) { - case string: - strVal = val - default: - jsonBytes, _ := json.Marshal(val) - strVal = string(jsonBytes) - } - pairs = append(pairs, k, strVal) - } - - err := client.MSet(ctx, pairs...).Err() - if err != nil { - return "", fmt.Errorf("MSET failed: %w", err) - } - result = map[string]interface{}{ - "success": true, - "count": len(values), - } - - case "hset": - if key == "" { - return "", fmt.Errorf("key is required for HSET operation") - } - - // Support both single field/value and multiple field_values - fieldValues := make(map[string]interface{}) - if fv, ok := args["field_values"].(map[string]interface{}); ok { - fieldValues = fv - } else if field, ok := args["field"].(string); ok && field != "" { - fieldValues[field] = args["value"] - } else { - return "", fmt.Errorf("field/value or field_values is required for HSET operation") - } - - values := make([]interface{}, 0, len(fieldValues)*2) - for k, v := range fieldValues { - var strVal string - switch val := v.(type) { - case string: - strVal = val - default: - jsonBytes, _ := json.Marshal(val) - strVal = string(jsonBytes) - } - values = append(values, k, strVal) - } - - count, err := client.HSet(ctx, key, values...).Result() - if err != nil { - return "", fmt.Errorf("HSET failed: %w", err) - } - result = map[string]interface{}{ - "success": true, - "key": key, - "fields_set": count, - } - - case "lpush": - if key == "" { - return "", fmt.Errorf("key is required for LPUSH operation") - } - members, ok := args["members"].([]interface{}) - if !ok || len(members) == 0 { - return "", fmt.Errorf("members array is required for LPUSH operation") - } - - count, err := client.LPush(ctx, key, members...).Result() - if err != nil { - return "", fmt.Errorf("LPUSH failed: %w", err) - } - result = map[string]interface{}{ - "success": true, - "key": key, - "list_length": count, - } - - case "rpush": - if key == "" { - return "", fmt.Errorf("key is required for RPUSH operation") - } - members, ok := args["members"].([]interface{}) - if !ok || len(members) == 0 { - return "", fmt.Errorf("members array is required for RPUSH operation") - } - - count, err := client.RPush(ctx, key, members...).Result() - if err != nil { - return "", fmt.Errorf("RPUSH failed: %w", err) - } - result = map[string]interface{}{ - "success": true, - "key": key, - "list_length": count, - } - - case "sadd": - if key == "" { - return "", fmt.Errorf("key is required for SADD operation") - } - members, ok := args["members"].([]interface{}) - if !ok || len(members) == 0 { - return "", fmt.Errorf("members array is required for SADD operation") - } - - count, err := client.SAdd(ctx, key, members...).Result() - if err != nil { - return "", fmt.Errorf("SADD failed: %w", err) - } - result = map[string]interface{}{ - "success": true, - "key": key, - "added_count": count, - } - - case "zadd": - if key == "" { - return "", fmt.Errorf("key is required for ZADD operation") - } - scoredMembers, ok := args["scored_members"].([]interface{}) - if !ok || len(scoredMembers) == 0 { - return "", fmt.Errorf("scored_members array is required for ZADD operation") - } - - zMembers := make([]redis.Z, 0, len(scoredMembers)) - for _, sm := range scoredMembers { - if m, ok := sm.(map[string]interface{}); ok { - score, _ := m["score"].(float64) - member := fmt.Sprintf("%v", m["member"]) - zMembers = append(zMembers, redis.Z{Score: score, Member: member}) - } - } - - count, err := client.ZAdd(ctx, key, zMembers...).Result() - if err != nil { - return "", fmt.Errorf("ZADD failed: %w", err) - } - result = map[string]interface{}{ - "success": true, - "key": key, - "added_count": count, - } - - case "incr": - if key == "" { - return "", fmt.Errorf("key is required for INCR operation") - } - newVal, err := client.Incr(ctx, key).Result() - if err != nil { - return "", fmt.Errorf("INCR failed: %w", err) - } - result = map[string]interface{}{ - "success": true, - "key": key, - "new_value": newVal, - } - - case "incrby": - if key == "" { - return "", fmt.Errorf("key is required for INCRBY operation") - } - increment, ok := args["increment"].(float64) - if !ok { - return "", fmt.Errorf("increment is required for INCRBY operation") - } - newVal, err := client.IncrBy(ctx, key, int64(increment)).Result() - if err != nil { - return "", fmt.Errorf("INCRBY failed: %w", err) - } - result = map[string]interface{}{ - "success": true, - "key": key, - "new_value": newVal, - } - - case "expire": - if key == "" { - return "", fmt.Errorf("key is required for EXPIRE operation") - } - ttl, ok := args["ttl_seconds"].(float64) - if !ok || ttl <= 0 { - return "", fmt.Errorf("positive ttl_seconds is required for EXPIRE operation") - } - - wasSet, err := client.Expire(ctx, key, time.Duration(ttl)*time.Second).Result() - if err != nil { - return "", fmt.Errorf("EXPIRE failed: %w", err) - } - result = map[string]interface{}{ - "success": true, - "key": key, - "was_set": wasSet, - "ttl_seconds": int64(ttl), - } - - case "del", "delete": - if key == "" { - return "", fmt.Errorf("key is required for DEL operation") - } - - deleted, err := client.Del(ctx, key).Result() - if err != nil { - return "", fmt.Errorf("DEL failed: %w", err) - } - result = map[string]interface{}{ - "success": true, - "key": key, - "deleted_count": deleted, - } - - default: - return "", fmt.Errorf("unsupported write action: %s", action) - } - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} diff --git a/backend/internal/tools/referralmonk_tool.go b/backend/internal/tools/referralmonk_tool.go deleted file mode 100644 index c65c99b8..00000000 --- a/backend/internal/tools/referralmonk_tool.go +++ /dev/null @@ -1,207 +0,0 @@ -package tools - -import ( - "bytes" - "encoding/json" - "fmt" - "io" - "net/http" - "time" -) - -// NewReferralMonkWhatsAppTool creates a ReferralMonk WhatsApp tool -func NewReferralMonkWhatsAppTool() *Tool { - return &Tool{ - Name: "referralmonk_whatsapp", - DisplayName: "ReferralMonk WhatsApp", - Description: `Send WhatsApp messages via ReferralMonk with template support. - -Features: -- Send templated WhatsApp messages for campaigns -- Support for template parameters (up to 3 parameters) -- International number support with country code -- External message ID tracking for analytics - -Use this for marketing campaigns, nurture flows, and templated notifications. -Numbers must include country code (e.g., 917550002919 for India).`, - Icon: "MessageSquare", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"referralmonk", "whatsapp", "template", "campaign", "message", "marketing", "ahaguru"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "mobile": map[string]interface{}{ - "type": "string", - "description": "Mobile number with country code (e.g., 917550002919 for +91 7550002919)", - }, - "template_name": map[string]interface{}{ - "type": "string", - "description": "WhatsApp template name/ID (e.g., demo_session_01)", - }, - "language": map[string]interface{}{ - "type": "string", - "description": "Language code for template (default: en)", - "default": "en", - }, - "param_1": map[string]interface{}{ - "type": "string", - "description": "First template parameter (e.g., user name)", - }, - "param_2": map[string]interface{}{ - "type": "string", - "description": "Second template parameter (e.g., link or URL)", - }, - "param_3": map[string]interface{}{ - "type": "string", - "description": "Third template parameter (e.g., signature or team name)", - }, - "external_message_id": map[string]interface{}{ - "type": "string", - "description": "External message ID for tracking (auto-generated if not provided)", - }, - }, - "required": []string{"mobile", "template_name"}, - }, - Execute: executeReferralMonkWhatsApp, - } -} - -func executeReferralMonkWhatsApp(args map[string]interface{}) (string, error) { - // Get ReferralMonk credentials - credData, err := GetCredentialData(args, "referralmonk") - if err != nil { - return "", fmt.Errorf("failed to get ReferralMonk credentials: %w", err) - } - - apiToken, _ := credData["api_token"].(string) - apiSecret, _ := credData["api_secret"].(string) - - if apiToken == "" || apiSecret == "" { - return "", fmt.Errorf("ReferralMonk credentials incomplete: api_token and api_secret are required") - } - - // Extract parameters - mobile, _ := args["mobile"].(string) - templateName, _ := args["template_name"].(string) - language, _ := args["language"].(string) - param1, _ := args["param_1"].(string) - param2, _ := args["param_2"].(string) - param3, _ := args["param_3"].(string) - externalMsgID, _ := args["external_message_id"].(string) - - // Validate required fields - if mobile == "" { - return "", fmt.Errorf("'mobile' number is required") - } - if templateName == "" { - return "", fmt.Errorf("'template_name' is required") - } - - // Set defaults - if language == "" { - language = "en" - } - if externalMsgID == "" { - externalMsgID = fmt.Sprintf("msg_%d", time.Now().Unix()) - } - - // Build template parameters array - each param is a separate object - var parameters []map[string]interface{} - if param1 != "" { - parameters = append(parameters, map[string]interface{}{ - "type": "text", - "text": param1, - }) - } - if param2 != "" { - parameters = append(parameters, map[string]interface{}{ - "type": "text", - "text": param2, - }) - } - if param3 != "" { - parameters = append(parameters, map[string]interface{}{ - "type": "text", - "text": param3, - }) - } - - // Build ReferralMonk API payload - payload := map[string]interface{}{ - "template_name": templateName, - "channel": "whatsapp", - "recipients": []map[string]interface{}{ - { - "mobile": mobile, - "language": language, - "externalMessageId": externalMsgID, - "components": []map[string]interface{}{ - { - "type": "body", - "parameters": parameters, - }, - }, - }, - }, - } - - // Marshal to JSON - jsonData, err := json.Marshal(payload) - if err != nil { - return "", fmt.Errorf("failed to marshal request: %w", err) - } - - // Make API request to ReferralMonk - apiURL := "https://ahaguru.referralmonk.com/api/campaign" - req, err := http.NewRequest("POST", apiURL, bytes.NewBuffer(jsonData)) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Api-Token", apiToken) - req.Header.Set("Api-Secret", apiSecret) - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to send request: %w", err) - } - defer resp.Body.Close() - - // Read response - respBody, _ := io.ReadAll(resp.Body) - - // Check for errors - if resp.StatusCode >= 400 { - return "", fmt.Errorf("ReferralMonk API error (status %d): %s", resp.StatusCode, string(respBody)) - } - - // Parse response - var apiResponse map[string]interface{} - if err := json.Unmarshal(respBody, &apiResponse); err != nil { - // If JSON parse fails, return raw response - apiResponse = map[string]interface{}{ - "raw_response": string(respBody), - } - } - - // Build output - output := map[string]interface{}{ - "success": true, - "mobile": mobile, - "template_name": templateName, - "external_message_id": externalMsgID, - "language": language, - "status_code": resp.StatusCode, - "response": apiResponse, - } - - jsonResult, _ := json.MarshalIndent(output, "", " ") - return string(jsonResult), nil -} diff --git a/backend/internal/tools/registry.go b/backend/internal/tools/registry.go deleted file mode 100644 index aad9723e..00000000 --- a/backend/internal/tools/registry.go +++ /dev/null @@ -1,472 +0,0 @@ -package tools - -import ( - "fmt" - "sync" -) - -// ToolSource represents where a tool comes from -type ToolSource string - -const ( - ToolSourceBuiltin ToolSource = "builtin" - ToolSourceMCPLocal ToolSource = "mcp_local" - ToolSourceComposio ToolSource = "composio" -) - -// Tool represents a callable tool with its metadata and execution function -type Tool struct { - Name string - DisplayName string // User-friendly name (e.g., "Search Web", "Calculate Math") - Description string - Parameters map[string]interface{} - Icon string // Lucide React icon name (e.g., "Calculator", "Search", "Clock") - Execute ExecuteFunc - Source ToolSource // "builtin" or "mcp_local" - UserID string // For user-specific MCP tools (empty for built-in) - Category string // Tool category: data_sources, computation, time, output, integration - Keywords []string // Keywords for smart recommendations -} - -// ExecuteFunc is the function signature for tool execution -type ExecuteFunc func(args map[string]interface{}) (string, error) - -// Registry manages all available tools -type Registry struct { - tools map[string]*Tool // Built-in tools (global) - userTools map[string]map[string]*Tool // User-specific tools: userID -> toolName -> Tool - mutex sync.RWMutex -} - -var ( - globalRegistry *Registry - once sync.Once -) - -// GetRegistry returns the global tool registry (singleton) -func GetRegistry() *Registry { - once.Do(func() { - globalRegistry = &Registry{ - tools: make(map[string]*Tool), - userTools: make(map[string]map[string]*Tool), - } - // Register built-in tools - registerBuiltInTools(globalRegistry) - }) - return globalRegistry -} - -// Register adds a new tool to the registry -func (r *Registry) Register(tool *Tool) error { - r.mutex.Lock() - defer r.mutex.Unlock() - - if tool.Name == "" { - return fmt.Errorf("tool name cannot be empty") - } - - if tool.Execute == nil { - return fmt.Errorf("tool %s must have an Execute function", tool.Name) - } - - if _, exists := r.tools[tool.Name]; exists { - return fmt.Errorf("tool %s is already registered", tool.Name) - } - - r.tools[tool.Name] = tool - return nil -} - -// Get retrieves a tool by name -func (r *Registry) Get(name string) (*Tool, bool) { - r.mutex.RLock() - defer r.mutex.RUnlock() - tool, exists := r.tools[name] - return tool, exists -} - -// List returns all registered tools in OpenAI tool format -func (r *Registry) List() []map[string]interface{} { - r.mutex.RLock() - defer r.mutex.RUnlock() - - tools := make([]map[string]interface{}, 0, len(r.tools)) - for _, tool := range r.tools { - tools = append(tools, map[string]interface{}{ - "type": "function", - "function": map[string]interface{}{ - "name": tool.Name, - "description": tool.Description, - "parameters": tool.Parameters, - }, - }) - } - return tools -} - -// Execute runs a tool by name with given arguments -func (r *Registry) Execute(name string, args map[string]interface{}) (string, error) { - tool, exists := r.Get(name) - if !exists { - return "", fmt.Errorf("tool %s not found", name) - } - return tool.Execute(args) -} - -// Count returns the number of registered tools -func (r *Registry) Count() int { - r.mutex.RLock() - defer r.mutex.RUnlock() - return len(r.tools) -} - -// GetToolsByCategory returns all tools in a specific category -func (r *Registry) GetToolsByCategory(category string) []*Tool { - r.mutex.RLock() - defer r.mutex.RUnlock() - - var categoryTools []*Tool - for _, tool := range r.tools { - if tool.Category == category { - categoryTools = append(categoryTools, tool) - } - } - return categoryTools -} - -// GetCategories returns a map of category names to their tool counts -func (r *Registry) GetCategories() map[string]int { - r.mutex.RLock() - defer r.mutex.RUnlock() - - categories := make(map[string]int) - for _, tool := range r.tools { - if tool.Category != "" { - categories[tool.Category]++ - } - } - return categories -} - -// registerBuiltInTools registers the default tools -func registerBuiltInTools(r *Registry) { - // Register time tool - _ = r.Register(NewTimeTool()) - - // Register search tool - _ = r.Register(NewSearchTool()) - - // Register image search tool - _ = r.Register(NewImageSearchTool()) - - // Register math tool - _ = r.Register(NewMathTool()) - - // Register scraper tool - _ = r.Register(NewScraperTool()) - - // Register document tool - _ = r.Register(NewDocumentTool()) - - // Register text file tool - _ = r.Register(NewTextFileTool()) - - // Register E2B-powered tools - _ = r.Register(NewDataAnalystTool()) - _ = r.Register(NewMLTrainerTool()) - _ = r.Register(NewAPITesterTool()) - _ = r.Register(NewPythonRunnerTool()) - _ = r.Register(NewHTMLToPDFTool()) - - // Register integration tools (webhook, discord, slack, telegram, google chat, sendgrid, brevo, zoom, twilio, referralmonk) - _ = r.Register(NewWebhookTool()) - _ = r.Register(NewDiscordTool()) - _ = r.Register(NewSlackTool()) - _ = r.Register(NewTelegramTool()) - _ = r.Register(NewGoogleChatTool()) - _ = r.Register(NewSendGridTool()) - _ = r.Register(NewBrevoTool()) - _ = r.Register(NewZoomTool()) - _ = r.Register(NewTwilioSMSTool()) - _ = r.Register(NewTwilioWhatsAppTool()) - _ = r.Register(NewReferralMonkWhatsAppTool()) - - // Register productivity tools (ClickUp, Calendly) - _ = r.Register(NewClickUpTasksTool()) - _ = r.Register(NewClickUpCreateTaskTool()) - _ = r.Register(NewClickUpUpdateTaskTool()) - _ = r.Register(NewCalendlyEventsTool()) - _ = r.Register(NewCalendlyEventTypesTool()) - _ = r.Register(NewCalendlyInviteesTool()) - - // Register CRM tools (LeadSquared) - _ = r.Register(NewLeadSquaredLeadsTool()) - _ = r.Register(NewLeadSquaredCreateLeadTool()) - _ = r.Register(NewLeadSquaredActivitiesTool()) - - // Register analytics tools (Mixpanel, PostHog) - _ = r.Register(NewMixpanelTrackTool()) - _ = r.Register(NewMixpanelUserProfileTool()) - _ = r.Register(NewPostHogCaptureTool()) - _ = r.Register(NewPostHogIdentifyTool()) - _ = r.Register(NewPostHogQueryTool()) - - // Register e-commerce tools (Shopify) - _ = r.Register(NewShopifyProductsTool()) - _ = r.Register(NewShopifyOrdersTool()) - _ = r.Register(NewShopifyCustomersTool()) - - // Register deployment tools (Netlify) - _ = r.Register(NewNetlifySitesTool()) - _ = r.Register(NewNetlifyDeploysTool()) - _ = r.Register(NewNetlifyTriggerBuildTool()) - - // Register Notion tools - _ = r.Register(NewNotionSearchTool()) - _ = r.Register(NewNotionQueryDatabaseTool()) - _ = r.Register(NewNotionCreatePageTool()) - _ = r.Register(NewNotionUpdatePageTool()) - - // Register GitHub tools - _ = r.Register(NewGitHubCreateIssueTool()) - _ = r.Register(NewGitHubListIssuesTool()) - _ = r.Register(NewGitHubGetRepoTool()) - _ = r.Register(NewGitHubAddCommentTool()) - - // Register Microsoft Teams tools - _ = r.Register(NewTeamsTool()) - - // Register GitLab tools - _ = r.Register(NewGitLabProjectsTool()) - _ = r.Register(NewGitLabIssuesTool()) - _ = r.Register(NewGitLabMRsTool()) - - // Register Linear tools - _ = r.Register(NewLinearIssuesTool()) - _ = r.Register(NewLinearCreateIssueTool()) - _ = r.Register(NewLinearUpdateIssueTool()) - - // Register Jira tools - _ = r.Register(NewJiraIssuesTool()) - _ = r.Register(NewJiraCreateIssueTool()) - _ = r.Register(NewJiraUpdateIssueTool()) - - // Register Airtable tools - _ = r.Register(NewAirtableListTool()) - _ = r.Register(NewAirtableReadTool()) - _ = r.Register(NewAirtableCreateTool()) - _ = r.Register(NewAirtableUpdateTool()) - - // Register Trello tools - _ = r.Register(NewTrelloBoardsTool()) - _ = r.Register(NewTrelloListsTool()) - _ = r.Register(NewTrelloCardsTool()) - _ = r.Register(NewTrelloCreateCardTool()) - - // Register HubSpot tools - _ = r.Register(NewHubSpotContactsTool()) - _ = r.Register(NewHubSpotDealsTool()) - _ = r.Register(NewHubSpotCompaniesTool()) - - // Register Mailchimp tools - _ = r.Register(NewMailchimpListsTool()) - _ = r.Register(NewMailchimpAddSubscriberTool()) - - // Register AWS S3 tools - _ = r.Register(NewS3ListTool()) - _ = r.Register(NewS3UploadTool()) - _ = r.Register(NewS3DownloadTool()) - _ = r.Register(NewS3DeleteTool()) - - // Register REST API tool - _ = r.Register(NewRESTAPITool()) - - // Register X (Twitter) tools - _ = r.Register(NewXSearchPostsTool()) - _ = r.Register(NewXPostTweetTool()) - _ = r.Register(NewXGetUserTool()) - _ = r.Register(NewXGetUserPostsTool()) - - // Register presentation tool - _ = r.Register(NewPresentationTool()) - - // Register file reading tools - _ = r.Register(NewReadDocumentTool()) - _ = r.Register(NewReadDataFileTool()) - _ = r.Register(NewReadSpreadsheetTool()) - - // Register image description tool - _ = r.Register(NewDescribeImageTool()) - - // Register file download tool - _ = r.Register(NewDownloadFileTool()) - - // Register audio transcription tool - _ = r.Register(NewTranscribeAudioTool()) - - // Register image generation tool - _ = r.Register(NewImageGenerationTool()) - - // Register image edit tool - _ = r.Register(NewImageEditTool()) - - // Register MongoDB tools - _ = r.Register(NewMongoDBQueryTool()) - _ = r.Register(NewMongoDBWriteTool()) - - // Register Redis tools - _ = r.Register(NewRedisReadTool()) - _ = r.Register(NewRedisWriteTool()) - - // Register Composio Google Sheets tools - _ = r.Register(NewComposioGoogleSheetsReadTool()) - _ = r.Register(NewComposioGoogleSheetsWriteTool()) - _ = r.Register(NewComposioGoogleSheetsAppendTool()) - _ = r.Register(NewComposioGoogleSheetsCreateTool()) - _ = r.Register(NewComposioGoogleSheetsInfoTool()) - _ = r.Register(NewComposioGoogleSheetsListSheetsTool()) - _ = r.Register(NewComposioGoogleSheetsSearchTool()) - _ = r.Register(NewComposioGoogleSheetsClearTool()) - _ = r.Register(NewComposioGoogleSheetsAddSheetTool()) - _ = r.Register(NewComposioGoogleSheetsDeleteSheetTool()) - _ = r.Register(NewComposioGoogleSheetsFindReplaceTool()) - _ = r.Register(NewComposioGoogleSheetsUpsertRowsTool()) - - // Register Composio Gmail tools - _ = r.Register(NewComposioGmailSendTool()) - _ = r.Register(NewComposioGmailFetchTool()) - _ = r.Register(NewComposioGmailGetMessageTool()) - _ = r.Register(NewComposioGmailReplyTool()) - _ = r.Register(NewComposioGmailCreateDraftTool()) - _ = r.Register(NewComposioGmailSendDraftTool()) - _ = r.Register(NewComposioGmailListDraftsTool()) - _ = r.Register(NewComposioGmailAddLabelTool()) - _ = r.Register(NewComposioGmailListLabelsTool()) - _ = r.Register(NewComposioGmailTrashTool()) - - // Register interactive prompt tool - _ = r.Register(NewAskUserTool()) -} - -// RegisterUserTool adds a user-specific MCP tool -func (r *Registry) RegisterUserTool(userID string, tool *Tool) error { - r.mutex.Lock() - defer r.mutex.Unlock() - - if tool.Name == "" { - return fmt.Errorf("tool name cannot be empty") - } - - if userID == "" { - return fmt.Errorf("user ID cannot be empty for user-specific tools") - } - - // Initialize user's tool map if it doesn't exist - if r.userTools[userID] == nil { - r.userTools[userID] = make(map[string]*Tool) - } - - // Set tool metadata - tool.UserID = userID - tool.Source = ToolSourceMCPLocal - - r.userTools[userID][tool.Name] = tool - return nil -} - -// UnregisterUserTool removes a specific tool for a user -func (r *Registry) UnregisterUserTool(userID string, toolName string) error { - r.mutex.Lock() - defer r.mutex.Unlock() - - if r.userTools[userID] == nil { - return fmt.Errorf("no tools registered for user %s", userID) - } - - delete(r.userTools[userID], toolName) - - // Clean up user's map if empty - if len(r.userTools[userID]) == 0 { - delete(r.userTools, userID) - } - - return nil -} - -// UnregisterAllUserTools removes all tools for a user (on disconnect) -func (r *Registry) UnregisterAllUserTools(userID string) { - r.mutex.Lock() - defer r.mutex.Unlock() - - delete(r.userTools, userID) -} - -// GetUserTools returns all tools available to a specific user (built-in + user's MCP tools) -func (r *Registry) GetUserTools(userID string) []map[string]interface{} { - r.mutex.RLock() - defer r.mutex.RUnlock() - - tools := make([]map[string]interface{}, 0) - - // Add built-in tools - for _, tool := range r.tools { - tools = append(tools, map[string]interface{}{ - "type": "function", - "function": map[string]interface{}{ - "name": tool.Name, - "description": tool.Description, - "parameters": tool.Parameters, - }, - }) - } - - // Add user's MCP tools - if r.userTools[userID] != nil { - for _, tool := range r.userTools[userID] { - tools = append(tools, map[string]interface{}{ - "type": "function", - "function": map[string]interface{}{ - "name": tool.Name, - "description": tool.Description, - "parameters": tool.Parameters, - }, - }) - } - } - - return tools -} - -// GetUserTool retrieves a tool by name for a specific user (checks both built-in and user tools) -func (r *Registry) GetUserTool(userID string, toolName string) (*Tool, bool) { - r.mutex.RLock() - defer r.mutex.RUnlock() - - // Check built-in tools first - if tool, exists := r.tools[toolName]; exists { - return tool, true - } - - // Check user's MCP tools - if r.userTools[userID] != nil { - if tool, exists := r.userTools[userID][toolName]; exists { - return tool, true - } - } - - return nil, false -} - -// CountUserTools returns the count of tools available to a user -func (r *Registry) CountUserTools(userID string) int { - r.mutex.RLock() - defer r.mutex.RUnlock() - - count := len(r.tools) // Built-in tools - - if r.userTools[userID] != nil { - count += len(r.userTools[userID]) - } - - return count -} diff --git a/backend/internal/tools/registry_test.go b/backend/internal/tools/registry_test.go deleted file mode 100644 index 8b64d0b2..00000000 --- a/backend/internal/tools/registry_test.go +++ /dev/null @@ -1,461 +0,0 @@ -package tools - -import ( - "errors" - "sync" - "testing" -) - -func TestNewRegistry(t *testing.T) { - // Note: We can't easily test GetRegistry() since it's a singleton, - // so we'll create fresh registries for testing - registry := &Registry{ - tools: make(map[string]*Tool), - } - - if registry.Count() != 0 { - t.Errorf("Expected 0 tools in new registry, got %d", registry.Count()) - } -} - -func TestRegistry_Register(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - tool := &Tool{ - Name: "test_tool", - Description: "A test tool", - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "input": map[string]interface{}{ - "type": "string", - }, - }, - }, - Execute: func(args map[string]interface{}) (string, error) { - return "success", nil - }, - } - - err := registry.Register(tool) - if err != nil { - t.Fatalf("Failed to register tool: %v", err) - } - - if registry.Count() != 1 { - t.Errorf("Expected 1 tool, got %d", registry.Count()) - } -} - -func TestRegistry_Register_EmptyName(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - tool := &Tool{ - Name: "", - Description: "A test tool", - Execute: func(args map[string]interface{}) (string, error) { - return "success", nil - }, - } - - err := registry.Register(tool) - if err == nil { - t.Error("Expected error for empty tool name, got nil") - } -} - -func TestRegistry_Register_NilExecute(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - tool := &Tool{ - Name: "test_tool", - Description: "A test tool", - Execute: nil, - } - - err := registry.Register(tool) - if err == nil { - t.Error("Expected error for nil Execute function, got nil") - } -} - -func TestRegistry_Register_Duplicate(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - tool := &Tool{ - Name: "test_tool", - Description: "A test tool", - Execute: func(args map[string]interface{}) (string, error) { - return "success", nil - }, - } - - // Register first time - err := registry.Register(tool) - if err != nil { - t.Fatalf("Failed to register tool: %v", err) - } - - // Try to register again - err = registry.Register(tool) - if err == nil { - t.Error("Expected error for duplicate tool registration, got nil") - } -} - -func TestRegistry_Get(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - tool := &Tool{ - Name: "test_tool", - Description: "A test tool", - Execute: func(args map[string]interface{}) (string, error) { - return "success", nil - }, - } - - registry.Register(tool) - - // Get existing tool - retrieved, exists := registry.Get("test_tool") - if !exists { - t.Error("Expected tool to exist") - } - - if retrieved.Name != "test_tool" { - t.Errorf("Expected tool name 'test_tool', got %s", retrieved.Name) - } -} - -func TestRegistry_Get_NotFound(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - _, exists := registry.Get("nonexistent_tool") - if exists { - t.Error("Expected tool to not exist") - } -} - -func TestRegistry_List(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - // Register multiple tools - tools := []*Tool{ - { - Name: "tool1", - Description: "First tool", - Parameters: map[string]interface{}{ - "type": "object", - }, - Execute: func(args map[string]interface{}) (string, error) { - return "success", nil - }, - }, - { - Name: "tool2", - Description: "Second tool", - Parameters: map[string]interface{}{ - "type": "object", - }, - Execute: func(args map[string]interface{}) (string, error) { - return "success", nil - }, - }, - } - - for _, tool := range tools { - registry.Register(tool) - } - - // List tools - toolsList := registry.List() - if len(toolsList) != 2 { - t.Errorf("Expected 2 tools in list, got %d", len(toolsList)) - } - - // Verify format - for _, toolDef := range toolsList { - if toolDef["type"] != "function" { - t.Error("Expected tool type to be 'function'") - } - - function, ok := toolDef["function"].(map[string]interface{}) - if !ok { - t.Fatal("Expected function to be a map") - } - - if function["name"] == nil { - t.Error("Expected function to have a name") - } - - if function["description"] == nil { - t.Error("Expected function to have a description") - } - - if function["parameters"] == nil { - t.Error("Expected function to have parameters") - } - } -} - -func TestRegistry_Execute(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - expectedResult := "test result" - tool := &Tool{ - Name: "test_tool", - Description: "A test tool", - Execute: func(args map[string]interface{}) (string, error) { - return expectedResult, nil - }, - } - - registry.Register(tool) - - // Execute tool - result, err := registry.Execute("test_tool", nil) - if err != nil { - t.Fatalf("Failed to execute tool: %v", err) - } - - if result != expectedResult { - t.Errorf("Expected result %s, got %s", expectedResult, result) - } -} - -func TestRegistry_Execute_WithArgs(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - tool := &Tool{ - Name: "echo_tool", - Description: "Echoes the input", - Execute: func(args map[string]interface{}) (string, error) { - input, ok := args["input"].(string) - if !ok { - return "", errors.New("input must be a string") - } - return input, nil - }, - } - - registry.Register(tool) - - // Execute with args - args := map[string]interface{}{ - "input": "hello world", - } - - result, err := registry.Execute("echo_tool", args) - if err != nil { - t.Fatalf("Failed to execute tool: %v", err) - } - - if result != "hello world" { - t.Errorf("Expected result 'hello world', got %s", result) - } -} - -func TestRegistry_Execute_Error(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - expectedError := errors.New("execution failed") - tool := &Tool{ - Name: "error_tool", - Description: "Always fails", - Execute: func(args map[string]interface{}) (string, error) { - return "", expectedError - }, - } - - registry.Register(tool) - - // Execute tool - _, err := registry.Execute("error_tool", nil) - if err == nil { - t.Error("Expected error from tool execution, got nil") - } - - if err.Error() != expectedError.Error() { - t.Errorf("Expected error %v, got %v", expectedError, err) - } -} - -func TestRegistry_Execute_NotFound(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - _, err := registry.Execute("nonexistent_tool", nil) - if err == nil { - t.Error("Expected error for nonexistent tool, got nil") - } -} - -func TestRegistry_Count(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - if registry.Count() != 0 { - t.Errorf("Expected 0 tools initially, got %d", registry.Count()) - } - - // Register tools - for i := 0; i < 5; i++ { - tool := &Tool{ - Name: string(rune('a' + i)), - Description: "Test tool", - Execute: func(args map[string]interface{}) (string, error) { - return "success", nil - }, - } - registry.Register(tool) - } - - if registry.Count() != 5 { - t.Errorf("Expected 5 tools, got %d", registry.Count()) - } -} - -func TestRegistry_ThreadSafety(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - var wg sync.WaitGroup - numGoroutines := 100 - - // Concurrent registrations - for i := 0; i < numGoroutines; i++ { - wg.Add(1) - go func(id int) { - defer wg.Done() - - tool := &Tool{ - Name: string(rune('a' + (id % 26))), - Description: "Test tool", - Execute: func(args map[string]interface{}) (string, error) { - return "success", nil - }, - } - // Ignore duplicate errors - _ = registry.Register(tool) - }(i) - } - - // Concurrent reads - for i := 0; i < numGoroutines; i++ { - wg.Add(1) - go func(id int) { - defer wg.Done() - registry.Get(string(rune('a' + (id % 26)))) - registry.List() - registry.Count() - }(i) - } - - wg.Wait() - - // Verify registry is still functional - if registry.Count() < 0 || registry.Count() > 26 { - t.Errorf("Unexpected tool count after concurrent operations: %d", registry.Count()) - } -} - -func TestGetRegistry_Singleton(t *testing.T) { - // Get registry multiple times - r1 := GetRegistry() - r2 := GetRegistry() - - // Should be the same instance - if r1 != r2 { - t.Error("Expected GetRegistry() to return the same instance") - } -} - -func TestGetRegistry_HasBuiltInTools(t *testing.T) { - registry := GetRegistry() - - // Check for built-in tools - expectedTools := []string{"get_current_time", "search_web"} - - for _, toolName := range expectedTools { - _, exists := registry.Get(toolName) - if !exists { - t.Errorf("Expected built-in tool %s to be registered", toolName) - } - } -} - -func TestRegistry_List_EmptyRegistry(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - list := registry.List() - if len(list) != 0 { - t.Errorf("Expected empty list, got %d tools", len(list)) - } -} - -func TestTool_ExecuteWithComplexArgs(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - tool := &Tool{ - Name: "complex_tool", - Description: "Handles complex arguments", - Execute: func(args map[string]interface{}) (string, error) { - // Test nested maps - nested, ok := args["nested"].(map[string]interface{}) - if !ok { - return "", errors.New("nested must be a map") - } - - value, ok := nested["key"].(string) - if !ok { - return "", errors.New("nested.key must be a string") - } - - return value, nil - }, - } - - registry.Register(tool) - - args := map[string]interface{}{ - "nested": map[string]interface{}{ - "key": "test_value", - }, - } - - result, err := registry.Execute("complex_tool", args) - if err != nil { - t.Fatalf("Failed to execute tool: %v", err) - } - - if result != "test_value" { - t.Errorf("Expected result 'test_value', got %s", result) - } -} diff --git a/backend/internal/tools/rest_api_tool.go b/backend/internal/tools/rest_api_tool.go deleted file mode 100644 index 0fb73f3a..00000000 --- a/backend/internal/tools/rest_api_tool.go +++ /dev/null @@ -1,208 +0,0 @@ -package tools - -import ( - "bytes" - "encoding/base64" - "encoding/json" - "fmt" - "io" - "net/http" - "strings" - "time" -) - -// NewRESTAPITool creates a generic REST API tool -func NewRESTAPITool() *Tool { - return &Tool{ - Name: "api_request", - DisplayName: "REST API Request", - Description: "Make HTTP requests to any REST API endpoint. Supports various authentication methods. Authentication is configured via credentials.", - Icon: "Globe", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"api", "rest", "http", "request", "get", "post", "put", "delete"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "method": map[string]interface{}{ - "type": "string", - "enum": []string{"GET", "POST", "PUT", "PATCH", "DELETE"}, - "description": "HTTP method (default: GET)", - }, - "endpoint": map[string]interface{}{ - "type": "string", - "description": "API endpoint path (appended to base URL from credentials)", - }, - "body": map[string]interface{}{ - "type": "object", - "description": "Request body for POST/PUT/PATCH requests", - }, - "query_params": map[string]interface{}{ - "type": "object", - "description": "Query parameters as key-value pairs", - }, - "headers": map[string]interface{}{ - "type": "object", - "description": "Additional headers as key-value pairs", - }, - }, - "required": []string{"endpoint"}, - }, - Execute: executeRESTAPI, - } -} - -func executeRESTAPI(args map[string]interface{}) (string, error) { - // Get credential data - credData, err := GetCredentialData(args, "rest_api") - if err != nil { - return "", fmt.Errorf("failed to get REST API credentials: %w", err) - } - - baseURL, _ := credData["base_url"].(string) - if baseURL == "" { - return "", fmt.Errorf("base_url is required in credentials") - } - - authType, _ := credData["auth_type"].(string) - authValue, _ := credData["auth_value"].(string) - authHeaderName, _ := credData["auth_header_name"].(string) - defaultHeaders, _ := credData["headers"].(string) - - // Get request parameters - method := "GET" - if m, ok := args["method"].(string); ok && m != "" { - method = strings.ToUpper(m) - } - - endpoint, _ := args["endpoint"].(string) - if endpoint == "" { - return "", fmt.Errorf("endpoint is required") - } - - // Build URL - url := strings.TrimSuffix(baseURL, "/") - if !strings.HasPrefix(endpoint, "/") { - endpoint = "/" + endpoint - } - url += endpoint - - // Add query parameters - if queryParams, ok := args["query_params"].(map[string]interface{}); ok && len(queryParams) > 0 { - params := []string{} - for k, v := range queryParams { - params = append(params, fmt.Sprintf("%s=%v", k, v)) - } - if strings.Contains(url, "?") { - url += "&" + strings.Join(params, "&") - } else { - url += "?" + strings.Join(params, "&") - } - } - - // Build request body - var reqBody io.Reader - if body, ok := args["body"].(map[string]interface{}); ok && len(body) > 0 { - jsonBody, err := json.Marshal(body) - if err != nil { - return "", fmt.Errorf("failed to marshal request body: %w", err) - } - reqBody = bytes.NewBuffer(jsonBody) - } - - // Create request - req, err := http.NewRequest(method, url, reqBody) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - // Set content type for requests with body - if reqBody != nil { - req.Header.Set("Content-Type", "application/json") - } - req.Header.Set("Accept", "application/json") - - // Parse and apply default headers from credentials - if defaultHeaders != "" { - var defHeaders map[string]string - if err := json.Unmarshal([]byte(defaultHeaders), &defHeaders); err == nil { - for k, v := range defHeaders { - req.Header.Set(k, v) - } - } - } - - // Apply additional headers from request - if headers, ok := args["headers"].(map[string]interface{}); ok { - for k, v := range headers { - if strVal, ok := v.(string); ok { - req.Header.Set(k, strVal) - } - } - } - - // Apply authentication - switch authType { - case "bearer": - req.Header.Set("Authorization", "Bearer "+authValue) - case "basic": - req.Header.Set("Authorization", "Basic "+base64.StdEncoding.EncodeToString([]byte(authValue))) - case "api_key_header": - if authHeaderName == "" { - authHeaderName = "X-API-Key" - } - req.Header.Set(authHeaderName, authValue) - case "api_key_query": - if strings.Contains(req.URL.RawQuery, "?") { - req.URL.RawQuery += "&api_key=" + authValue - } else { - req.URL.RawQuery = "api_key=" + authValue - } - } - - // Execute request - client := &http.Client{Timeout: 60 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - // Read response - respBody, err := io.ReadAll(resp.Body) - if err != nil { - return "", fmt.Errorf("failed to read response: %w", err) - } - - // Try to parse response as JSON - var responseData interface{} - if err := json.Unmarshal(respBody, &responseData); err != nil { - // Not JSON, use as string - responseData = string(respBody) - } - - // Build result - result := map[string]interface{}{ - "success": resp.StatusCode >= 200 && resp.StatusCode < 300, - "status_code": resp.StatusCode, - "status": resp.Status, - "data": responseData, - } - - // Include response headers - respHeaders := map[string]string{} - for k, v := range resp.Header { - if len(v) > 0 { - respHeaders[k] = v[0] - } - } - result["headers"] = respHeaders - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} - diff --git a/backend/internal/tools/s3_tool.go b/backend/internal/tools/s3_tool.go deleted file mode 100644 index 31cf9c58..00000000 --- a/backend/internal/tools/s3_tool.go +++ /dev/null @@ -1,510 +0,0 @@ -package tools - -import ( - "bytes" - "crypto/hmac" - "crypto/sha256" - "encoding/hex" - "encoding/json" - "encoding/xml" - "fmt" - "io" - "net/http" - "sort" - "strings" - "time" -) - -// NewS3ListTool creates a tool for listing S3 objects -func NewS3ListTool() *Tool { - return &Tool{ - Name: "s3_list", - DisplayName: "List S3 Objects", - Description: "List objects in an AWS S3 bucket. Authentication is handled automatically via configured credentials.", - Icon: "FolderOpen", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"aws", "s3", "list", "files", "objects", "bucket"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "bucket": map[string]interface{}{ - "type": "string", - "description": "S3 bucket name", - }, - "prefix": map[string]interface{}{ - "type": "string", - "description": "Filter objects by prefix (folder path)", - }, - "max_keys": map[string]interface{}{ - "type": "number", - "description": "Maximum number of objects to return (default 100)", - }, - }, - "required": []string{"bucket"}, - }, - Execute: executeS3List, - } -} - -// NewS3UploadTool creates a tool for uploading to S3 -func NewS3UploadTool() *Tool { - return &Tool{ - Name: "s3_upload", - DisplayName: "Upload to S3", - Description: "Upload content to an AWS S3 bucket. Authentication is handled automatically.", - Icon: "Upload", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"aws", "s3", "upload", "put", "file"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "bucket": map[string]interface{}{ - "type": "string", - "description": "S3 bucket name", - }, - "key": map[string]interface{}{ - "type": "string", - "description": "Object key (file path in bucket)", - }, - "content": map[string]interface{}{ - "type": "string", - "description": "Content to upload", - }, - "content_type": map[string]interface{}{ - "type": "string", - "description": "Content type (e.g., 'text/plain', 'application/json')", - }, - }, - "required": []string{"bucket", "key", "content"}, - }, - Execute: executeS3Upload, - } -} - -// NewS3DownloadTool creates a tool for downloading from S3 -func NewS3DownloadTool() *Tool { - return &Tool{ - Name: "s3_download", - DisplayName: "Download from S3", - Description: "Download an object from AWS S3. Authentication is handled automatically.", - Icon: "Download", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"aws", "s3", "download", "get", "file"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "bucket": map[string]interface{}{ - "type": "string", - "description": "S3 bucket name", - }, - "key": map[string]interface{}{ - "type": "string", - "description": "Object key (file path in bucket)", - }, - }, - "required": []string{"bucket", "key"}, - }, - Execute: executeS3Download, - } -} - -// NewS3DeleteTool creates a tool for deleting S3 objects -func NewS3DeleteTool() *Tool { - return &Tool{ - Name: "s3_delete", - DisplayName: "Delete S3 Object", - Description: "Delete an object from AWS S3. Authentication is handled automatically.", - Icon: "Trash2", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"aws", "s3", "delete", "remove", "file"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "bucket": map[string]interface{}{ - "type": "string", - "description": "S3 bucket name", - }, - "key": map[string]interface{}{ - "type": "string", - "description": "Object key to delete", - }, - }, - "required": []string{"bucket", "key"}, - }, - Execute: executeS3Delete, - } -} - -type s3Config struct { - AccessKeyID string - SecretAccessKey string - Region string - Bucket string -} - -func getS3Config(args map[string]interface{}) (*s3Config, error) { - credData, err := GetCredentialData(args, "aws_s3") - if err != nil { - return nil, fmt.Errorf("failed to get AWS credentials: %w", err) - } - - accessKey, _ := credData["access_key_id"].(string) - secretKey, _ := credData["secret_access_key"].(string) - region, _ := credData["region"].(string) - bucket, _ := credData["bucket"].(string) - - if accessKey == "" || secretKey == "" { - return nil, fmt.Errorf("access_key_id and secret_access_key are required") - } - - if region == "" { - region = "us-east-1" - } - - return &s3Config{ - AccessKeyID: accessKey, - SecretAccessKey: secretKey, - Region: region, - Bucket: bucket, - }, nil -} - -func hmacSHA256(key []byte, data string) []byte { - h := hmac.New(sha256.New, key) - h.Write([]byte(data)) - return h.Sum(nil) -} - -func hashSHA256(data string) string { - h := sha256.Sum256([]byte(data)) - return hex.EncodeToString(h[:]) -} - -func s3Request(method, bucket, key, region, accessKey, secretKey string, body []byte, contentType string) (*http.Response, error) { - host := fmt.Sprintf("%s.s3.%s.amazonaws.com", bucket, region) - endpoint := fmt.Sprintf("https://%s%s", host, key) - - t := time.Now().UTC() - amzDate := t.Format("20060102T150405Z") - dateStamp := t.Format("20060102") - - var bodyReader io.Reader - payloadHash := hashSHA256("") - if body != nil { - bodyReader = bytes.NewReader(body) - payloadHash = hashSHA256(string(body)) - } - - req, err := http.NewRequest(method, endpoint, bodyReader) - if err != nil { - return nil, err - } - - req.Header.Set("Host", host) - req.Header.Set("X-Amz-Date", amzDate) - req.Header.Set("X-Amz-Content-Sha256", payloadHash) - if contentType != "" { - req.Header.Set("Content-Type", contentType) - } - - // Create canonical request - signedHeaders := []string{"host", "x-amz-content-sha256", "x-amz-date"} - if contentType != "" { - signedHeaders = append(signedHeaders, "content-type") - } - sort.Strings(signedHeaders) - signedHeadersStr := strings.Join(signedHeaders, ";") - - canonicalHeaders := fmt.Sprintf("host:%s\nx-amz-content-sha256:%s\nx-amz-date:%s\n", host, payloadHash, amzDate) - if contentType != "" { - canonicalHeaders = fmt.Sprintf("content-type:%s\nhost:%s\nx-amz-content-sha256:%s\nx-amz-date:%s\n", contentType, host, payloadHash, amzDate) - } - - canonicalRequest := fmt.Sprintf("%s\n%s\n%s\n%s\n%s\n%s", - method, - key, - req.URL.RawQuery, - canonicalHeaders, - signedHeadersStr, - payloadHash, - ) - - // Create string to sign - algorithm := "AWS4-HMAC-SHA256" - credentialScope := fmt.Sprintf("%s/%s/s3/aws4_request", dateStamp, region) - stringToSign := fmt.Sprintf("%s\n%s\n%s\n%s", - algorithm, - amzDate, - credentialScope, - hashSHA256(canonicalRequest), - ) - - // Calculate signature - kDate := hmacSHA256([]byte("AWS4"+secretKey), dateStamp) - kRegion := hmacSHA256(kDate, region) - kService := hmacSHA256(kRegion, "s3") - kSigning := hmacSHA256(kService, "aws4_request") - signature := hex.EncodeToString(hmacSHA256(kSigning, stringToSign)) - - // Add authorization header - authHeader := fmt.Sprintf("%s Credential=%s/%s, SignedHeaders=%s, Signature=%s", - algorithm, - accessKey, - credentialScope, - signedHeadersStr, - signature, - ) - req.Header.Set("Authorization", authHeader) - - client := &http.Client{Timeout: 60 * time.Second} - return client.Do(req) -} - -type listBucketResult struct { - XMLName xml.Name `xml:"ListBucketResult"` - Contents []struct { - Key string `xml:"Key"` - LastModified string `xml:"LastModified"` - Size int64 `xml:"Size"` - StorageClass string `xml:"StorageClass"` - } `xml:"Contents"` -} - -func executeS3List(args map[string]interface{}) (string, error) { - cfg, err := getS3Config(args) - if err != nil { - return "", err - } - - bucket, _ := args["bucket"].(string) - if bucket == "" { - bucket = cfg.Bucket - } - if bucket == "" { - return "", fmt.Errorf("bucket is required") - } - - key := "/" - if prefix, ok := args["prefix"].(string); ok && prefix != "" { - key = "/?prefix=" + prefix - } - - maxKeys := 100 - if mk, ok := args["max_keys"].(float64); ok && mk > 0 { - maxKeys = int(mk) - } - if strings.Contains(key, "?") { - key += fmt.Sprintf("&max-keys=%d", maxKeys) - } else { - key += fmt.Sprintf("?max-keys=%d", maxKeys) - } - - resp, err := s3Request("GET", bucket, key, cfg.Region, cfg.AccessKeyID, cfg.SecretAccessKey, nil, "") - if err != nil { - return "", fmt.Errorf("S3 request failed: %w", err) - } - defer resp.Body.Close() - - body, err := io.ReadAll(resp.Body) - if err != nil { - return "", fmt.Errorf("failed to read response: %w", err) - } - - if resp.StatusCode >= 400 { - return "", fmt.Errorf("S3 error (status %d): %s", resp.StatusCode, string(body)) - } - - var result listBucketResult - if err := xml.Unmarshal(body, &result); err != nil { - return "", fmt.Errorf("failed to parse response: %w", err) - } - - objects := make([]map[string]interface{}, 0) - for _, obj := range result.Contents { - objects = append(objects, map[string]interface{}{ - "key": obj.Key, - "size": obj.Size, - "last_modified": obj.LastModified, - "storage_class": obj.StorageClass, - }) - } - - response := map[string]interface{}{ - "success": true, - "bucket": bucket, - "count": len(objects), - "objects": objects, - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - -func executeS3Upload(args map[string]interface{}) (string, error) { - cfg, err := getS3Config(args) - if err != nil { - return "", err - } - - bucket, _ := args["bucket"].(string) - if bucket == "" { - bucket = cfg.Bucket - } - if bucket == "" { - return "", fmt.Errorf("bucket is required") - } - - key, _ := args["key"].(string) - content, _ := args["content"].(string) - if key == "" || content == "" { - return "", fmt.Errorf("key and content are required") - } - - if !strings.HasPrefix(key, "/") { - key = "/" + key - } - - contentType := "text/plain" - if ct, ok := args["content_type"].(string); ok && ct != "" { - contentType = ct - } - - resp, err := s3Request("PUT", bucket, key, cfg.Region, cfg.AccessKeyID, cfg.SecretAccessKey, []byte(content), contentType) - if err != nil { - return "", fmt.Errorf("S3 request failed: %w", err) - } - defer resp.Body.Close() - - if resp.StatusCode >= 400 { - body, _ := io.ReadAll(resp.Body) - return "", fmt.Errorf("S3 error (status %d): %s", resp.StatusCode, string(body)) - } - - response := map[string]interface{}{ - "success": true, - "message": "Object uploaded successfully", - "bucket": bucket, - "key": strings.TrimPrefix(key, "/"), - "size": len(content), - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - -func executeS3Download(args map[string]interface{}) (string, error) { - cfg, err := getS3Config(args) - if err != nil { - return "", err - } - - bucket, _ := args["bucket"].(string) - if bucket == "" { - bucket = cfg.Bucket - } - if bucket == "" { - return "", fmt.Errorf("bucket is required") - } - - key, _ := args["key"].(string) - if key == "" { - return "", fmt.Errorf("key is required") - } - - if !strings.HasPrefix(key, "/") { - key = "/" + key - } - - resp, err := s3Request("GET", bucket, key, cfg.Region, cfg.AccessKeyID, cfg.SecretAccessKey, nil, "") - if err != nil { - return "", fmt.Errorf("S3 request failed: %w", err) - } - defer resp.Body.Close() - - body, err := io.ReadAll(resp.Body) - if err != nil { - return "", fmt.Errorf("failed to read response: %w", err) - } - - if resp.StatusCode >= 400 { - return "", fmt.Errorf("S3 error (status %d): %s", resp.StatusCode, string(body)) - } - - response := map[string]interface{}{ - "success": true, - "bucket": bucket, - "key": strings.TrimPrefix(key, "/"), - "content": string(body), - "size": len(body), - "content_type": resp.Header.Get("Content-Type"), - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - -func executeS3Delete(args map[string]interface{}) (string, error) { - cfg, err := getS3Config(args) - if err != nil { - return "", err - } - - bucket, _ := args["bucket"].(string) - if bucket == "" { - bucket = cfg.Bucket - } - if bucket == "" { - return "", fmt.Errorf("bucket is required") - } - - key, _ := args["key"].(string) - if key == "" { - return "", fmt.Errorf("key is required") - } - - if !strings.HasPrefix(key, "/") { - key = "/" + key - } - - resp, err := s3Request("DELETE", bucket, key, cfg.Region, cfg.AccessKeyID, cfg.SecretAccessKey, nil, "") - if err != nil { - return "", fmt.Errorf("S3 request failed: %w", err) - } - defer resp.Body.Close() - - if resp.StatusCode >= 400 { - body, _ := io.ReadAll(resp.Body) - return "", fmt.Errorf("S3 error (status %d): %s", resp.StatusCode, string(body)) - } - - response := map[string]interface{}{ - "success": true, - "message": "Object deleted successfully", - "bucket": bucket, - "key": strings.TrimPrefix(key, "/"), - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - diff --git a/backend/internal/tools/scraper_tool.go b/backend/internal/tools/scraper_tool.go deleted file mode 100644 index 9c61ee28..00000000 --- a/backend/internal/tools/scraper_tool.go +++ /dev/null @@ -1,235 +0,0 @@ -package tools - -import ( - "bytes" - "claraverse/internal/security" - "context" - "fmt" - "io" - "net/http" - "net/url" - "time" - - "github.com/markusmobius/go-trafilatura" - cache "github.com/patrickmn/go-cache" - "github.com/temoto/robotstxt" - "golang.org/x/time/rate" -) - -// ScraperTool provides web scraping capabilities -type ScraperTool struct { - cache *cache.Cache - rateLimiter *rate.Limiter - client *http.Client -} - -var scraperToolInstance *ScraperTool - -func init() { - scraperToolInstance = &ScraperTool{ - cache: cache.New(1*time.Hour, 10*time.Minute), - rateLimiter: rate.NewLimiter(rate.Limit(10.0), 20), // 10 req/s global - client: &http.Client{ - Timeout: 60 * time.Second, - }, - } -} - -// NewScraperTool creates the scrape_web tool -func NewScraperTool() *Tool { - return &Tool{ - Name: "scrape_web", - DisplayName: "Scrape Web Page", - Description: "Extract clean, readable content from a web page URL. Returns main article content without ads, navigation, or other boilerplate. Respects robots.txt and rate limits. Best for articles, blog posts, and documentation pages.", - Icon: "Globe", - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "url": map[string]interface{}{ - "type": "string", - "description": "The URL of the web page to scrape (must be a valid HTTP/HTTPS URL)", - }, - "max_length": map[string]interface{}{ - "type": "number", - "description": "Optional maximum content length in characters (default: 50000, max: 100000)", - "default": 50000, - }, - "format": map[string]interface{}{ - "type": "string", - "description": "Output format: 'markdown' or 'text' (default: markdown)", - "enum": []string{"markdown", "text"}, - "default": "markdown", - }, - }, - "required": []string{"url"}, - }, - Execute: executeScrapWeb, - Source: ToolSourceBuiltin, - Category: "data_sources", - Keywords: []string{"scrape", "fetch", "extract", "web", "page", "content", "article", "url", "website", "html", "crawl"}, - } -} - -func executeScrapWeb(args map[string]interface{}) (string, error) { - // Extract URL parameter - urlStr, ok := args["url"].(string) - if !ok || urlStr == "" { - return "", fmt.Errorf("url parameter is required and must be a string") - } - - // Validate URL - if err := validateURL(urlStr); err != nil { - return "", err - } - - // Extract max_length parameter - maxLength := 50000 - if ml, ok := args["max_length"].(float64); ok { - if ml > 100000 { - ml = 100000 - } - if ml < 1000 { - ml = 1000 - } - maxLength = int(ml) - } - - // Extract format parameter - format := "markdown" - if f, ok := args["format"].(string); ok { - if f == "text" || f == "markdown" { - format = f - } - } - - // Check cache - cacheKey := fmt.Sprintf("%s:%s", urlStr, format) - if cached, found := scraperToolInstance.cache.Get(cacheKey); found { - return cached.(string), nil - } - - // Check robots.txt - if allowed, err := checkRobots(urlStr); err == nil && !allowed { - return "", fmt.Errorf("blocked by robots.txt") - } - - // Apply rate limiting - ctx, cancel := context.WithTimeout(context.Background(), 60*time.Second) - defer cancel() - - if err := scraperToolInstance.rateLimiter.Wait(ctx); err != nil { - return "", fmt.Errorf("rate limit exceeded, try again later") - } - - // Fetch URL - req, err := http.NewRequestWithContext(ctx, "GET", urlStr, nil) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("User-Agent", "ClaraVerse-Bot/1.0 (+https://claraverse.example.com/bot)") - req.Header.Set("Accept", "text/html,application/xhtml+xml") - - resp, err := scraperToolInstance.client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to fetch URL: %w", err) - } - defer resp.Body.Close() - - if resp.StatusCode != 200 { - return "", fmt.Errorf("HTTP error %d: %s", resp.StatusCode, resp.Status) - } - - // Read body with limit - body, err := io.ReadAll(io.LimitReader(resp.Body, 10*1024*1024)) // 10MB limit - if err != nil { - return "", fmt.Errorf("failed to read response: %w", err) - } - - // Extract content - parsedURL, _ := url.Parse(urlStr) - opts := trafilatura.Options{ - OriginalURL: parsedURL, - } - - result, err := trafilatura.Extract(bytes.NewReader(body), opts) - if err != nil || result == nil || result.ContentText == "" { - return "", fmt.Errorf("failed to extract content from page") - } - - // Use extracted content (already plain text or markdown) - content := result.ContentText - - // Apply length limit - if len(content) > maxLength { - content = content[:maxLength] + "\n\n[Content truncated due to length limit]" - } - - // Add metadata - metadata := fmt.Sprintf("# %s\n\n", result.Metadata.Title) - if result.Metadata.Author != "" { - metadata += fmt.Sprintf("**Author:** %s \n", result.Metadata.Author) - } - if !result.Metadata.Date.IsZero() { - metadata += fmt.Sprintf("**Published:** %s \n", result.Metadata.Date.Format("January 2, 2006")) - } - metadata += fmt.Sprintf("**Source:** %s \n\n---\n\n", urlStr) - - finalContent := metadata + content - - // Cache result - scraperToolInstance.cache.Set(cacheKey, finalContent, cache.DefaultExpiration) - - return finalContent, nil -} - -func validateURL(urlStr string) error { - // Use centralized SSRF protection which includes: - // - Private IP range blocking (10.x, 172.16-31.x, 192.168.x, etc.) - // - Localhost/loopback blocking - // - Cloud metadata endpoint blocking (169.254.169.254, metadata.google.internal) - // - DNS resolution checks to catch hostname-based bypasses - // - IPv6 private address blocking - return security.ValidateURLForSSRF(urlStr) -} - -func checkRobots(urlStr string) (bool, error) { - parsedURL, err := url.Parse(urlStr) - if err != nil { - return true, err - } - - robotsURL := parsedURL.Scheme + "://" + parsedURL.Host + "/robots.txt" - - client := &http.Client{Timeout: 10 * time.Second} - resp, err := client.Get(robotsURL) - if err != nil { - return true, nil // Allow if robots.txt doesn't exist - } - defer resp.Body.Close() - - if resp.StatusCode != 200 { - return true, nil - } - - body, err := io.ReadAll(io.LimitReader(resp.Body, 1*1024*1024)) - if err != nil { - return true, nil - } - - robotsData, err := robotstxt.FromBytes(body) - if err != nil { - return true, nil - } - - group := robotsData.FindGroup("ClaraVerse-Bot") - if group == nil { - group = robotsData.FindGroup("*") - } - - if group != nil { - return group.Test(parsedURL.Path), nil - } - - return true, nil -} diff --git a/backend/internal/tools/search_tool.go b/backend/internal/tools/search_tool.go deleted file mode 100644 index 8d67300a..00000000 --- a/backend/internal/tools/search_tool.go +++ /dev/null @@ -1,355 +0,0 @@ -package tools - -import ( - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "net/url" - "os" - "strings" - "sync" - "sync/atomic" - "time" -) - -// SearchBalancer manages round-robin load balancing across multiple SearXNG instances -type SearchBalancer struct { - urls []string - counter uint64 -} - -// Search result cache with TTL -type searchCache struct { - mu sync.RWMutex - cache map[string]*cacheEntry - maxSize int -} - -type cacheEntry struct { - result string - timestamp time.Time - ttl time.Duration -} - -var searchBalancer *SearchBalancer -var globalSearchCache = &searchCache{ - cache: make(map[string]*cacheEntry), - maxSize: 100, // Store up to 100 recent searches -} - -func init() { - searchBalancer = &SearchBalancer{} - searchBalancer.loadURLs() -} - -// loadURLs loads SearXNG URLs from environment variables -func (sb *SearchBalancer) loadURLs() { - // Check for SEARXNG_URLS (comma-separated list) first - urlsEnv := os.Getenv("SEARXNG_URLS") - if urlsEnv != "" { - urls := strings.Split(urlsEnv, ",") - for _, u := range urls { - trimmed := strings.TrimSpace(u) - if trimmed != "" { - // Normalize URL (remove trailing slash) - trimmed = strings.TrimSuffix(trimmed, "/") - sb.urls = append(sb.urls, trimmed) - } - } - } - - // Fallback to single SEARXNG_URL if SEARXNG_URLS is not set - if len(sb.urls) == 0 { - singleURL := os.Getenv("SEARXNG_URL") - if singleURL == "" { - singleURL = "http://localhost:8080" - } - singleURL = strings.TrimSuffix(singleURL, "/") - sb.urls = append(sb.urls, singleURL) - } - - log.Printf("🔍 [SEARCH] Initialized round-robin balancer with %d SearXNG instance(s): %v", len(sb.urls), sb.urls) -} - -// getNextURL returns the next URL in round-robin fashion -func (sb *SearchBalancer) getNextURL() string { - if len(sb.urls) == 0 { - return "http://localhost:8080" - } - idx := atomic.AddUint64(&sb.counter, 1) - 1 - return sb.urls[idx%uint64(len(sb.urls))] -} - -// getURLCount returns the number of available URLs -func (sb *SearchBalancer) getURLCount() int { - return len(sb.urls) -} - -// getURLAtIndex returns URL at specific index (for retry logic) -func (sb *SearchBalancer) getURLAtIndex(startIdx uint64, offset int) string { - if len(sb.urls) == 0 { - return "http://localhost:8080" - } - idx := (startIdx + uint64(offset)) % uint64(len(sb.urls)) - return sb.urls[idx] -} - -func (c *searchCache) get(key string) (string, bool) { - c.mu.RLock() - defer c.mu.RUnlock() - - entry, exists := c.cache[key] - if !exists { - return "", false - } - - // Check if cache entry is still valid - if time.Since(entry.timestamp) > entry.ttl { - return "", false - } - - return entry.result, true -} - -func (c *searchCache) set(key, result string, ttl time.Duration) { - c.mu.Lock() - defer c.mu.Unlock() - - // Simple LRU: if cache is full, remove oldest entry - if len(c.cache) >= c.maxSize { - var oldestKey string - var oldestTime time.Time - for k, v := range c.cache { - if oldestKey == "" || v.timestamp.Before(oldestTime) { - oldestKey = k - oldestTime = v.timestamp - } - } - delete(c.cache, oldestKey) - } - - c.cache[key] = &cacheEntry{ - result: result, - timestamp: time.Now(), - ttl: ttl, - } -} - -// NewSearchTool creates the search_web tool -func NewSearchTool() *Tool { - return &Tool{ - Name: "search_web", - DisplayName: "Search Web", - Description: "Search the web using SearXNG for current information, news, articles, or any topic", - Icon: "Search", - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "query": map[string]interface{}{ - "type": "string", - "description": "The search query to look up on the web", - }, - }, - "required": []string{"query"}, - }, - Execute: executeSearchWeb, - Source: ToolSourceBuiltin, - Category: "data_sources", - Keywords: []string{"search", "find", "lookup", "research", "web", "internet", "news", "articles", "information", "query", "google"}, - } -} - -func executeSearchWeb(args map[string]interface{}) (string, error) { - query, ok := args["query"].(string) - if !ok { - return "", fmt.Errorf("query parameter is required and must be a string") - } - - query = strings.TrimSpace(query) - log.Printf("🔍 [SEARCH-WEB] Starting search for: '%s'", query) - - // Check cache first (5 minute TTL) - cacheKey := strings.ToLower(query) - if cached, found := globalSearchCache.get(cacheKey); found { - log.Printf("✅ [SEARCH-WEB] Cache hit for: '%s'", query) - return cached, nil - } - - // Get the starting index for this request - startIdx := atomic.LoadUint64(&searchBalancer.counter) - urlCount := searchBalancer.getURLCount() - - // Try search with round-robin load balancing - result, err := searchWithBalancer(query, startIdx, urlCount) - if err != nil || strings.Contains(result, "No results found") { - // Try simplified query - log.Printf("⚠️ [SEARCH-WEB] Original query failed, trying simplified version") - simplifiedQuery := simplifyQuery(query) - if simplifiedQuery != query { - log.Printf("🔄 [SEARCH-WEB] Simplified query: '%s' -> '%s'", query, simplifiedQuery) - result, err = searchWithBalancer(simplifiedQuery, startIdx, urlCount) - } - } - - if err != nil { - log.Printf("❌ [SEARCH-WEB] Search failed after retries: %v", err) - return "", err - } - - // Cache successful results - if !strings.Contains(result, "No results found") { - globalSearchCache.set(cacheKey, result, 5*time.Minute) - log.Printf("✅ [SEARCH-WEB] Cached results for: '%s'", query) - } - - return result, nil -} - -// searchWithBalancer performs search using round-robin load balancing across multiple instances -func searchWithBalancer(query string, startIdx uint64, urlCount int) (string, error) { - var lastErr error - - // Try each SearXNG instance in round-robin order - for attempt := 0; attempt < urlCount; attempt++ { - // Get next URL (first attempt uses round-robin, subsequent attempts try next in sequence) - var searxngURL string - if attempt == 0 { - searxngURL = searchBalancer.getNextURL() - } else { - searxngURL = searchBalancer.getURLAtIndex(startIdx, attempt) - } - - log.Printf("🔍 [SEARCH] Attempt %d/%d using SearXNG instance: %s", attempt+1, urlCount, searxngURL) - - result, err := performSearch(searxngURL, query) - if err == nil { - log.Printf("✅ [SEARCH] Success with instance: %s", searxngURL) - return result, nil - } - - log.Printf("⚠️ [SEARCH] Instance %s failed: %v", searxngURL, err) - lastErr = err - - // Brief delay before trying next instance - if attempt < urlCount-1 { - time.Sleep(100 * time.Millisecond) - } - } - - return "", fmt.Errorf("all %d SearXNG instances failed, last error: %v", urlCount, lastErr) -} - -// performSearch executes a single search request -func performSearch(searxngURL, query string) (string, error) { - // Build search URL (let SearXNG use all enabled engines for better redundancy) - searchURL := fmt.Sprintf("%s/search?q=%s&format=json&safesearch=1", - searxngURL, url.QueryEscape(query)) - - // Make HTTP request with required headers - req, err := http.NewRequest("GET", searchURL, nil) - if err != nil { - return "", fmt.Errorf("failed to create request: %v", err) - } - req.Header.Set("User-Agent", "ClaraVerse/1.0 (Bot)") - req.Header.Set("Accept", "application/json") - req.Header.Set("X-Forwarded-For", "127.0.0.1") - req.Header.Set("X-Real-IP", "127.0.0.1") - - client := &http.Client{ - Timeout: 30 * time.Second, // Add timeout to prevent hanging - } - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("search request failed: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != http.StatusOK { - return "", fmt.Errorf("search failed with status: %d", resp.StatusCode) - } - - // Parse response - body, err := io.ReadAll(resp.Body) - if err != nil { - return "", fmt.Errorf("failed to read search response: %v", err) - } - - var searchResults struct { - Results []struct { - Title string `json:"title"` - URL string `json:"url"` - Content string `json:"content"` - } `json:"results"` - } - - if err := json.Unmarshal(body, &searchResults); err != nil { - return "", fmt.Errorf("failed to parse search results: %v", err) - } - - if len(searchResults.Results) == 0 { - return "No results found for your query.", nil - } - - // Format results (limit to top 10 for better coverage) - result := fmt.Sprintf("Found %d results for '%s':\n\n", len(searchResults.Results), query) - maxResults := 10 - if len(searchResults.Results) < maxResults { - maxResults = len(searchResults.Results) - } - - for i := 0; i < maxResults; i++ { - res := searchResults.Results[i] - result += fmt.Sprintf("[%d] %s\n URL: %s\n %s\n\n", - i+1, res.Title, res.URL, res.Content) - } - - // Add citation reference section for easy copying - result += "\n---\nSOURCES FOR CITATION (use these in your response):\n" - for i := 0; i < maxResults; i++ { - res := searchResults.Results[i] - result += fmt.Sprintf("[%d]: [%s](%s)\n", i+1, res.Title, res.URL) - } - - log.Printf("✅ [SEARCH-WEB] Found %d results for '%s'", len(searchResults.Results), query) - return result, nil -} - -// simplifyQuery removes complex filters and date ranges to get broader results -func simplifyQuery(query string) string { - // Remove years (2024, 2025, etc.) - query = strings.ReplaceAll(query, "2024", "") - query = strings.ReplaceAll(query, "2025", "") - - // Remove common date-related words - dateWords := []string{"latest", "recent", "new", "updates", "update", "news"} - for _, word := range dateWords { - query = strings.ReplaceAll(query, " "+word, "") - query = strings.ReplaceAll(query, word+" ", "") - } - - // Remove version numbers (v0.1.2, 0.1.2, etc.) - query = strings.ReplaceAll(query, "v0.1.2", "") - query = strings.ReplaceAll(query, "0.1.2", "") - - // Remove release-related words - releaseWords := []string{"release", "released", "version"} - for _, word := range releaseWords { - query = strings.ReplaceAll(query, " "+word, "") - query = strings.ReplaceAll(query, word+" ", "") - } - - // Remove quotes - query = strings.ReplaceAll(query, "\"", "") - - // Clean up multiple spaces - query = strings.Join(strings.Fields(query), " ") - - // If query is now too short, return original - if len(strings.TrimSpace(query)) < 3 { - return strings.TrimSpace(query) - } - - return strings.TrimSpace(query) -} diff --git a/backend/internal/tools/sendgrid_tool.go b/backend/internal/tools/sendgrid_tool.go deleted file mode 100644 index dff56a44..00000000 --- a/backend/internal/tools/sendgrid_tool.go +++ /dev/null @@ -1,621 +0,0 @@ -package tools - -import ( - "bytes" - "encoding/base64" - "encoding/json" - "fmt" - "io" - "mime" - "net/http" - "os" - "path/filepath" - "strings" - "sync" - "time" -) - -// Email rate limiter - prevents sending more than 1 email per recipient per minute -var ( - emailRateLimiter = make(map[string]time.Time) - rateLimiterMutex sync.RWMutex - rateLimitWindow = 1 * time.Minute -) - -// checkRateLimit checks if an email can be sent to the recipient -// Returns true if allowed, false if rate limited -func checkRateLimit(email string) (bool, time.Duration) { - rateLimiterMutex.RLock() - lastSent, exists := emailRateLimiter[strings.ToLower(email)] - rateLimiterMutex.RUnlock() - - if !exists { - return true, 0 - } - - elapsed := time.Since(lastSent) - if elapsed < rateLimitWindow { - return false, rateLimitWindow - elapsed - } - - return true, 0 -} - -// recordEmailSent records that an email was sent to a recipient -func recordEmailSent(email string) { - rateLimiterMutex.Lock() - emailRateLimiter[strings.ToLower(email)] = time.Now() - rateLimiterMutex.Unlock() -} - -// cleanupOldRateLimits removes expired entries (called periodically) -func cleanupOldRateLimits() { - rateLimiterMutex.Lock() - defer rateLimiterMutex.Unlock() - - now := time.Now() - for email, lastSent := range emailRateLimiter { - if now.Sub(lastSent) > rateLimitWindow { - delete(emailRateLimiter, email) - } - } -} - -// NewSendGridTool creates a SendGrid email sending tool -func NewSendGridTool() *Tool { - // Start a background cleanup goroutine - go func() { - ticker := time.NewTicker(5 * time.Minute) - for range ticker.C { - cleanupOldRateLimits() - } - }() - - return &Tool{ - Name: "send_email", - DisplayName: "Send Email (SendGrid)", - Description: `Send emails via SendGrid API. Supports plain text and HTML emails with file attachments. - -Features: -- Send to single or multiple recipients (to, cc, bcc) -- HTML and plain text email bodies -- Custom sender name and reply-to address -- File attachments via URL (supports generated PDFs, secure files, and external URLs) -- Rate limited: 1 email per recipient per minute to prevent spam - -ATTACHMENTS: To attach files (like generated PDFs), use file_url with the full download URL. -For secure files, use the download_url returned by file-generating tools (e.g., /api/files/{id}?code={code}). - -Authentication is handled automatically via configured SendGrid credentials. Do NOT ask users for API keys. -The sender email (from_email) can be configured in credentials as default, or overridden per-request. - -IMPORTANT: This tool has rate limiting. If you've already sent an email to a recipient within the last minute, the tool will return a message indicating the email was already sent. Do NOT retry sending - wait for the cooldown period.`, - Icon: "Mail", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"email", "sendgrid", "send", "mail", "message", "notification", "newsletter", "transactional", "attachment"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "api_key": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Resolved from credentials. Do not ask user for this.", - }, - "to": map[string]interface{}{ - "type": "string", - "description": "Recipient email address(es). For multiple recipients, separate with commas (e.g., 'user1@example.com, user2@example.com')", - }, - "from_email": map[string]interface{}{ - "type": "string", - "description": "Sender email address (optional). Must be a verified sender in SendGrid. If not provided, uses the default from_email configured in credentials.", - }, - "from_name": map[string]interface{}{ - "type": "string", - "description": "Sender display name (optional, e.g., 'John Doe' or 'My Company')", - }, - "subject": map[string]interface{}{ - "type": "string", - "description": "Email subject line", - }, - "text_content": map[string]interface{}{ - "type": "string", - "description": "Plain text email body. Either text_content or html_content (or both) must be provided.", - }, - "html_content": map[string]interface{}{ - "type": "string", - "description": "HTML email body for rich formatting. Either text_content or html_content (or both) must be provided.", - }, - "cc": map[string]interface{}{ - "type": "string", - "description": "CC recipient(s). For multiple, separate with commas.", - }, - "bcc": map[string]interface{}{ - "type": "string", - "description": "BCC recipient(s). For multiple, separate with commas.", - }, - "reply_to": map[string]interface{}{ - "type": "string", - "description": "Reply-to email address (optional)", - }, - "file_url": map[string]interface{}{ - "type": "string", - "description": "URL to download a file to attach. Supports both absolute URLs and relative paths starting with /api/files/. For secure files generated by other tools, use the full download_url including the access code (e.g., '/api/files/{id}?code={code}').", - }, - "file_name": map[string]interface{}{ - "type": "string", - "description": "Filename for the attached file (optional, will be inferred from URL or Content-Disposition if not provided)", - }, - "attachments": map[string]interface{}{ - "type": "array", - "description": "Alternative: File attachments as array of objects with 'content' (base64), 'filename', and 'type' (mime type). Use file_url for simpler attachment.", - "items": map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "content": map[string]interface{}{ - "type": "string", - "description": "Base64 encoded file content", - }, - "filename": map[string]interface{}{ - "type": "string", - "description": "Filename with extension", - }, - "type": map[string]interface{}{ - "type": "string", - "description": "MIME type (e.g., 'application/pdf', 'image/png')", - }, - }, - }, - }, - }, - "required": []string{"to", "subject"}, - }, - Execute: executeSendGridEmail, - } -} - -// SendGridPersonalization represents email recipient personalization -type SendGridPersonalization struct { - To []SendGridEmail `json:"to"` - CC []SendGridEmail `json:"cc,omitempty"` - BCC []SendGridEmail `json:"bcc,omitempty"` -} - -// SendGridEmail represents an email address with optional name -type SendGridEmail struct { - Email string `json:"email"` - Name string `json:"name,omitempty"` -} - -// SendGridContent represents email content (text or html) -type SendGridContent struct { - Type string `json:"type"` - Value string `json:"value"` -} - -// SendGridAttachment represents an email attachment -type SendGridAttachment struct { - Content string `json:"content"` - Filename string `json:"filename"` - Type string `json:"type,omitempty"` - Disposition string `json:"disposition,omitempty"` -} - -// SendGridRequest represents the full SendGrid API request -type SendGridRequest struct { - Personalizations []SendGridPersonalization `json:"personalizations"` - From SendGridEmail `json:"from"` - ReplyTo *SendGridEmail `json:"reply_to,omitempty"` - Subject string `json:"subject"` - Content []SendGridContent `json:"content"` - Attachments []SendGridAttachment `json:"attachments,omitempty"` -} - -func executeSendGridEmail(args map[string]interface{}) (string, error) { - // Get all credential data first (we need both api_key and from_email) - credData, credErr := GetCredentialData(args, "sendgrid") - - // Resolve API key from credential or direct parameter - apiKey, err := ResolveAPIKey(args, "sendgrid", "api_key") - if err != nil { - return "", fmt.Errorf("failed to get SendGrid API key: %w. Please configure SendGrid credentials first.", err) - } - - // Validate API key format - if !strings.HasPrefix(apiKey, "SG.") { - return "", fmt.Errorf("invalid SendGrid API key format (should start with 'SG.')") - } - - // Extract required parameters - toStr, ok := args["to"].(string) - if !ok || toStr == "" { - return "", fmt.Errorf("'to' email address is required") - } - - // Get from_email - first check args, then fall back to credential data - fromEmail, _ := args["from_email"].(string) - if fromEmail == "" && credErr == nil && credData != nil { - // Try to get from credential data - if credFromEmail, ok := credData["from_email"].(string); ok { - fromEmail = credFromEmail - } - } - if fromEmail == "" { - return "", fmt.Errorf("'from_email' is required - either provide it in the request or configure a default in SendGrid credentials") - } - - subject, ok := args["subject"].(string) - if !ok || subject == "" { - return "", fmt.Errorf("'subject' is required") - } - - // Extract content (at least one is required) - textContent, _ := args["text_content"].(string) - htmlContent, _ := args["html_content"].(string) - - if textContent == "" && htmlContent == "" { - return "", fmt.Errorf("either 'text_content' or 'html_content' is required") - } - - // Parse recipient email addresses - toEmails := parseEmailList(toStr) - if len(toEmails) == 0 { - return "", fmt.Errorf("at least one valid 'to' email address is required") - } - - // Check rate limits for all recipients BEFORE sending - var rateLimitedEmails []string - var allowedEmails []SendGridEmail - for _, email := range toEmails { - allowed, waitTime := checkRateLimit(email.Email) - if !allowed { - rateLimitedEmails = append(rateLimitedEmails, fmt.Sprintf("%s (wait %ds)", email.Email, int(waitTime.Seconds()))) - } else { - allowedEmails = append(allowedEmails, email) - } - } - - // If ALL recipients are rate limited, return early with a clear message - if len(allowedEmails) == 0 { - result := map[string]interface{}{ - "success": false, - "already_sent": true, - "rate_limited": true, - "message": "Email already sent to all recipients within the last minute. Please wait before sending again.", - "blocked_emails": rateLimitedEmails, - "cooldown": "1 minute per recipient", - } - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil - } - - // Build personalization with only allowed emails - personalization := SendGridPersonalization{ - To: allowedEmails, - } - - // Parse CC recipients (also rate limited) - if ccStr, ok := args["cc"].(string); ok && ccStr != "" { - ccEmails := parseEmailList(ccStr) - for _, email := range ccEmails { - allowed, _ := checkRateLimit(email.Email) - if allowed { - personalization.CC = append(personalization.CC, email) - } - } - } - - // Parse BCC recipients (also rate limited) - if bccStr, ok := args["bcc"].(string); ok && bccStr != "" { - bccEmails := parseEmailList(bccStr) - for _, email := range bccEmails { - allowed, _ := checkRateLimit(email.Email) - if allowed { - personalization.BCC = append(personalization.BCC, email) - } - } - } - - // Build from address - from := SendGridEmail{Email: fromEmail} - if fromName, ok := args["from_name"].(string); ok && fromName != "" { - from.Name = fromName - } - - // Build content array - var sgContent []SendGridContent - if textContent != "" { - sgContent = append(sgContent, SendGridContent{ - Type: "text/plain", - Value: textContent, - }) - } - if htmlContent != "" { - sgContent = append(sgContent, SendGridContent{ - Type: "text/html", - Value: htmlContent, - }) - } - - // Build request - request := SendGridRequest{ - Personalizations: []SendGridPersonalization{personalization}, - From: from, - Subject: subject, - Content: sgContent, - } - - // Add reply-to if provided - if replyTo, ok := args["reply_to"].(string); ok && replyTo != "" { - request.ReplyTo = &SendGridEmail{Email: replyTo} - } - - // Track if we attached a file via URL - var fileAttached bool - var attachedFileName string - - // Handle file_url attachment (like Discord does) - if fileURL, ok := args["file_url"].(string); ok && fileURL != "" { - fileName, _ := args["file_name"].(string) - fileData, resolvedFileName, mimeType, fetchErr := fetchFileForEmail(fileURL, fileName) - if fetchErr != nil { - return "", fmt.Errorf("failed to fetch file from URL: %w", fetchErr) - } - - // Encode file content as base64 for SendGrid - base64Content := base64.StdEncoding.EncodeToString(fileData) - - attachment := SendGridAttachment{ - Content: base64Content, - Filename: resolvedFileName, - Type: mimeType, - Disposition: "attachment", - } - request.Attachments = append(request.Attachments, attachment) - fileAttached = true - attachedFileName = resolvedFileName - } - - // Add manual attachments if provided (legacy support) - if attachments, ok := args["attachments"].([]interface{}); ok && len(attachments) > 0 { - for _, att := range attachments { - if attMap, ok := att.(map[string]interface{}); ok { - attachment := SendGridAttachment{ - Disposition: "attachment", - } - if content, ok := attMap["content"].(string); ok { - attachment.Content = content - } - if filename, ok := attMap["filename"].(string); ok { - attachment.Filename = filename - } - if mimeType, ok := attMap["type"].(string); ok { - attachment.Type = mimeType - } - if attachment.Content != "" && attachment.Filename != "" { - request.Attachments = append(request.Attachments, attachment) - } - } - } - } - - // Serialize request - jsonPayload, err := json.Marshal(request) - if err != nil { - return "", fmt.Errorf("failed to serialize request: %w", err) - } - - // Create HTTP client - client := &http.Client{ - Timeout: 60 * time.Second, // Increased timeout for large attachments - } - - // Create request to SendGrid API - req, err := http.NewRequest("POST", "https://api.sendgrid.com/v3/mail/send", bytes.NewBuffer(jsonPayload)) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Authorization", "Bearer "+apiKey) - - // Execute request - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - // Read response body - respBody, err := io.ReadAll(resp.Body) - if err != nil { - return "", fmt.Errorf("failed to read response: %w", err) - } - - // SendGrid returns 202 Accepted on success - success := resp.StatusCode == 202 - - // Record successful sends for rate limiting - if success { - for _, email := range allowedEmails { - recordEmailSent(email.Email) - } - for _, email := range personalization.CC { - recordEmailSent(email.Email) - } - for _, email := range personalization.BCC { - recordEmailSent(email.Email) - } - } - - // Build result - result := map[string]interface{}{ - "success": success, - "status_code": resp.StatusCode, - "email_sent": success, - "recipients": len(allowedEmails), - "subject": subject, - "from": fromEmail, - } - - // Include rate limited info if some were blocked - if len(rateLimitedEmails) > 0 { - result["rate_limited_emails"] = rateLimitedEmails - result["partial_send"] = true - } - - if len(personalization.CC) > 0 { - result["cc_count"] = len(personalization.CC) - } - if len(personalization.BCC) > 0 { - result["bcc_count"] = len(personalization.BCC) - } - if len(request.Attachments) > 0 { - result["attachments_count"] = len(request.Attachments) - } - if fileAttached { - result["file_attached"] = true - result["attached_file"] = attachedFileName - } - - // Include response for debugging - if !success { - // Parse error response - var errorResp map[string]interface{} - if err := json.Unmarshal(respBody, &errorResp); err == nil { - result["error"] = errorResp - } else { - result["error"] = string(respBody) - } - result["status"] = resp.Status - } else { - if fileAttached { - result["message"] = fmt.Sprintf("Email with attachment '%s' sent successfully to %d recipient(s)", attachedFileName, len(allowedEmails)) - } else { - result["message"] = fmt.Sprintf("Email sent successfully to %d recipient(s)", len(allowedEmails)) - } - } - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} - -// fetchFileForEmail fetches a file from a URL for email attachment -// Returns: file content, filename, mime type, error -func fetchFileForEmail(fileURL, providedFileName string) ([]byte, string, string, error) { - // Resolve relative URLs to absolute URLs - actualURL := fileURL - if strings.HasPrefix(fileURL, "/api/") { - // Use BACKEND_URL env var for internal API calls - backendURL := os.Getenv("BACKEND_URL") - if backendURL == "" { - backendURL = "http://localhost:3001" - } - actualURL = backendURL + fileURL - } - - // Create HTTP client - client := &http.Client{ - Timeout: 30 * time.Second, - } - - resp, err := client.Get(actualURL) - if err != nil { - return nil, "", "", fmt.Errorf("failed to fetch file: %w", err) - } - defer resp.Body.Close() - - if resp.StatusCode != http.StatusOK { - body, _ := io.ReadAll(resp.Body) - return nil, "", "", fmt.Errorf("failed to fetch file: status %d - %s", resp.StatusCode, string(body)) - } - - // Read the file content - data, err := io.ReadAll(resp.Body) - if err != nil { - return nil, "", "", fmt.Errorf("failed to read file: %w", err) - } - - // Determine filename - fileName := providedFileName - if fileName == "" { - // Try to get from Content-Disposition header - if cd := resp.Header.Get("Content-Disposition"); cd != "" { - _, params, err := mime.ParseMediaType(cd) - if err == nil { - if fn, ok := params["filename"]; ok { - fileName = fn - } - } - } - // Fallback: extract from URL - if fileName == "" { - urlPath := strings.Split(strings.Split(fileURL, "?")[0], "/") - if len(urlPath) > 0 { - lastPart := urlPath[len(urlPath)-1] - if strings.Contains(lastPart, ".") { - fileName = lastPart - } - } - } - // Final fallback - if fileName == "" { - fileName = "attachment" - } - } - - // Determine MIME type - mimeType := resp.Header.Get("Content-Type") - if mimeType == "" || mimeType == "application/octet-stream" { - // Try to infer from filename extension - ext := strings.ToLower(filepath.Ext(fileName)) - switch ext { - case ".pdf": - mimeType = "application/pdf" - case ".png": - mimeType = "image/png" - case ".jpg", ".jpeg": - mimeType = "image/jpeg" - case ".gif": - mimeType = "image/gif" - case ".doc": - mimeType = "application/msword" - case ".docx": - mimeType = "application/vnd.openxmlformats-officedocument.wordprocessingml.document" - case ".xls": - mimeType = "application/vnd.ms-excel" - case ".xlsx": - mimeType = "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet" - case ".csv": - mimeType = "text/csv" - case ".txt": - mimeType = "text/plain" - case ".html": - mimeType = "text/html" - case ".json": - mimeType = "application/json" - case ".zip": - mimeType = "application/zip" - default: - mimeType = "application/octet-stream" - } - } - - return data, fileName, mimeType, nil -} - -// parseEmailList parses a comma-separated list of email addresses -func parseEmailList(emailStr string) []SendGridEmail { - var emails []SendGridEmail - parts := strings.Split(emailStr, ",") - for _, part := range parts { - email := strings.TrimSpace(part) - if email != "" && strings.Contains(email, "@") { - emails = append(emails, SendGridEmail{Email: email}) - } - } - return emails -} diff --git a/backend/internal/tools/shopify_tool.go b/backend/internal/tools/shopify_tool.go deleted file mode 100644 index bc679680..00000000 --- a/backend/internal/tools/shopify_tool.go +++ /dev/null @@ -1,306 +0,0 @@ -package tools - -import ( - "encoding/json" - "fmt" - "io" - "net/http" - "net/url" - "strings" - "time" -) - -// NewShopifyProductsTool creates a Shopify products listing tool -func NewShopifyProductsTool() *Tool { - return &Tool{ - Name: "shopify_products", - DisplayName: "Shopify Products", - Description: `List products from a Shopify store. - -Returns product details including title, description, variants, and inventory. -Authentication is handled automatically via configured Shopify credentials.`, - Icon: "ShoppingCart", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"shopify", "products", "ecommerce", "inventory", "store"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system.", - }, - "limit": map[string]interface{}{ - "type": "number", - "description": "Number of products to return (max 250, default 50)", - }, - "product_type": map[string]interface{}{ - "type": "string", - "description": "Filter by product type", - }, - "vendor": map[string]interface{}{ - "type": "string", - "description": "Filter by vendor", - }, - "status": map[string]interface{}{ - "type": "string", - "description": "Filter by status: active, archived, or draft", - }, - }, - "required": []string{}, - }, - Execute: executeShopifyProducts, - } -} - -// NewShopifyOrdersTool creates a Shopify orders listing tool -func NewShopifyOrdersTool() *Tool { - return &Tool{ - Name: "shopify_orders", - DisplayName: "Shopify Orders", - Description: `List orders from a Shopify store. - -Returns order details including line items, customer info, and fulfillment status. -Authentication is handled automatically via configured Shopify credentials.`, - Icon: "Package", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"shopify", "orders", "ecommerce", "sales", "fulfillment"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system.", - }, - "limit": map[string]interface{}{ - "type": "number", - "description": "Number of orders to return (max 250, default 50)", - }, - "status": map[string]interface{}{ - "type": "string", - "description": "Filter by status: open, closed, cancelled, any", - }, - "financial_status": map[string]interface{}{ - "type": "string", - "description": "Filter: pending, authorized, partially_paid, paid, refunded, voided", - }, - "fulfillment_status": map[string]interface{}{ - "type": "string", - "description": "Filter: shipped, partial, unshipped, any", - }, - }, - "required": []string{}, - }, - Execute: executeShopifyOrders, - } -} - -// NewShopifyCustomersTool creates a Shopify customers listing tool -func NewShopifyCustomersTool() *Tool { - return &Tool{ - Name: "shopify_customers", - DisplayName: "Shopify Customers", - Description: `List customers from a Shopify store. - -Returns customer details including name, email, orders count, and total spent. -Authentication is handled automatically via configured Shopify credentials.`, - Icon: "Users", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"shopify", "customers", "ecommerce", "crm"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system.", - }, - "limit": map[string]interface{}{ - "type": "number", - "description": "Number of customers to return (max 250, default 50)", - }, - }, - "required": []string{}, - }, - Execute: executeShopifyCustomers, - } -} - -func buildShopifyRequest(storeURL, accessToken, endpoint string, queryParams url.Values) (*http.Request, error) { - storeURL = strings.TrimPrefix(storeURL, "https://") - storeURL = strings.TrimPrefix(storeURL, "http://") - storeURL = strings.TrimSuffix(storeURL, "/") - - apiURL := fmt.Sprintf("https://%s/admin/api/2025-01/%s.json", storeURL, endpoint) - if len(queryParams) > 0 { - apiURL += "?" + queryParams.Encode() - } - - req, err := http.NewRequest("GET", apiURL, nil) - if err != nil { - return nil, err - } - req.Header.Set("X-Shopify-Access-Token", accessToken) - req.Header.Set("Content-Type", "application/json") - return req, nil -} - -func executeShopifyProducts(args map[string]interface{}) (string, error) { - credData, err := GetCredentialData(args, "shopify") - if err != nil { - return "", fmt.Errorf("failed to get Shopify credentials: %w", err) - } - - storeURL, _ := credData["store_url"].(string) - accessToken, _ := credData["access_token"].(string) - if storeURL == "" || accessToken == "" { - return "", fmt.Errorf("Shopify credentials incomplete: store_url and access_token are required") - } - - queryParams := url.Values{} - if limit, ok := args["limit"].(float64); ok && limit > 0 { - queryParams.Set("limit", fmt.Sprintf("%d", int(limit))) - } else { - queryParams.Set("limit", "50") - } - if pt, ok := args["product_type"].(string); ok && pt != "" { - queryParams.Set("product_type", pt) - } - if v, ok := args["vendor"].(string); ok && v != "" { - queryParams.Set("vendor", v) - } - if s, ok := args["status"].(string); ok && s != "" { - queryParams.Set("status", s) - } - - req, err := buildShopifyRequest(storeURL, accessToken, "products", queryParams) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to send request: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - var result map[string]interface{} - json.Unmarshal(body, &result) - - if resp.StatusCode >= 400 { - errMsg := "unknown error" - if e, ok := result["errors"].(string); ok { - errMsg = e - } - return "", fmt.Errorf("Shopify API error: %s", errMsg) - } - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} - -func executeShopifyOrders(args map[string]interface{}) (string, error) { - credData, err := GetCredentialData(args, "shopify") - if err != nil { - return "", fmt.Errorf("failed to get Shopify credentials: %w", err) - } - - storeURL, _ := credData["store_url"].(string) - accessToken, _ := credData["access_token"].(string) - if storeURL == "" || accessToken == "" { - return "", fmt.Errorf("Shopify credentials incomplete: store_url and access_token are required") - } - - queryParams := url.Values{} - if limit, ok := args["limit"].(float64); ok && limit > 0 { - queryParams.Set("limit", fmt.Sprintf("%d", int(limit))) - } else { - queryParams.Set("limit", "50") - } - if s, ok := args["status"].(string); ok && s != "" { - queryParams.Set("status", s) - } - if fs, ok := args["financial_status"].(string); ok && fs != "" { - queryParams.Set("financial_status", fs) - } - if ffs, ok := args["fulfillment_status"].(string); ok && ffs != "" { - queryParams.Set("fulfillment_status", ffs) - } - - req, err := buildShopifyRequest(storeURL, accessToken, "orders", queryParams) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to send request: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - var result map[string]interface{} - json.Unmarshal(body, &result) - - if resp.StatusCode >= 400 { - errMsg := "unknown error" - if e, ok := result["errors"].(string); ok { - errMsg = e - } - return "", fmt.Errorf("Shopify API error: %s", errMsg) - } - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} - -func executeShopifyCustomers(args map[string]interface{}) (string, error) { - credData, err := GetCredentialData(args, "shopify") - if err != nil { - return "", fmt.Errorf("failed to get Shopify credentials: %w", err) - } - - storeURL, _ := credData["store_url"].(string) - accessToken, _ := credData["access_token"].(string) - if storeURL == "" || accessToken == "" { - return "", fmt.Errorf("Shopify credentials incomplete: store_url and access_token are required") - } - - queryParams := url.Values{} - if limit, ok := args["limit"].(float64); ok && limit > 0 { - queryParams.Set("limit", fmt.Sprintf("%d", int(limit))) - } else { - queryParams.Set("limit", "50") - } - - req, err := buildShopifyRequest(storeURL, accessToken, "customers", queryParams) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to send request: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - var result map[string]interface{} - json.Unmarshal(body, &result) - - if resp.StatusCode >= 400 { - errMsg := "unknown error" - if e, ok := result["errors"].(string); ok { - errMsg = e - } - return "", fmt.Errorf("Shopify API error: %s", errMsg) - } - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} diff --git a/backend/internal/tools/slack_tool.go b/backend/internal/tools/slack_tool.go deleted file mode 100644 index ada3cfa0..00000000 --- a/backend/internal/tools/slack_tool.go +++ /dev/null @@ -1,184 +0,0 @@ -package tools - -import ( - "bytes" - "encoding/json" - "fmt" - "io" - "net/http" - "strings" - "time" -) - -// NewSlackTool creates a Slack webhook messaging tool -func NewSlackTool() *Tool { - return &Tool{ - Name: "send_slack_message", - DisplayName: "Send Slack Message", - Description: "Send a message to Slack via incoming webhook. Just provide the message text - webhook authentication is handled automatically via configured credentials. Do NOT ask the user for webhook URLs. Supports Block Kit for rich formatting.", - Icon: "Hash", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"slack", "message", "chat", "notify", "webhook", "channel", "workspace", "notification"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "webhook_url": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Resolved from credentials. Do not ask user for this.", - }, - "text": map[string]interface{}{ - "type": "string", - "description": "Message text. This is the main content that will be posted to the channel.", - }, - "username": map[string]interface{}{ - "type": "string", - "description": "Override the default webhook username (optional)", - }, - "icon_emoji": map[string]interface{}{ - "type": "string", - "description": "Override the default webhook icon with an emoji (e.g., ':robot_face:')", - }, - "icon_url": map[string]interface{}{ - "type": "string", - "description": "Override the default webhook icon with an image URL (optional)", - }, - "channel": map[string]interface{}{ - "type": "string", - "description": "Override the default channel (e.g., '#general' or '@username'). Requires additional webhook permissions.", - }, - "unfurl_links": map[string]interface{}{ - "type": "boolean", - "description": "Enable/disable link unfurling (default: true)", - }, - "unfurl_media": map[string]interface{}{ - "type": "boolean", - "description": "Enable/disable media unfurling (default: true)", - }, - }, - "required": []string{"text"}, - }, - Execute: executeSlackMessage, - } -} - -func executeSlackMessage(args map[string]interface{}) (string, error) { - // Resolve webhook URL from credential or direct parameter - webhookURL, err := ResolveWebhookURL(args, "slack") - if err != nil { - // Fallback: check for direct webhook_url if credential resolution failed - if url, ok := args["webhook_url"].(string); ok && url != "" { - webhookURL = url - } else { - return "", fmt.Errorf("failed to get webhook URL: %w", err) - } - } - - // Validate Slack webhook URL - if !strings.Contains(webhookURL, "hooks.slack.com") { - return "", fmt.Errorf("invalid Slack webhook URL (must contain hooks.slack.com)") - } - - // Extract text (required) - text, ok := args["text"].(string) - if !ok || text == "" { - return "", fmt.Errorf("text is required") - } - - // Slack has a soft limit of ~40,000 characters but recommend keeping under 4000 - if len(text) > 40000 { - text = text[:39997] + "..." - } - - // Build Slack webhook payload - payload := map[string]interface{}{ - "text": text, - } - - // Optional username override - if username, ok := args["username"].(string); ok && username != "" { - payload["username"] = username - } - - // Optional icon emoji - if iconEmoji, ok := args["icon_emoji"].(string); ok && iconEmoji != "" { - payload["icon_emoji"] = iconEmoji - } - - // Optional icon URL - if iconURL, ok := args["icon_url"].(string); ok && iconURL != "" { - payload["icon_url"] = iconURL - } - - // Optional channel override - if channel, ok := args["channel"].(string); ok && channel != "" { - payload["channel"] = channel - } - - // Optional unfurl settings - if unfurlLinks, ok := args["unfurl_links"].(bool); ok { - payload["unfurl_links"] = unfurlLinks - } - - if unfurlMedia, ok := args["unfurl_media"].(bool); ok { - payload["unfurl_media"] = unfurlMedia - } - - // Serialize payload - jsonPayload, err := json.Marshal(payload) - if err != nil { - return "", fmt.Errorf("failed to serialize payload: %w", err) - } - - // Create HTTP client - client := &http.Client{ - Timeout: 30 * time.Second, - } - - // Create request - req, err := http.NewRequest("POST", webhookURL, bytes.NewBuffer(jsonPayload)) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Content-Type", "application/json") - - // Execute request - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - // Read response body - respBody, err := io.ReadAll(resp.Body) - if err != nil { - return "", fmt.Errorf("failed to read response: %w", err) - } - - // Slack returns "ok" on success - success := resp.StatusCode == 200 && string(respBody) == "ok" - - // Build result - result := map[string]interface{}{ - "success": success, - "status_code": resp.StatusCode, - "message_sent": success, - "text_length": len(text), - } - - // Include response body for debugging - if !success { - result["error"] = string(respBody) - result["status"] = resp.Status - } else { - result["message"] = "Slack message sent successfully" - } - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} diff --git a/backend/internal/tools/spreadsheet_tool.go b/backend/internal/tools/spreadsheet_tool.go deleted file mode 100644 index e7605dcc..00000000 --- a/backend/internal/tools/spreadsheet_tool.go +++ /dev/null @@ -1,521 +0,0 @@ -package tools - -import ( - "claraverse/internal/filecache" - "encoding/csv" - "encoding/json" - "fmt" - "log" - "os" - "path/filepath" - "strings" - - "github.com/xuri/excelize/v2" -) - -// NewReadSpreadsheetTool creates a tool for reading Excel/CSV/spreadsheet files -func NewReadSpreadsheetTool() *Tool { - return &Tool{ - Name: "read_spreadsheet", - DisplayName: "Read Spreadsheet", - Description: `Reads and parses spreadsheet files (Excel, CSV, TSV). Returns structured data with headers and rows. - -SUPPORTED FORMATS: -- .xlsx (Excel 2007+, OpenXML) ✓ -- .xls (Legacy Excel) - converted to text -- .csv (Comma-separated values) ✓ -- .tsv (Tab-separated values) ✓ - -USE THIS TOOL when user uploads: -- Excel files (.xlsx, .xls) -- CSV/TSV data files -- Any spreadsheet for analysis - -Returns: headers, rows (as arrays), row count, and sheet names for Excel files.`, - Icon: "FileSpreadsheet", - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "file_id": map[string]interface{}{ - "type": "string", - "description": "The file ID of the uploaded spreadsheet (from upload response)", - }, - "sheet_name": map[string]interface{}{ - "type": "string", - "description": "For Excel files: specific sheet name to read. If not provided, reads the first/active sheet.", - }, - "max_rows": map[string]interface{}{ - "type": "integer", - "description": "Maximum number of data rows to return (default: 200, max: 1000). Headers are always included.", - }, - "include_empty": map[string]interface{}{ - "type": "boolean", - "description": "Include empty rows in output (default: false)", - }, - }, - "required": []string{"file_id"}, - }, - Execute: executeReadSpreadsheet, - Source: ToolSourceBuiltin, - Category: "data_sources", - Keywords: []string{"excel", "xlsx", "xls", "csv", "tsv", "spreadsheet", "sheet", "workbook", "data", "table", "read", "parse"}, - } -} - -func executeReadSpreadsheet(args map[string]interface{}) (string, error) { - // Extract parameters - fileID, ok := args["file_id"].(string) - if !ok || fileID == "" { - return "", fmt.Errorf("file_id parameter is required") - } - - sheetName := "" - if sn, ok := args["sheet_name"].(string); ok { - sheetName = sn - } - - maxRows := 200 - if mr, ok := args["max_rows"].(float64); ok { - maxRows = int(mr) - if maxRows > 1000 { - maxRows = 1000 - } - if maxRows < 1 { - maxRows = 1 - } - } - - includeEmpty := false - if ie, ok := args["include_empty"].(bool); ok { - includeEmpty = ie - } - - // Extract user context - userID, _ := args["__user_id__"].(string) - conversationID, _ := args["__conversation_id__"].(string) - delete(args, "__user_id__") - delete(args, "__conversation_id__") - - log.Printf("📊 [READ-SPREADSHEET] Reading file_id=%s sheet=%s maxRows=%d (user=%s)", fileID, sheetName, maxRows, userID) - - // Get file from cache - fileCacheService := filecache.GetService() - var file *filecache.CachedFile - var err error - - if userID != "" && conversationID != "" { - file, err = fileCacheService.GetByUserAndConversation(fileID, userID, conversationID) - if err != nil { - file, _ = fileCacheService.Get(fileID) - if file != nil && file.UserID != userID { - return "", fmt.Errorf("access denied: you don't have permission to read this file") - } - } - } else { - file, _ = fileCacheService.Get(fileID) - } - - if file == nil { - return "", fmt.Errorf("file not found or expired. Files are available for 30 minutes after upload") - } - - // Determine file type - filename := strings.ToLower(file.Filename) - mimeType := strings.ToLower(file.MimeType) - - var result *SpreadsheetResult - - switch { - case strings.HasSuffix(filename, ".xlsx") || - mimeType == "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet": - result, err = readXLSX(file, sheetName, maxRows, includeEmpty) - - case strings.HasSuffix(filename, ".xls") || - mimeType == "application/vnd.ms-excel": - result, err = readXLS(file, maxRows) - - case strings.HasSuffix(filename, ".csv") || - mimeType == "text/csv": - result, err = readCSV(file, maxRows, ',', includeEmpty) - - case strings.HasSuffix(filename, ".tsv") || - mimeType == "text/tab-separated-values": - result, err = readCSV(file, maxRows, '\t', includeEmpty) - - default: - return "", fmt.Errorf("unsupported file type: %s. Supported: .xlsx, .xls, .csv, .tsv", file.MimeType) - } - - if err != nil { - log.Printf("❌ [READ-SPREADSHEET] Failed to read %s: %v", file.Filename, err) - return "", err - } - - // Build response - response := map[string]interface{}{ - "success": true, - "file_id": file.FileID, - "filename": file.Filename, - "file_type": result.FileType, - "headers": result.Headers, - "column_count": len(result.Headers), - "rows": result.Rows, - "row_count": len(result.Rows), - "total_rows": result.TotalRows, - "truncated": result.Truncated, - "sheet_name": result.SheetName, - "available_sheets": result.SheetNames, - } - - // Add preview text for LLM context - preview := generatePreview(result) - response["preview"] = preview - - responseJSON, _ := json.MarshalIndent(response, "", " ") - log.Printf("✅ [READ-SPREADSHEET] Read %s: %d columns, %d rows (sheet: %s)", - file.Filename, len(result.Headers), len(result.Rows), result.SheetName) - - return string(responseJSON), nil -} - -// SpreadsheetResult holds parsed spreadsheet data -type SpreadsheetResult struct { - FileType string - Headers []string - Rows [][]string - TotalRows int - Truncated bool - SheetName string - SheetNames []string -} - -// readXLSX reads .xlsx files using excelize -func readXLSX(file *filecache.CachedFile, sheetName string, maxRows int, includeEmpty bool) (*SpreadsheetResult, error) { - if file.FilePath == "" { - return nil, fmt.Errorf("file content not available") - } - - // Open Excel file - f, err := excelize.OpenFile(file.FilePath) - if err != nil { - return nil, fmt.Errorf("failed to open Excel file: %w", err) - } - defer f.Close() - - // Get sheet names - sheetNames := f.GetSheetList() - if len(sheetNames) == 0 { - return nil, fmt.Errorf("no sheets found in workbook") - } - - // Select sheet - targetSheet := sheetName - if targetSheet == "" { - // Use active sheet or first sheet - targetSheet = f.GetSheetName(f.GetActiveSheetIndex()) - if targetSheet == "" { - targetSheet = sheetNames[0] - } - } - - // Verify sheet exists - sheetExists := false - for _, s := range sheetNames { - if s == targetSheet { - sheetExists = true - break - } - } - if !sheetExists { - return nil, fmt.Errorf("sheet '%s' not found. Available sheets: %v", targetSheet, sheetNames) - } - - // Read all rows - rows, err := f.GetRows(targetSheet) - if err != nil { - return nil, fmt.Errorf("failed to read sheet '%s': %w", targetSheet, err) - } - - if len(rows) == 0 { - return &SpreadsheetResult{ - FileType: "xlsx", - Headers: []string{}, - Rows: [][]string{}, - TotalRows: 0, - SheetName: targetSheet, - SheetNames: sheetNames, - }, nil - } - - // First row is headers - headers := rows[0] - - // Normalize headers (trim whitespace, handle empty) - for i, h := range headers { - headers[i] = strings.TrimSpace(h) - if headers[i] == "" { - headers[i] = fmt.Sprintf("Column_%d", i+1) - } - } - - // Process data rows - dataRows := make([][]string, 0, len(rows)-1) - totalRows := 0 - - for i := 1; i < len(rows); i++ { - row := rows[i] - - // Skip empty rows unless requested - if !includeEmpty && isEmptyRow(row) { - continue - } - - totalRows++ - - // Pad row to match header count - for len(row) < len(headers) { - row = append(row, "") - } - // Trim row if longer than headers - if len(row) > len(headers) { - row = row[:len(headers)] - } - - // Trim cell values - for j := range row { - row[j] = strings.TrimSpace(row[j]) - } - - if len(dataRows) < maxRows { - dataRows = append(dataRows, row) - } - } - - return &SpreadsheetResult{ - FileType: "xlsx", - Headers: headers, - Rows: dataRows, - TotalRows: totalRows, - Truncated: totalRows > maxRows, - SheetName: targetSheet, - SheetNames: sheetNames, - }, nil -} - -// readXLS handles legacy .xls files -func readXLS(file *filecache.CachedFile, maxRows int) (*SpreadsheetResult, error) { - // For .xls files, try to read as text or return helpful error - // Note: Full .xls support would require additional library like github.com/extrame/xls - - if file.FilePath == "" { - return nil, fmt.Errorf("file content not available") - } - - // Read raw content - content, err := os.ReadFile(file.FilePath) - if err != nil { - return nil, fmt.Errorf("failed to read file: %w", err) - } - - // Try to extract text content (basic approach) - // .xls is a binary format, but we can try to extract readable strings - text := extractTextFromBinary(content) - - if text == "" { - return nil, fmt.Errorf("legacy .xls format detected. Please convert to .xlsx or .csv for full support. You can do this in Excel: File > Save As > Excel Workbook (.xlsx)") - } - - // Return extracted text as single column - lines := strings.Split(text, "\n") - headers := []string{"Extracted_Content"} - rows := make([][]string, 0) - - for i, line := range lines { - if i >= maxRows { - break - } - line = strings.TrimSpace(line) - if line != "" { - rows = append(rows, []string{line}) - } - } - - return &SpreadsheetResult{ - FileType: "xls", - Headers: headers, - Rows: rows, - TotalRows: len(rows), - Truncated: len(lines) > maxRows, - SheetName: "Sheet1", - SheetNames: []string{"Sheet1"}, - }, nil -} - -// readCSV reads CSV/TSV files -func readCSV(file *filecache.CachedFile, maxRows int, delimiter rune, includeEmpty bool) (*SpreadsheetResult, error) { - var content string - - if file.FilePath != "" { - data, err := os.ReadFile(file.FilePath) - if err != nil { - return nil, fmt.Errorf("failed to read file: %w", err) - } - content = string(data) - } else if file.ExtractedText != nil { - content = file.ExtractedText.String() - } else { - return nil, fmt.Errorf("file content not available") - } - - // Parse CSV - reader := csv.NewReader(strings.NewReader(content)) - reader.Comma = delimiter - reader.LazyQuotes = true - reader.TrimLeadingSpace = true - - records, err := reader.ReadAll() - if err != nil { - return nil, fmt.Errorf("failed to parse CSV: %w", err) - } - - if len(records) == 0 { - return &SpreadsheetResult{ - FileType: "csv", - Headers: []string{}, - Rows: [][]string{}, - TotalRows: 0, - SheetName: filepath.Base(file.Filename), - SheetNames: []string{filepath.Base(file.Filename)}, - }, nil - } - - // First row is headers - headers := records[0] - for i, h := range headers { - headers[i] = strings.TrimSpace(h) - if headers[i] == "" { - headers[i] = fmt.Sprintf("Column_%d", i+1) - } - } - - // Process data rows - dataRows := make([][]string, 0, len(records)-1) - totalRows := 0 - - for i := 1; i < len(records); i++ { - row := records[i] - - if !includeEmpty && isEmptyRow(row) { - continue - } - - totalRows++ - - // Normalize row length - for len(row) < len(headers) { - row = append(row, "") - } - if len(row) > len(headers) { - row = row[:len(headers)] - } - - for j := range row { - row[j] = strings.TrimSpace(row[j]) - } - - if len(dataRows) < maxRows { - dataRows = append(dataRows, row) - } - } - - fileType := "csv" - if delimiter == '\t' { - fileType = "tsv" - } - - return &SpreadsheetResult{ - FileType: fileType, - Headers: headers, - Rows: dataRows, - TotalRows: totalRows, - Truncated: totalRows > maxRows, - SheetName: filepath.Base(file.Filename), - SheetNames: []string{filepath.Base(file.Filename)}, - }, nil -} - -// isEmptyRow checks if a row has no content -func isEmptyRow(row []string) bool { - for _, cell := range row { - if strings.TrimSpace(cell) != "" { - return false - } - } - return true -} - -// extractTextFromBinary tries to extract readable text from binary files -func extractTextFromBinary(data []byte) string { - var builder strings.Builder - var current strings.Builder - - for _, b := range data { - // Keep printable ASCII and common whitespace - if (b >= 32 && b <= 126) || b == '\n' || b == '\r' || b == '\t' { - current.WriteByte(b) - } else { - // If we have accumulated text, write it - text := strings.TrimSpace(current.String()) - if len(text) > 3 { // Only keep strings longer than 3 chars - if builder.Len() > 0 { - builder.WriteString("\n") - } - builder.WriteString(text) - } - current.Reset() - } - } - - // Don't forget the last segment - text := strings.TrimSpace(current.String()) - if len(text) > 3 { - if builder.Len() > 0 { - builder.WriteString("\n") - } - builder.WriteString(text) - } - - return builder.String() -} - -// generatePreview creates a text preview of the data for LLM context -func generatePreview(result *SpreadsheetResult) string { - var builder strings.Builder - - builder.WriteString(fmt.Sprintf("=== %s Data Preview ===\n", strings.ToUpper(result.FileType))) - builder.WriteString(fmt.Sprintf("Sheet: %s | Columns: %d | Rows: %d", result.SheetName, len(result.Headers), result.TotalRows)) - if result.Truncated { - builder.WriteString(fmt.Sprintf(" (showing first %d)", len(result.Rows))) - } - builder.WriteString("\n\n") - - // Headers - builder.WriteString("Headers: ") - builder.WriteString(strings.Join(result.Headers, " | ")) - builder.WriteString("\n\n") - - // Sample rows (first 5) - sampleCount := 5 - if len(result.Rows) < sampleCount { - sampleCount = len(result.Rows) - } - - if sampleCount > 0 { - builder.WriteString("Sample data:\n") - for i := 0; i < sampleCount; i++ { - builder.WriteString(fmt.Sprintf(" Row %d: %s\n", i+1, strings.Join(result.Rows[i], " | "))) - } - } - - return builder.String() -} diff --git a/backend/internal/tools/teams_tool.go b/backend/internal/tools/teams_tool.go deleted file mode 100644 index 1701d9bf..00000000 --- a/backend/internal/tools/teams_tool.go +++ /dev/null @@ -1,143 +0,0 @@ -package tools - -import ( - "bytes" - "encoding/json" - "fmt" - "io" - "net/http" - "strings" - "time" -) - -// NewTeamsTool creates a Microsoft Teams webhook messaging tool -func NewTeamsTool() *Tool { - return &Tool{ - Name: "send_teams_message", - DisplayName: "Send Teams Message", - Description: "Send a message to Microsoft Teams via incoming webhook. Just provide the message text - webhook authentication is handled automatically via configured credentials. Do NOT ask the user for webhook URLs. Supports Adaptive Cards for rich formatting.", - Icon: "MessageSquare", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"teams", "microsoft", "message", "chat", "notify", "webhook", "channel", "office365", "notification"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "webhook_url": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Resolved from credentials. Do not ask user for this.", - }, - "text": map[string]interface{}{ - "type": "string", - "description": "Message text. This is the main content that will be posted to the channel.", - }, - "title": map[string]interface{}{ - "type": "string", - "description": "Optional title for the message card", - }, - "theme_color": map[string]interface{}{ - "type": "string", - "description": "Theme color for the message card (hex without #, e.g., '0076D7')", - }, - }, - "required": []string{"text"}, - }, - Execute: executeTeamsMessage, - } -} - -func executeTeamsMessage(args map[string]interface{}) (string, error) { - // Resolve webhook URL from credential or direct parameter - webhookURL, err := ResolveWebhookURL(args, "teams") - if err != nil { - if url, ok := args["webhook_url"].(string); ok && url != "" { - webhookURL = url - } else { - return "", fmt.Errorf("failed to get webhook URL: %w", err) - } - } - - // Validate Teams webhook URL - if !strings.Contains(webhookURL, "webhook.office.com") && !strings.Contains(webhookURL, "outlook.office.com") { - return "", fmt.Errorf("invalid Teams webhook URL") - } - - // Extract text (required) - text, ok := args["text"].(string) - if !ok || text == "" { - return "", fmt.Errorf("text is required") - } - - // Build Teams MessageCard payload - payload := map[string]interface{}{ - "@type": "MessageCard", - "@context": "http://schema.org/extensions", - "text": text, - } - - // Optional title - if title, ok := args["title"].(string); ok && title != "" { - payload["title"] = title - } - - // Optional theme color - if themeColor, ok := args["theme_color"].(string); ok && themeColor != "" { - payload["themeColor"] = themeColor - } - - // Serialize payload - jsonPayload, err := json.Marshal(payload) - if err != nil { - return "", fmt.Errorf("failed to serialize payload: %w", err) - } - - // Create HTTP client - client := &http.Client{Timeout: 30 * time.Second} - - // Create request - req, err := http.NewRequest("POST", webhookURL, bytes.NewBuffer(jsonPayload)) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Content-Type", "application/json") - - // Execute request - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - // Read response body - respBody, err := io.ReadAll(resp.Body) - if err != nil { - return "", fmt.Errorf("failed to read response: %w", err) - } - - // Teams returns "1" on success - success := resp.StatusCode == 200 - - // Build result - result := map[string]interface{}{ - "success": success, - "status_code": resp.StatusCode, - "message_sent": success, - "text_length": len(text), - } - - if !success { - result["error"] = string(respBody) - result["status"] = resp.Status - } else { - result["message"] = "Teams message sent successfully" - } - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} - diff --git a/backend/internal/tools/telegram_tool.go b/backend/internal/tools/telegram_tool.go deleted file mode 100644 index 83408636..00000000 --- a/backend/internal/tools/telegram_tool.go +++ /dev/null @@ -1,196 +0,0 @@ -package tools - -import ( - "bytes" - "encoding/json" - "fmt" - "io" - "net/http" - "time" -) - -// NewTelegramTool creates a Telegram Bot API messaging tool -func NewTelegramTool() *Tool { - return &Tool{ - Name: "send_telegram_message", - DisplayName: "Send Telegram Message", - Description: "Send a message to Telegram via Bot API. Just provide the message content and chat ID - bot authentication is handled automatically via configured credentials. Do NOT ask the user for bot tokens. Supports markdown and HTML formatting.", - Icon: "Send", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"telegram", "message", "chat", "notify", "bot", "notification", "messenger"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "bot_token": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Resolved from credentials. Do not ask user for this.", - }, - "chat_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Resolved from credentials. The chat ID to send messages to. Can be overridden for dynamic use cases.", - }, - "text": map[string]interface{}{ - "type": "string", - "description": "Message text (max 4096 characters). This is the main message content.", - }, - "parse_mode": map[string]interface{}{ - "type": "string", - "description": "Text formatting mode: 'MarkdownV2', 'HTML', or 'Markdown' (legacy). Optional.", - "enum": []string{"MarkdownV2", "HTML", "Markdown"}, - }, - "disable_notification": map[string]interface{}{ - "type": "boolean", - "description": "Send message silently without notification sound. Optional, defaults to false.", - }, - "disable_link_preview": map[string]interface{}{ - "type": "boolean", - "description": "Disable link previews for URLs in the message. Optional, defaults to false.", - }, - }, - "required": []string{"text"}, - }, - Execute: executeTelegramMessage, - } -} - -func executeTelegramMessage(args map[string]interface{}) (string, error) { - // Get all credential data (both bot_token and chat_id) - credData, err := GetCredentialData(args, "telegram") - if err != nil { - return "", fmt.Errorf("failed to get Telegram credentials: %w. Please configure Telegram credentials first.", err) - } - - // Extract bot_token from credential - botToken, ok := credData["bot_token"].(string) - if !ok || botToken == "" { - return "", fmt.Errorf("Telegram credentials incomplete: bot_token is required") - } - - // Extract chat_id - first check args (for override), then fall back to credential - chatID, _ := args["chat_id"].(string) - if chatID == "" { - // Fall back to chat_id from credential - if credChatID, ok := credData["chat_id"].(string); ok { - chatID = credChatID - } - } - - if chatID == "" { - return "", fmt.Errorf("chat_id is required - either provide it in the request or configure a default in Telegram credentials") - } - - // Extract text (required) - text, ok := args["text"].(string) - if !ok || text == "" { - return "", fmt.Errorf("text is required") - } - - // Truncate text if too long (Telegram limit is 4096) - if len(text) > 4096 { - text = text[:4093] + "..." - } - - // Build Telegram API payload - payload := map[string]interface{}{ - "chat_id": chatID, - "text": text, - } - - // Optional parse_mode - if parseMode, ok := args["parse_mode"].(string); ok && parseMode != "" { - payload["parse_mode"] = parseMode - } - - // Optional disable_notification - if disableNotification, ok := args["disable_notification"].(bool); ok && disableNotification { - payload["disable_notification"] = true - } - - // Optional link_preview_options (replaces deprecated disable_web_page_preview) - if disablePreview, ok := args["disable_link_preview"].(bool); ok && disablePreview { - payload["link_preview_options"] = map[string]interface{}{ - "is_disabled": true, - } - } - - // Build API URL - apiURL := fmt.Sprintf("https://api.telegram.org/bot%s/sendMessage", botToken) - - // Serialize payload - jsonPayload, err := json.Marshal(payload) - if err != nil { - return "", fmt.Errorf("failed to serialize payload: %w", err) - } - - // Create HTTP client - client := &http.Client{ - Timeout: 30 * time.Second, - } - - // Create request - req, err := http.NewRequest("POST", apiURL, bytes.NewBuffer(jsonPayload)) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - req.Header.Set("Content-Type", "application/json") - - // Execute request - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - // Read response body - respBody, err := io.ReadAll(resp.Body) - if err != nil { - return "", fmt.Errorf("failed to read response: %w", err) - } - - // Parse Telegram response - var telegramResp map[string]interface{} - if err := json.Unmarshal(respBody, &telegramResp); err != nil { - return "", fmt.Errorf("failed to parse response: %w", err) - } - - // Check if request was successful - success := false - if ok, exists := telegramResp["ok"].(bool); exists { - success = ok - } - - // Build result - result := map[string]interface{}{ - "success": success, - "status_code": resp.StatusCode, - "message_sent": success, - "chat_id": chatID, - "text_length": len(text), - } - - // Include message_id if successful - if success { - if msgResult, ok := telegramResp["result"].(map[string]interface{}); ok { - if msgID, ok := msgResult["message_id"].(float64); ok { - result["message_id"] = int(msgID) - } - } - result["message"] = "Telegram message sent successfully" - } else { - // Include error details - if description, ok := telegramResp["description"].(string); ok { - result["error"] = description - } - if errorCode, ok := telegramResp["error_code"].(float64); ok { - result["error_code"] = int(errorCode) - } - } - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} diff --git a/backend/internal/tools/textfile_tool.go b/backend/internal/tools/textfile_tool.go deleted file mode 100644 index ccb703c9..00000000 --- a/backend/internal/tools/textfile_tool.go +++ /dev/null @@ -1,151 +0,0 @@ -package tools - -import ( - "claraverse/internal/securefile" - "encoding/json" - "fmt" - "log" - "strings" -) - -// NewTextFileTool creates the create_text_file tool -func NewTextFileTool() *Tool { - return &Tool{ - Name: "create_text_file", - DisplayName: "Create Text File", - Description: `Creates a downloadable text-based file with specified content and extension. The file is stored for 30 days and requires an access code to download. Supports various formats like .txt, .json, .yaml, .xml, .csv, .md, .css, .js, .py, .go, .sh, .sql, .log, .ini, .toml, .env, and more. - -IMPORTANT: Do NOT use this tool to create .html files unless the user explicitly asks to "create a file", "save as file", or "download as file". For HTML content, use artifacts instead - they render HTML directly in the chat without needing file downloads. Only create .html files when the user specifically wants a downloadable file.`, - Icon: "FileCode", - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "content": map[string]interface{}{ - "type": "string", - "description": "The text content of the file", - }, - "filename": map[string]interface{}{ - "type": "string", - "description": "Desired filename without extension (e.g., 'config', 'data', 'script')", - }, - "extension": map[string]interface{}{ - "type": "string", - "description": "File extension without the dot (e.g., 'txt', 'json', 'yaml', 'csv', 'md', 'py', 'js')", - }, - }, - "required": []string{"content", "extension"}, - }, - Execute: executeCreateTextFile, - Source: ToolSourceBuiltin, - Category: "output", - Keywords: []string{"file", "text", "create", "generate", "save", "export", "json", "yaml", "csv", "code", "config", "write"}, - } -} - -func executeCreateTextFile(args map[string]interface{}) (string, error) { - // Extract parameters - content, ok := args["content"].(string) - if !ok || content == "" { - return "", fmt.Errorf("content is required") - } - - extension, ok := args["extension"].(string) - if !ok || extension == "" { - return "", fmt.Errorf("extension is required") - } - - // Clean extension (remove leading dot if present) - extension = strings.TrimPrefix(extension, ".") - - filename, _ := args["filename"].(string) - if filename == "" { - filename = "file" - } - - // Extract injected user context (set by ChatService) - userID, _ := args["__user_id__"].(string) - if userID == "" { - userID = "system" // Fallback for tools executed outside user context - } - - // Clean up internal parameters before logging - delete(args, "__user_id__") - delete(args, "__conversation_id__") - - log.Printf("📝 [TEXTFILE-TOOL] Generating text file: %s.%s (user: %s, length: %d chars)", filename, extension, userID, len(content)) - - // Determine MIME type based on extension - mimeType := getMimeTypeFromExtension(extension) - - // Full filename with extension - fullFilename := fmt.Sprintf("%s.%s", filename, extension) - - // Store in secure file service with 30-day retention and access code - secureFileService := securefile.GetService() - secureResult, err := secureFileService.CreateFile(userID, []byte(content), fullFilename, mimeType) - if err != nil { - log.Printf("❌ [TEXTFILE-TOOL] Failed to store file securely: %v", err) - return "", fmt.Errorf("failed to store file: %w", err) - } - - // Format result for AI - response := map[string]interface{}{ - "success": true, - "file_id": secureResult.ID, - "filename": secureResult.Filename, - "download_url": secureResult.DownloadURL, - "access_code": secureResult.AccessCode, - "size": secureResult.Size, - "file_type": "text", - "extension": extension, - "expires_at": secureResult.ExpiresAt.Format("2006-01-02"), - "message": fmt.Sprintf("Text file '%s' created successfully. Download link (valid for 30 days): %s", secureResult.Filename, secureResult.DownloadURL), - } - - responseJSON, _ := json.Marshal(response) - - log.Printf("✅ [TEXTFILE-TOOL] Text file generated and stored securely: %s (%d bytes, expires: %s)", - secureResult.Filename, secureResult.Size, secureResult.ExpiresAt.Format("2006-01-02")) - - return string(responseJSON), nil -} - -// getMimeTypeFromExtension returns the MIME type for a given file extension -func getMimeTypeFromExtension(ext string) string { - mimeTypes := map[string]string{ - "txt": "text/plain", - "json": "application/json", - "yaml": "application/x-yaml", - "yml": "application/x-yaml", - "xml": "application/xml", - "csv": "text/csv", - "md": "text/markdown", - "html": "text/html", - "htm": "text/html", - "css": "text/css", - "js": "text/javascript", - "ts": "text/typescript", - "py": "text/x-python", - "go": "text/x-go", - "rs": "text/x-rust", - "java": "text/x-java", - "c": "text/x-c", - "cpp": "text/x-c++", - "h": "text/x-c", - "hpp": "text/x-c++", - "sh": "application/x-sh", - "bash": "application/x-sh", - "sql": "application/sql", - "log": "text/plain", - "ini": "text/plain", - "toml": "application/toml", - "env": "text/plain", - "conf": "text/plain", - "cfg": "text/plain", - } - - if mime, ok := mimeTypes[strings.ToLower(ext)]; ok { - return mime - } - return "text/plain" -} diff --git a/backend/internal/tools/time_tool.go b/backend/internal/tools/time_tool.go deleted file mode 100644 index 165a5e5b..00000000 --- a/backend/internal/tools/time_tool.go +++ /dev/null @@ -1,47 +0,0 @@ -package tools - -import ( - "fmt" - "time" -) - -// NewTimeTool creates the get_current_time tool -func NewTimeTool() *Tool { - return &Tool{ - Name: "get_current_time", - DisplayName: "Get Current Time", - Description: "Get the current time in a specific timezone", - Icon: "Clock", - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "timezone": map[string]interface{}{ - "type": "string", - "description": "Timezone name (e.g., 'America/New_York', 'Asia/Tokyo', 'UTC'). Defaults to UTC.", - "default": "UTC", - }, - }, - "required": []string{}, // Timezone is now optional - }, - Execute: executeGetCurrentTime, - Source: ToolSourceBuiltin, - Category: "time", - Keywords: []string{"time", "date", "clock", "now", "current", "timezone", "datetime", "timestamp"}, - } -} - -func executeGetCurrentTime(args map[string]interface{}) (string, error) { - // Default to UTC if timezone not provided - timezone := "UTC" - if tz, ok := args["timezone"].(string); ok && tz != "" { - timezone = tz - } - - loc, err := time.LoadLocation(timezone) - if err != nil { - return "", fmt.Errorf("invalid timezone '%s', use format like 'America/New_York' or 'UTC'", timezone) - } - - currentTime := time.Now().In(loc) - return currentTime.Format("2006-01-02 15:04:05 MST"), nil -} diff --git a/backend/internal/tools/time_tool_test.go b/backend/internal/tools/time_tool_test.go deleted file mode 100644 index 555ddd84..00000000 --- a/backend/internal/tools/time_tool_test.go +++ /dev/null @@ -1,268 +0,0 @@ -package tools - -import ( - "strings" - "testing" - "time" -) - -func TestNewTimeTool(t *testing.T) { - tool := NewTimeTool() - - if tool.Name != "get_current_time" { - t.Errorf("Expected tool name 'get_current_time', got %s", tool.Name) - } - - if tool.DisplayName != "Get Current Time" { - t.Errorf("Expected display name 'Get Current Time', got %s", tool.DisplayName) - } - - if tool.Category != "time" { - t.Errorf("Expected category 'time', got %s", tool.Category) - } - - if tool.Source != ToolSourceBuiltin { - t.Errorf("Expected source 'builtin', got %s", tool.Source) - } - - if len(tool.Keywords) == 0 { - t.Error("Expected keywords to be populated") - } - - // Verify keywords contain expected terms - expectedKeywords := []string{"time", "date", "clock", "now"} - for _, keyword := range expectedKeywords { - found := false - for _, k := range tool.Keywords { - if k == keyword { - found = true - break - } - } - if !found { - t.Errorf("Expected keyword '%s' to be present", keyword) - } - } -} - -func TestExecuteGetCurrentTime_DefaultUTC(t *testing.T) { - // Execute without timezone - should default to UTC - args := map[string]interface{}{} - - result, err := executeGetCurrentTime(args) - if err != nil { - t.Fatalf("Failed to execute get_current_time with default timezone: %v", err) - } - - // Verify result contains UTC - if !strings.Contains(result, "UTC") { - t.Errorf("Expected result to contain 'UTC', got: %s", result) - } - - // Verify format (should be parseable) - _, parseErr := time.Parse("2006-01-02 15:04:05 MST", result) - if parseErr != nil { - t.Errorf("Result has invalid time format: %s", result) - } -} - -func TestExecuteGetCurrentTime_WithTimezone(t *testing.T) { - testCases := []struct { - timezone string - expectedPart string - }{ - {"UTC", "UTC"}, - {"America/New_York", "EST"}, // or EDT depending on daylight savings - {"Asia/Tokyo", "JST"}, - {"Europe/London", "GMT"}, // or BST depending on daylight savings - } - - for _, tc := range testCases { - t.Run(tc.timezone, func(t *testing.T) { - args := map[string]interface{}{ - "timezone": tc.timezone, - } - - result, err := executeGetCurrentTime(args) - if err != nil { - t.Fatalf("Failed to execute get_current_time with timezone %s: %v", tc.timezone, err) - } - - // Verify result is not empty - if result == "" { - t.Error("Expected non-empty result") - } - - // Verify format is parseable - _, parseErr := time.Parse("2006-01-02 15:04:05 MST", result) - if parseErr != nil { - t.Errorf("Result has invalid time format for timezone %s: %s", tc.timezone, result) - } - }) - } -} - -func TestExecuteGetCurrentTime_EmptyTimezone(t *testing.T) { - // Empty string should default to UTC - args := map[string]interface{}{ - "timezone": "", - } - - result, err := executeGetCurrentTime(args) - if err != nil { - t.Fatalf("Failed to execute get_current_time with empty timezone: %v", err) - } - - // Verify result contains UTC - if !strings.Contains(result, "UTC") { - t.Errorf("Expected result to contain 'UTC' for empty timezone, got: %s", result) - } -} - -func TestExecuteGetCurrentTime_InvalidTimezone(t *testing.T) { - args := map[string]interface{}{ - "timezone": "Invalid/Timezone", - } - - _, err := executeGetCurrentTime(args) - if err == nil { - t.Error("Expected error for invalid timezone, got nil") - } - - // Verify error message mentions the invalid timezone - if !strings.Contains(err.Error(), "Invalid/Timezone") { - t.Errorf("Expected error message to mention 'Invalid/Timezone', got: %v", err) - } -} - -func TestExecuteGetCurrentTime_FormatConsistency(t *testing.T) { - // Execute multiple times to ensure format consistency - args := map[string]interface{}{ - "timezone": "UTC", - } - - for i := 0; i < 5; i++ { - result, err := executeGetCurrentTime(args) - if err != nil { - t.Fatalf("Failed to execute get_current_time on iteration %d: %v", i, err) - } - - // Verify format (YYYY-MM-DD HH:MM:SS TZ) - parts := strings.Split(result, " ") - if len(parts) != 3 { - t.Errorf("Expected result to have 3 parts (date, time, timezone), got: %s", result) - } - - // Verify date part - dateParts := strings.Split(parts[0], "-") - if len(dateParts) != 3 { - t.Errorf("Expected date to have 3 parts (YYYY-MM-DD), got: %s", parts[0]) - } - - // Verify time part - timeParts := strings.Split(parts[1], ":") - if len(timeParts) != 3 { - t.Errorf("Expected time to have 3 parts (HH:MM:SS), got: %s", parts[1]) - } - } -} - -func TestTimeTool_IntegrationWithRegistry(t *testing.T) { - registry := &Registry{ - tools: make(map[string]*Tool), - } - - // Register time tool - err := registry.Register(NewTimeTool()) - if err != nil { - t.Fatalf("Failed to register time tool: %v", err) - } - - // Execute via registry with no args (should default to UTC) - result, err := registry.Execute("get_current_time", map[string]interface{}{}) - if err != nil { - t.Fatalf("Failed to execute time tool via registry: %v", err) - } - - if !strings.Contains(result, "UTC") { - t.Errorf("Expected result to contain 'UTC', got: %s", result) - } - - // Execute with timezone - result, err = registry.Execute("get_current_time", map[string]interface{}{ - "timezone": "America/New_York", - }) - if err != nil { - t.Fatalf("Failed to execute time tool with timezone via registry: %v", err) - } - - if result == "" { - t.Error("Expected non-empty result") - } -} - -func TestTimeTool_ParametersStructure(t *testing.T) { - tool := NewTimeTool() - - params := tool.Parameters - - // Verify type is object - if params["type"] != "object" { - t.Errorf("Expected parameters type 'object', got %v", params["type"]) - } - - // Verify properties exist - properties, ok := params["properties"].(map[string]interface{}) - if !ok { - t.Fatal("Properties should be a map") - } - - // Verify timezone property - timezone, ok := properties["timezone"].(map[string]interface{}) - if !ok { - t.Fatal("Timezone property should be a map") - } - - if timezone["type"] != "string" { - t.Errorf("Expected timezone type 'string', got %v", timezone["type"]) - } - - if timezone["default"] != "UTC" { - t.Errorf("Expected timezone default 'UTC', got %v", timezone["default"]) - } - - // Verify required is empty (timezone is optional) - required, ok := params["required"].([]string) - if !ok { - t.Fatal("Required should be a string array") - } - - if len(required) != 0 { - t.Errorf("Expected no required parameters, got %d", len(required)) - } -} - -func TestExecuteGetCurrentTime_CategoryAndKeywords(t *testing.T) { - tool := NewTimeTool() - - // Test that GetToolsByCategory would find this tool - registry := &Registry{ - tools: make(map[string]*Tool), - } - registry.Register(tool) - - timeTools := registry.GetToolsByCategory("time") - if len(timeTools) != 1 { - t.Errorf("Expected 1 time tool, got %d", len(timeTools)) - } - - if timeTools[0].Name != "get_current_time" { - t.Errorf("Expected get_current_time, got %s", timeTools[0].Name) - } - - // Test GetCategories - categories := registry.GetCategories() - if categories["time"] != 1 { - t.Errorf("Expected 1 tool in time category, got %d", categories["time"]) - } -} diff --git a/backend/internal/tools/tools.test b/backend/internal/tools/tools.test deleted file mode 100755 index 23c66bfe..00000000 Binary files a/backend/internal/tools/tools.test and /dev/null differ diff --git a/backend/internal/tools/transcribe_audio_tool.go b/backend/internal/tools/transcribe_audio_tool.go deleted file mode 100644 index 71621af8..00000000 --- a/backend/internal/tools/transcribe_audio_tool.go +++ /dev/null @@ -1,181 +0,0 @@ -package tools - -import ( - "claraverse/internal/audio" - "claraverse/internal/filecache" - "encoding/json" - "fmt" - "log" - "strings" -) - -// NewTranscribeAudioTool creates the transcribe_audio tool for speech-to-text -func NewTranscribeAudioTool() *Tool { - return &Tool{ - Name: "transcribe_audio", - DisplayName: "Transcribe Audio", - Description: "Transcribes speech from audio files to text using OpenAI Whisper. Supports MP3, WAV, M4A, OGG, FLAC, and WebM formats. Can translate non-English audio to English.", - Icon: "Mic", - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "file_id": map[string]interface{}{ - "type": "string", - "description": "The file ID of the uploaded audio file (from the upload response)", - }, - "language": map[string]interface{}{ - "type": "string", - "description": "Optional language code (e.g., 'en' for English, 'es' for Spanish, 'fr' for French). Auto-detected if not specified.", - }, - "prompt": map[string]interface{}{ - "type": "string", - "description": "Optional prompt to guide the transcription style or provide context (e.g., technical terms, names)", - }, - "translate_to_english": map[string]interface{}{ - "type": "boolean", - "description": "If true, translates the audio to English regardless of the source language. Useful for non-English audio files.", - "default": false, - }, - }, - "required": []string{"file_id"}, - }, - Execute: executeTranscribeAudio, - Source: ToolSourceBuiltin, - Category: "data_sources", - Keywords: []string{"audio", "transcribe", "speech", "voice", "whisper", "mp3", "wav", "speech-to-text", "stt", "translate"}, - } -} - -func executeTranscribeAudio(args map[string]interface{}) (string, error) { - // Extract file_id parameter - fileID, ok := args["file_id"].(string) - if !ok || fileID == "" { - return "", fmt.Errorf("file_id parameter is required and must be a string") - } - - // Extract optional language parameter - language := "" - if l, ok := args["language"].(string); ok { - language = l - } - - // Extract optional prompt parameter - prompt := "" - if p, ok := args["prompt"].(string); ok { - prompt = p - } - - // Extract translate_to_english parameter - translateToEnglish := false - if t, ok := args["translate_to_english"].(bool); ok { - translateToEnglish = t - } - - // Extract user context (injected by tool executor) - userID, _ := args["__user_id__"].(string) - - // Clean up internal parameters - delete(args, "__user_id__") - delete(args, "__conversation_id__") - - action := "Transcribing" - if translateToEnglish { - action = "Translating to English" - } - log.Printf("🎵 [TRANSCRIBE-AUDIO] %s audio file_id=%s language=%s (user=%s)", action, fileID, language, userID) - - // Get file cache service to validate file exists and user has access - fileCacheService := filecache.GetService() - - var file *filecache.CachedFile - if userID != "" { - var err error - file, err = fileCacheService.GetByUser(fileID, userID) - if err != nil { - // Try without user validation for workflow context - file, _ = fileCacheService.Get(fileID) - if file != nil && file.UserID != userID { - log.Printf("🚫 [TRANSCRIBE-AUDIO] Access denied: file %s belongs to different user", fileID) - return "", fmt.Errorf("access denied: you don't have permission to access this file") - } - } - } else { - file, _ = fileCacheService.Get(fileID) - } - - if file == nil { - log.Printf("❌ [TRANSCRIBE-AUDIO] File not found: %s", fileID) - return "", fmt.Errorf("audio file not found or has expired. Files are only available for 30 minutes after upload") - } - - // Validate it's an audio file - if !strings.HasPrefix(file.MimeType, "audio/") { - log.Printf("⚠️ [TRANSCRIBE-AUDIO] File is not audio: %s (%s)", fileID, file.MimeType) - return "", fmt.Errorf("file is not an audio file (type: %s). Supported formats: MP3, WAV, M4A, OGG, FLAC, WebM", file.MimeType) - } - - // Check if format is supported - if !audio.IsSupportedFormat(file.MimeType) { - return "", fmt.Errorf("audio format not supported: %s. Supported formats: %s", file.MimeType, strings.Join(audio.GetSupportedFormats(), ", ")) - } - - // Verify file path exists - if file.FilePath == "" { - return "", fmt.Errorf("audio file path not available") - } - - // Get the audio service - audioService := audio.GetService() - if audioService == nil { - return "", fmt.Errorf("audio service not available. Please configure your OpenAI API key") - } - - // Build the request - req := &audio.TranscribeRequest{ - AudioPath: file.FilePath, - Language: language, - Prompt: prompt, - TranslateToEnglish: translateToEnglish, - } - - // Call audio service - result, err := audioService.Transcribe(req) - if err != nil { - log.Printf("❌ [TRANSCRIBE-AUDIO] Transcription failed: %v", err) - return "", fmt.Errorf("failed to transcribe audio: %v", err) - } - - // Build response - response := map[string]interface{}{ - "success": true, - "file_id": fileID, - "filename": file.Filename, - "text": result.Text, - } - - if result.Language != "" { - response["detected_language"] = result.Language - } - if result.Duration > 0 { - response["duration_seconds"] = result.Duration - } - if language != "" { - response["requested_language"] = language - } - if translateToEnglish { - response["translated_to_english"] = true - } - - // Add word count - words := strings.Fields(result.Text) - response["word_count"] = len(words) - - responseJSON, err := json.Marshal(response) - if err != nil { - return "", fmt.Errorf("failed to marshal response: %w", err) - } - - log.Printf("✅ [TRANSCRIBE-AUDIO] Successfully transcribed %s (%d words, %.1fs)", file.Filename, len(words), result.Duration) - - return string(responseJSON), nil -} diff --git a/backend/internal/tools/trello_tool.go b/backend/internal/tools/trello_tool.go deleted file mode 100644 index 8533c9d5..00000000 --- a/backend/internal/tools/trello_tool.go +++ /dev/null @@ -1,423 +0,0 @@ -package tools - -import ( - "bytes" - "encoding/json" - "fmt" - "io" - "net/http" - "net/url" - "time" -) - -const trelloAPIBase = "https://api.trello.com/1" - -// NewTrelloBoardsTool creates a tool for listing Trello boards -func NewTrelloBoardsTool() *Tool { - return &Tool{ - Name: "trello_boards", - DisplayName: "List Trello Boards", - Description: "List all boards accessible to the authenticated user. Authentication is handled automatically via configured credentials.", - Icon: "Layout", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"trello", "boards", "list", "project", "kanban"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - }, - "required": []string{}, - }, - Execute: executeTrelloBoards, - } -} - -// NewTrelloListsTool creates a tool for listing Trello lists -func NewTrelloListsTool() *Tool { - return &Tool{ - Name: "trello_lists", - DisplayName: "List Trello Lists", - Description: "List all lists in a Trello board. Authentication is handled automatically.", - Icon: "List", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"trello", "lists", "board", "columns"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "board_id": map[string]interface{}{ - "type": "string", - "description": "Trello Board ID", - }, - }, - "required": []string{"board_id"}, - }, - Execute: executeTrelloLists, - } -} - -// NewTrelloCardsTool creates a tool for listing Trello cards -func NewTrelloCardsTool() *Tool { - return &Tool{ - Name: "trello_cards", - DisplayName: "List Trello Cards", - Description: "List cards from a Trello board or list. Authentication is handled automatically.", - Icon: "CreditCard", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"trello", "cards", "tasks", "items"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "board_id": map[string]interface{}{ - "type": "string", - "description": "Trello Board ID (required if list_id not provided)", - }, - "list_id": map[string]interface{}{ - "type": "string", - "description": "Trello List ID (optional, filters cards to this list)", - }, - }, - "required": []string{}, - }, - Execute: executeTrelloCards, - } -} - -// NewTrelloCreateCardTool creates a tool for creating Trello cards -func NewTrelloCreateCardTool() *Tool { - return &Tool{ - Name: "trello_create_card", - DisplayName: "Create Trello Card", - Description: "Create a new card in a Trello list. Authentication is handled automatically.", - Icon: "Plus", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"trello", "card", "create", "task", "add"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "list_id": map[string]interface{}{ - "type": "string", - "description": "Trello List ID where the card will be created", - }, - "name": map[string]interface{}{ - "type": "string", - "description": "Card title/name", - }, - "desc": map[string]interface{}{ - "type": "string", - "description": "Card description", - }, - "due": map[string]interface{}{ - "type": "string", - "description": "Due date (ISO 8601 format)", - }, - "labels": map[string]interface{}{ - "type": "string", - "description": "Comma-separated label IDs", - }, - }, - "required": []string{"list_id", "name"}, - }, - Execute: executeTrelloCreateCard, - } -} - -func trelloRequest(method, endpoint, apiKey, token string, body interface{}) (interface{}, error) { - // Add auth params to URL - u, _ := url.Parse(trelloAPIBase + endpoint) - q := u.Query() - q.Set("key", apiKey) - q.Set("token", token) - u.RawQuery = q.Encode() - - var reqBody io.Reader - if body != nil { - jsonBody, err := json.Marshal(body) - if err != nil { - return nil, fmt.Errorf("failed to marshal request body: %w", err) - } - reqBody = bytes.NewBuffer(jsonBody) - } - - req, err := http.NewRequest(method, u.String(), reqBody) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - if body != nil { - req.Header.Set("Content-Type", "application/json") - } - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return nil, fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - respBody, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - if resp.StatusCode >= 400 { - return nil, fmt.Errorf("Trello API error: %s (status %d)", string(respBody), resp.StatusCode) - } - - var result interface{} - if err := json.Unmarshal(respBody, &result); err != nil { - return nil, fmt.Errorf("failed to parse response: %w", err) - } - - return result, nil -} - -func getTrelloCredentials(args map[string]interface{}) (string, string, error) { - credData, err := GetCredentialData(args, "trello") - if err != nil { - return "", "", fmt.Errorf("failed to get Trello credentials: %w", err) - } - - apiKey, _ := credData["api_key"].(string) - token, _ := credData["token"].(string) - - if apiKey == "" || token == "" { - return "", "", fmt.Errorf("api_key and token are required") - } - - return apiKey, token, nil -} - -func executeTrelloBoards(args map[string]interface{}) (string, error) { - apiKey, token, err := getTrelloCredentials(args) - if err != nil { - return "", err - } - - result, err := trelloRequest("GET", "/members/me/boards", apiKey, token, nil) - if err != nil { - return "", err - } - - boards, ok := result.([]interface{}) - if !ok { - return "", fmt.Errorf("unexpected response format") - } - - simplifiedBoards := make([]map[string]interface{}, 0) - for _, b := range boards { - board, ok := b.(map[string]interface{}) - if !ok { - continue - } - simplifiedBoards = append(simplifiedBoards, map[string]interface{}{ - "id": board["id"], - "name": board["name"], - "url": board["url"], - "closed": board["closed"], - }) - } - - response := map[string]interface{}{ - "success": true, - "count": len(simplifiedBoards), - "boards": simplifiedBoards, - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - -func executeTrelloLists(args map[string]interface{}) (string, error) { - apiKey, token, err := getTrelloCredentials(args) - if err != nil { - return "", err - } - - boardID, _ := args["board_id"].(string) - if boardID == "" { - return "", fmt.Errorf("board_id is required") - } - - endpoint := fmt.Sprintf("/boards/%s/lists", boardID) - result, err := trelloRequest("GET", endpoint, apiKey, token, nil) - if err != nil { - return "", err - } - - lists, ok := result.([]interface{}) - if !ok { - return "", fmt.Errorf("unexpected response format") - } - - simplifiedLists := make([]map[string]interface{}, 0) - for _, l := range lists { - list, ok := l.(map[string]interface{}) - if !ok { - continue - } - simplifiedLists = append(simplifiedLists, map[string]interface{}{ - "id": list["id"], - "name": list["name"], - "closed": list["closed"], - "pos": list["pos"], - }) - } - - response := map[string]interface{}{ - "success": true, - "count": len(simplifiedLists), - "lists": simplifiedLists, - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - -func executeTrelloCards(args map[string]interface{}) (string, error) { - apiKey, token, err := getTrelloCredentials(args) - if err != nil { - return "", err - } - - boardID, _ := args["board_id"].(string) - listID, _ := args["list_id"].(string) - - var endpoint string - if listID != "" { - endpoint = fmt.Sprintf("/lists/%s/cards", listID) - } else if boardID != "" { - endpoint = fmt.Sprintf("/boards/%s/cards", boardID) - } else { - return "", fmt.Errorf("either board_id or list_id is required") - } - - result, err := trelloRequest("GET", endpoint, apiKey, token, nil) - if err != nil { - return "", err - } - - cards, ok := result.([]interface{}) - if !ok { - return "", fmt.Errorf("unexpected response format") - } - - simplifiedCards := make([]map[string]interface{}, 0) - for _, c := range cards { - card, ok := c.(map[string]interface{}) - if !ok { - continue - } - simplifiedCards = append(simplifiedCards, map[string]interface{}{ - "id": card["id"], - "name": card["name"], - "desc": card["desc"], - "url": card["url"], - "due": card["due"], - "closed": card["closed"], - "idList": card["idList"], - "labels": card["labels"], - "pos": card["pos"], - }) - } - - response := map[string]interface{}{ - "success": true, - "count": len(simplifiedCards), - "cards": simplifiedCards, - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - -func executeTrelloCreateCard(args map[string]interface{}) (string, error) { - apiKey, token, err := getTrelloCredentials(args) - if err != nil { - return "", err - } - - listID, _ := args["list_id"].(string) - name, _ := args["name"].(string) - - if listID == "" || name == "" { - return "", fmt.Errorf("list_id and name are required") - } - - // Build query params - endpoint := "/cards" - u, _ := url.Parse(trelloAPIBase + endpoint) - q := u.Query() - q.Set("key", apiKey) - q.Set("token", token) - q.Set("idList", listID) - q.Set("name", name) - - if desc, ok := args["desc"].(string); ok && desc != "" { - q.Set("desc", desc) - } - if due, ok := args["due"].(string); ok && due != "" { - q.Set("due", due) - } - if labels, ok := args["labels"].(string); ok && labels != "" { - q.Set("idLabels", labels) - } - - u.RawQuery = q.Encode() - - req, err := http.NewRequest("POST", u.String(), nil) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - respBody, err := io.ReadAll(resp.Body) - if err != nil { - return "", fmt.Errorf("failed to read response: %w", err) - } - - if resp.StatusCode >= 400 { - return "", fmt.Errorf("Trello API error: %s", string(respBody)) - } - - var result map[string]interface{} - if err := json.Unmarshal(respBody, &result); err != nil { - return "", fmt.Errorf("failed to parse response: %w", err) - } - - response := map[string]interface{}{ - "success": true, - "message": "Card created successfully", - "card_id": result["id"], - "url": result["url"], - "name": result["name"], - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - diff --git a/backend/internal/tools/twilio_tool.go b/backend/internal/tools/twilio_tool.go deleted file mode 100644 index 2b1e5036..00000000 --- a/backend/internal/tools/twilio_tool.go +++ /dev/null @@ -1,278 +0,0 @@ -package tools - -import ( - "encoding/base64" - "encoding/json" - "fmt" - "io" - "net/http" - "net/url" - "strings" - "time" -) - -// NewTwilioSMSTool creates a Twilio SMS tool -func NewTwilioSMSTool() *Tool { - return &Tool{ - Name: "twilio_send_sms", - DisplayName: "Twilio SMS", - Description: `Send SMS or MMS messages via Twilio. - -Features: -- Send text messages to any phone number -- Send MMS with media attachments -- Support for international numbers - -Numbers must be in E.164 format (e.g., +1234567890). -Authentication is handled automatically via configured Twilio credentials.`, - Icon: "Phone", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"twilio", "sms", "text", "message", "phone", "mms"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "to": map[string]interface{}{ - "type": "string", - "description": "Destination phone number in E.164 format (e.g., +1234567890)", - }, - "body": map[string]interface{}{ - "type": "string", - "description": "The text message content (max 1600 characters)", - }, - "from": map[string]interface{}{ - "type": "string", - "description": "Twilio phone number to send from (uses default if not specified)", - }, - "media_url": map[string]interface{}{ - "type": "string", - "description": "URL of media to include (for MMS)", - }, - }, - "required": []string{"to", "body"}, - }, - Execute: executeTwilioSMS, - } -} - -// NewTwilioWhatsAppTool creates a Twilio WhatsApp tool -func NewTwilioWhatsAppTool() *Tool { - return &Tool{ - Name: "twilio_send_whatsapp", - DisplayName: "Twilio WhatsApp", - Description: `Send WhatsApp messages via Twilio. - -Features: -- Send WhatsApp messages -- Send media attachments -- Requires a Twilio WhatsApp-enabled number - -Numbers must be in E.164 format (e.g., +1234567890). -Authentication is handled automatically via configured Twilio credentials.`, - Icon: "MessageSquare", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"twilio", "whatsapp", "message", "chat"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "to": map[string]interface{}{ - "type": "string", - "description": "Destination WhatsApp number in E.164 format (e.g., +1234567890)", - }, - "body": map[string]interface{}{ - "type": "string", - "description": "The message content", - }, - "from": map[string]interface{}{ - "type": "string", - "description": "Twilio WhatsApp number to send from", - }, - "media_url": map[string]interface{}{ - "type": "string", - "description": "URL of media to include", - }, - }, - "required": []string{"to", "body"}, - }, - Execute: executeTwilioWhatsApp, - } -} - -func executeTwilioSMS(args map[string]interface{}) (string, error) { - credData, err := GetCredentialData(args, "twilio") - if err != nil { - return "", fmt.Errorf("failed to get Twilio credentials: %w", err) - } - - accountSID, _ := credData["account_sid"].(string) - authToken, _ := credData["auth_token"].(string) - defaultFrom, _ := credData["from_number"].(string) - - if accountSID == "" || authToken == "" { - return "", fmt.Errorf("Twilio credentials incomplete: account_sid and auth_token are required") - } - - to, _ := args["to"].(string) - body, _ := args["body"].(string) - from, _ := args["from"].(string) - mediaURL, _ := args["media_url"].(string) - - if to == "" { - return "", fmt.Errorf("'to' phone number is required") - } - if body == "" { - return "", fmt.Errorf("'body' message content is required") - } - - if from == "" { - from = defaultFrom - } - if from == "" { - return "", fmt.Errorf("'from' phone number is required") - } - - // Build form data - data := url.Values{} - data.Set("To", to) - data.Set("From", from) - data.Set("Body", body) - if mediaURL != "" { - data.Set("MediaUrl", mediaURL) - } - - apiURL := fmt.Sprintf("https://api.twilio.com/2010-04-01/Accounts/%s/Messages.json", accountSID) - req, err := http.NewRequest("POST", apiURL, strings.NewReader(data.Encode())) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - auth := base64.StdEncoding.EncodeToString([]byte(accountSID + ":" + authToken)) - req.Header.Set("Authorization", "Basic "+auth) - req.Header.Set("Content-Type", "application/x-www-form-urlencoded") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to send request: %w", err) - } - defer resp.Body.Close() - - respBody, _ := io.ReadAll(resp.Body) - var result map[string]interface{} - json.Unmarshal(respBody, &result) - - if resp.StatusCode >= 400 { - errMsg := "unknown error" - if msg, ok := result["message"].(string); ok { - errMsg = msg - } - return "", fmt.Errorf("Twilio API error: %s", errMsg) - } - - output := map[string]interface{}{ - "success": true, - "message_sid": result["sid"], - "status": result["status"], - "to": result["to"], - "from": result["from"], - } - jsonResult, _ := json.MarshalIndent(output, "", " ") - return string(jsonResult), nil -} - -func executeTwilioWhatsApp(args map[string]interface{}) (string, error) { - credData, err := GetCredentialData(args, "twilio") - if err != nil { - return "", fmt.Errorf("failed to get Twilio credentials: %w", err) - } - - accountSID, _ := credData["account_sid"].(string) - authToken, _ := credData["auth_token"].(string) - defaultFrom, _ := credData["from_number"].(string) - - if accountSID == "" || authToken == "" { - return "", fmt.Errorf("Twilio credentials incomplete: account_sid and auth_token are required") - } - - to, _ := args["to"].(string) - body, _ := args["body"].(string) - from, _ := args["from"].(string) - mediaURL, _ := args["media_url"].(string) - - if to == "" { - return "", fmt.Errorf("'to' phone number is required") - } - if body == "" { - return "", fmt.Errorf("'body' message content is required") - } - - // Format WhatsApp numbers - if !strings.HasPrefix(to, "whatsapp:") { - to = "whatsapp:" + to - } - if from == "" { - from = defaultFrom - } - if from == "" { - return "", fmt.Errorf("'from' WhatsApp number is required") - } - if !strings.HasPrefix(from, "whatsapp:") { - from = "whatsapp:" + from - } - - data := url.Values{} - data.Set("To", to) - data.Set("From", from) - data.Set("Body", body) - if mediaURL != "" { - data.Set("MediaUrl", mediaURL) - } - - apiURL := fmt.Sprintf("https://api.twilio.com/2010-04-01/Accounts/%s/Messages.json", accountSID) - req, err := http.NewRequest("POST", apiURL, strings.NewReader(data.Encode())) - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - auth := base64.StdEncoding.EncodeToString([]byte(accountSID + ":" + authToken)) - req.Header.Set("Authorization", "Basic "+auth) - req.Header.Set("Content-Type", "application/x-www-form-urlencoded") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("failed to send request: %w", err) - } - defer resp.Body.Close() - - respBody, _ := io.ReadAll(resp.Body) - var result map[string]interface{} - json.Unmarshal(respBody, &result) - - if resp.StatusCode >= 400 { - errMsg := "unknown error" - if msg, ok := result["message"].(string); ok { - errMsg = msg - } - return "", fmt.Errorf("Twilio API error: %s", errMsg) - } - - output := map[string]interface{}{ - "success": true, - "message_sid": result["sid"], - "status": result["status"], - "to": result["to"], - "from": result["from"], - } - jsonResult, _ := json.MarshalIndent(output, "", " ") - return string(jsonResult), nil -} diff --git a/backend/internal/tools/webhook_tool.go b/backend/internal/tools/webhook_tool.go deleted file mode 100644 index adb11621..00000000 --- a/backend/internal/tools/webhook_tool.go +++ /dev/null @@ -1,185 +0,0 @@ -package tools - -import ( - "bytes" - "claraverse/internal/security" - "encoding/json" - "fmt" - "io" - "net/http" - "strings" - "time" -) - -// NewWebhookTool creates a new generic webhook/HTTP tool -func NewWebhookTool() *Tool { - return &Tool{ - Name: "send_webhook", - DisplayName: "Send Webhook", - Description: "Send HTTP requests to any URL. Use for APIs, webhooks, notifications, and integrations. Supports GET, POST, PUT, DELETE methods with custom headers and body.", - Icon: "Send", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"webhook", "http", "api", "request", "post", "get", "send", "notify", "integration", "rest"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "ID of a stored webhook/REST API credential (preferred over raw url)", - }, - "url": map[string]interface{}{ - "type": "string", - "description": "Target URL - only use if no credential_id is provided (must include http:// or https://)", - }, - "method": map[string]interface{}{ - "type": "string", - "description": "HTTP method to use", - "enum": []string{"GET", "POST", "PUT", "DELETE", "PATCH"}, - "default": "POST", - }, - "headers": map[string]interface{}{ - "type": "object", - "description": "HTTP headers to include (optional). Example: {\"Authorization\": \"Bearer token\"}", - "additionalProperties": map[string]interface{}{ - "type": "string", - }, - }, - "body": map[string]interface{}{ - "type": "string", - "description": "Request body (typically JSON string for POST/PUT requests)", - }, - "content_type": map[string]interface{}{ - "type": "string", - "description": "Content-Type header value", - "default": "application/json", - }, - }, - "required": []string{}, - }, - Execute: executeWebhook, - } -} - -func executeWebhook(args map[string]interface{}) (string, error) { - // Resolve URL from credential or direct parameter - // Try multiple integration types that use webhooks/URLs - url, err := ResolveWebhookURL(args, "custom_webhook") - if err != nil { - // Try rest_api type - url, err = ResolveWebhookURL(args, "rest_api") - if err != nil { - // Fallback: check for direct url if credential resolution failed - if u, ok := args["url"].(string); ok && u != "" { - url = u - } else { - return "", fmt.Errorf("either url or credential_id is required") - } - } - } - - // Validate URL - if !strings.HasPrefix(url, "http://") && !strings.HasPrefix(url, "https://") { - return "", fmt.Errorf("url must start with http:// or https://") - } - - // SSRF protection: block requests to internal/private networks - if err := security.ValidateURLForSSRF(url); err != nil { - return "", fmt.Errorf("SSRF protection: %w", err) - } - - // Extract method (default: POST) - method := "POST" - if m, ok := args["method"].(string); ok && m != "" { - method = strings.ToUpper(m) - } - - // Extract body - body := "" - if b, ok := args["body"].(string); ok { - body = b - } - - // Extract content type (default: application/json) - contentType := "application/json" - if ct, ok := args["content_type"].(string); ok && ct != "" { - contentType = ct - } - - // Extract headers - headers := make(map[string]string) - if headersRaw, ok := args["headers"].(map[string]interface{}); ok { - for key, value := range headersRaw { - headers[key] = fmt.Sprintf("%v", value) - } - } - - // Create HTTP client with timeout - client := &http.Client{ - Timeout: 30 * time.Second, - } - - // Create request - var req *http.Request - - if body != "" { - req, err = http.NewRequest(method, url, bytes.NewBufferString(body)) - } else { - req, err = http.NewRequest(method, url, nil) - } - - if err != nil { - return "", fmt.Errorf("failed to create request: %w", err) - } - - // Set content type for requests with body - if body != "" { - req.Header.Set("Content-Type", contentType) - } - - // Set custom headers - for key, value := range headers { - req.Header.Set(key, value) - } - - // Execute request - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - // Read response body - respBody, err := io.ReadAll(resp.Body) - if err != nil { - return "", fmt.Errorf("failed to read response: %w", err) - } - - // Truncate response if too long - respBodyStr := string(respBody) - if len(respBodyStr) > 5000 { - respBodyStr = respBodyStr[:5000] + "... (truncated)" - } - - // Build result - result := map[string]interface{}{ - "success": resp.StatusCode >= 200 && resp.StatusCode < 300, - "status_code": resp.StatusCode, - "status": resp.Status, - "url": url, - "method": method, - "response": respBodyStr, - } - - // Add response headers - respHeaders := make(map[string]string) - for key, values := range resp.Header { - if len(values) > 0 { - respHeaders[key] = values[0] - } - } - result["response_headers"] = respHeaders - - jsonResult, _ := json.MarshalIndent(result, "", " ") - return string(jsonResult), nil -} diff --git a/backend/internal/tools/x_twitter_tool.go b/backend/internal/tools/x_twitter_tool.go deleted file mode 100644 index a3a0cc1b..00000000 --- a/backend/internal/tools/x_twitter_tool.go +++ /dev/null @@ -1,610 +0,0 @@ -package tools - -import ( - "bytes" - "crypto/hmac" - "crypto/sha1" - "encoding/base64" - "encoding/json" - "fmt" - "io" - "math/rand" - "net/http" - "net/url" - "sort" - "strconv" - "strings" - "time" -) - -const xAPIBase = "https://api.x.com/2" - -// NewXSearchPostsTool creates a tool for searching X posts -func NewXSearchPostsTool() *Tool { - return &Tool{ - Name: "x_search_posts", - DisplayName: "Search X Posts", - Description: "Search for posts on X (Twitter) using the v2 API. Supports advanced query operators. Authentication is handled automatically via configured credentials.", - Icon: "Search", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"x", "twitter", "search", "posts", "tweets", "social"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "query": map[string]interface{}{ - "type": "string", - "description": "Search query. Supports operators like from:, to:, #hashtag, @mention, has:media, is:retweet, lang:, etc.", - }, - "max_results": map[string]interface{}{ - "type": "number", - "description": "Maximum number of results (10-100, default 10)", - }, - "sort_order": map[string]interface{}{ - "type": "string", - "enum": []string{"recency", "relevancy"}, - "description": "Sort order (default: recency)", - }, - }, - "required": []string{"query"}, - }, - Execute: executeXSearchPosts, - } -} - -// NewXPostTweetTool creates a tool for posting tweets -func NewXPostTweetTool() *Tool { - return &Tool{ - Name: "x_post_tweet", - DisplayName: "Post to X", - Description: "Post a new tweet to X (Twitter). Requires OAuth 1.0a credentials (API Key, API Secret, Access Token, Access Token Secret). Authentication is handled automatically.", - Icon: "Send", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"x", "twitter", "post", "tweet", "publish", "social"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "text": map[string]interface{}{ - "type": "string", - "description": "The text content of the tweet (max 280 characters)", - }, - "reply_to": map[string]interface{}{ - "type": "string", - "description": "Tweet ID to reply to (optional)", - }, - "quote_tweet_id": map[string]interface{}{ - "type": "string", - "description": "Tweet ID to quote (optional)", - }, - }, - "required": []string{"text"}, - }, - Execute: executeXPostTweet, - } -} - -// NewXGetUserTool creates a tool for getting user info -func NewXGetUserTool() *Tool { - return &Tool{ - Name: "x_get_user", - DisplayName: "Get X User", - Description: "Get information about an X (Twitter) user by username. Authentication is handled automatically.", - Icon: "User", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"x", "twitter", "user", "profile", "account"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "username": map[string]interface{}{ - "type": "string", - "description": "X username (without @)", - }, - }, - "required": []string{"username"}, - }, - Execute: executeXGetUser, - } -} - -// NewXGetUserPostsTool creates a tool for getting a user's posts -func NewXGetUserPostsTool() *Tool { - return &Tool{ - Name: "x_get_user_posts", - DisplayName: "Get User's X Posts", - Description: "Get recent posts from an X (Twitter) user. Authentication is handled automatically.", - Icon: "FileText", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"x", "twitter", "user", "posts", "tweets", "timeline"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "user_id": map[string]interface{}{ - "type": "string", - "description": "X user ID (numeric)", - }, - "username": map[string]interface{}{ - "type": "string", - "description": "X username (alternative to user_id, will be resolved)", - }, - "max_results": map[string]interface{}{ - "type": "number", - "description": "Maximum number of results (5-100, default 10)", - }, - }, - "required": []string{}, - }, - Execute: executeXGetUserPosts, - } -} - -type xCredentials struct { - BearerToken string - APIKey string - APISecret string - AccessToken string - AccessTokenSecret string -} - -func getXCredentials(args map[string]interface{}) (*xCredentials, error) { - credData, err := GetCredentialData(args, "x_twitter") - if err != nil { - return nil, fmt.Errorf("failed to get X credentials: %w", err) - } - - creds := &xCredentials{ - BearerToken: credData["bearer_token"].(string), - APIKey: "", - APISecret: "", - AccessToken: "", - AccessTokenSecret: "", - } - - if apiKey, ok := credData["api_key"].(string); ok { - creds.APIKey = apiKey - } - if apiSecret, ok := credData["api_secret"].(string); ok { - creds.APISecret = apiSecret - } - if accessToken, ok := credData["access_token"].(string); ok { - creds.AccessToken = accessToken - } - if accessTokenSecret, ok := credData["access_token_secret"].(string); ok { - creds.AccessTokenSecret = accessTokenSecret - } - - if creds.BearerToken == "" { - return nil, fmt.Errorf("bearer_token is required") - } - - return creds, nil -} - -func xBearerRequest(method, endpoint, bearerToken string, body interface{}) (map[string]interface{}, error) { - var reqBody io.Reader - if body != nil { - jsonBody, err := json.Marshal(body) - if err != nil { - return nil, fmt.Errorf("failed to marshal request body: %w", err) - } - reqBody = bytes.NewBuffer(jsonBody) - } - - req, err := http.NewRequest(method, xAPIBase+endpoint, reqBody) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Authorization", "Bearer "+bearerToken) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return nil, fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - respBody, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - var result map[string]interface{} - if err := json.Unmarshal(respBody, &result); err != nil { - return nil, fmt.Errorf("failed to parse response: %w", err) - } - - if resp.StatusCode >= 400 { - errMsg := "X API error" - if errors, ok := result["errors"].([]interface{}); ok && len(errors) > 0 { - if errObj, ok := errors[0].(map[string]interface{}); ok { - if msg, ok := errObj["message"].(string); ok { - errMsg = msg - } - } - } - if detail, ok := result["detail"].(string); ok { - errMsg = detail - } - return nil, fmt.Errorf("%s (status %d)", errMsg, resp.StatusCode) - } - - return result, nil -} - -// OAuth 1.0a signing for posting tweets -func generateOAuthSignature(method, urlStr string, params map[string]string, consumerSecret, tokenSecret string) string { - // Sort parameters - keys := make([]string, 0, len(params)) - for k := range params { - keys = append(keys, k) - } - sort.Strings(keys) - - // Create parameter string - var paramPairs []string - for _, k := range keys { - paramPairs = append(paramPairs, fmt.Sprintf("%s=%s", url.QueryEscape(k), url.QueryEscape(params[k]))) - } - paramString := strings.Join(paramPairs, "&") - - // Create signature base string - signatureBase := fmt.Sprintf("%s&%s&%s", - strings.ToUpper(method), - url.QueryEscape(urlStr), - url.QueryEscape(paramString), - ) - - // Create signing key - signingKey := fmt.Sprintf("%s&%s", url.QueryEscape(consumerSecret), url.QueryEscape(tokenSecret)) - - // Generate HMAC-SHA1 - h := hmac.New(sha1.New, []byte(signingKey)) - h.Write([]byte(signatureBase)) - signature := base64.StdEncoding.EncodeToString(h.Sum(nil)) - - return signature -} - -func generateNonce() string { - const letters = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789" - b := make([]byte, 32) - for i := range b { - b[i] = letters[rand.Intn(len(letters))] - } - return string(b) -} - -func xOAuthRequest(method, endpoint string, creds *xCredentials, body interface{}) (map[string]interface{}, error) { - urlStr := xAPIBase + endpoint - - // OAuth parameters - oauthParams := map[string]string{ - "oauth_consumer_key": creds.APIKey, - "oauth_nonce": generateNonce(), - "oauth_signature_method": "HMAC-SHA1", - "oauth_timestamp": strconv.FormatInt(time.Now().Unix(), 10), - "oauth_token": creds.AccessToken, - "oauth_version": "1.0", - } - - // Generate signature - signature := generateOAuthSignature(method, urlStr, oauthParams, creds.APISecret, creds.AccessTokenSecret) - oauthParams["oauth_signature"] = signature - - // Build Authorization header - var authParts []string - for k, v := range oauthParams { - authParts = append(authParts, fmt.Sprintf(`%s="%s"`, k, url.QueryEscape(v))) - } - sort.Strings(authParts) - authHeader := "OAuth " + strings.Join(authParts, ", ") - - // Create request - var reqBody io.Reader - if body != nil { - jsonBody, err := json.Marshal(body) - if err != nil { - return nil, fmt.Errorf("failed to marshal request body: %w", err) - } - reqBody = bytes.NewBuffer(jsonBody) - } - - req, err := http.NewRequest(method, urlStr, reqBody) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Authorization", authHeader) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return nil, fmt.Errorf("request failed: %w", err) - } - defer resp.Body.Close() - - respBody, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - var result map[string]interface{} - if err := json.Unmarshal(respBody, &result); err != nil { - return nil, fmt.Errorf("failed to parse response: %w", err) - } - - if resp.StatusCode >= 400 { - errMsg := "X API error" - if detail, ok := result["detail"].(string); ok { - errMsg = detail - } - if title, ok := result["title"].(string); ok { - errMsg = title - } - return nil, fmt.Errorf("%s (status %d)", errMsg, resp.StatusCode) - } - - return result, nil -} - -func executeXSearchPosts(args map[string]interface{}) (string, error) { - creds, err := getXCredentials(args) - if err != nil { - return "", err - } - - query, _ := args["query"].(string) - if query == "" { - return "", fmt.Errorf("query is required") - } - - maxResults := 10 - if mr, ok := args["max_results"].(float64); ok && mr > 0 { - maxResults = int(mr) - if maxResults < 10 { - maxResults = 10 - } - if maxResults > 100 { - maxResults = 100 - } - } - - // Build endpoint with query params - params := url.Values{} - params.Set("query", query) - params.Set("max_results", strconv.Itoa(maxResults)) - params.Set("tweet.fields", "created_at,public_metrics,author_id,conversation_id") - params.Set("expansions", "author_id") - params.Set("user.fields", "name,username,verified") - - if sortOrder, ok := args["sort_order"].(string); ok && sortOrder != "" { - params.Set("sort_order", sortOrder) - } - - endpoint := "/tweets/search/recent?" + params.Encode() - - result, err := xBearerRequest("GET", endpoint, creds.BearerToken, nil) - if err != nil { - return "", err - } - - // Process results - posts := []map[string]interface{}{} - if data, ok := result["data"].([]interface{}); ok { - for _, p := range data { - if post, ok := p.(map[string]interface{}); ok { - posts = append(posts, post) - } - } - } - - // Get user info - users := map[string]map[string]interface{}{} - if includes, ok := result["includes"].(map[string]interface{}); ok { - if usersData, ok := includes["users"].([]interface{}); ok { - for _, u := range usersData { - if user, ok := u.(map[string]interface{}); ok { - if id, ok := user["id"].(string); ok { - users[id] = user - } - } - } - } - } - - // Enrich posts with user info - for i, post := range posts { - if authorID, ok := post["author_id"].(string); ok { - if user, exists := users[authorID]; exists { - posts[i]["author"] = user - } - } - } - - response := map[string]interface{}{ - "success": true, - "count": len(posts), - "posts": posts, - "meta": result["meta"], - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - -func executeXPostTweet(args map[string]interface{}) (string, error) { - creds, err := getXCredentials(args) - if err != nil { - return "", err - } - - // Check for OAuth 1.0a credentials - if creds.APIKey == "" || creds.APISecret == "" || creds.AccessToken == "" || creds.AccessTokenSecret == "" { - return "", fmt.Errorf("posting tweets requires OAuth 1.0a credentials (api_key, api_secret, access_token, access_token_secret)") - } - - text, _ := args["text"].(string) - if text == "" { - return "", fmt.Errorf("text is required") - } - - if len(text) > 280 { - return "", fmt.Errorf("tweet text exceeds 280 characters") - } - - body := map[string]interface{}{ - "text": text, - } - - if replyTo, ok := args["reply_to"].(string); ok && replyTo != "" { - body["reply"] = map[string]interface{}{ - "in_reply_to_tweet_id": replyTo, - } - } - - if quoteTweetID, ok := args["quote_tweet_id"].(string); ok && quoteTweetID != "" { - body["quote_tweet_id"] = quoteTweetID - } - - result, err := xOAuthRequest("POST", "/tweets", creds, body) - if err != nil { - return "", err - } - - data, _ := result["data"].(map[string]interface{}) - - response := map[string]interface{}{ - "success": true, - "message": "Tweet posted successfully", - "tweet_id": data["id"], - "text": data["text"], - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - -func executeXGetUser(args map[string]interface{}) (string, error) { - creds, err := getXCredentials(args) - if err != nil { - return "", err - } - - username, _ := args["username"].(string) - if username == "" { - return "", fmt.Errorf("username is required") - } - - // Remove @ if present - username = strings.TrimPrefix(username, "@") - - params := url.Values{} - params.Set("user.fields", "created_at,description,public_metrics,verified,profile_image_url,location,url") - - endpoint := fmt.Sprintf("/users/by/username/%s?%s", username, params.Encode()) - - result, err := xBearerRequest("GET", endpoint, creds.BearerToken, nil) - if err != nil { - return "", err - } - - data, _ := result["data"].(map[string]interface{}) - - response := map[string]interface{}{ - "success": true, - "user": data, - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - -func executeXGetUserPosts(args map[string]interface{}) (string, error) { - creds, err := getXCredentials(args) - if err != nil { - return "", err - } - - userID, _ := args["user_id"].(string) - username, _ := args["username"].(string) - - // If username provided, resolve to user ID first - if userID == "" && username != "" { - username = strings.TrimPrefix(username, "@") - userResult, err := xBearerRequest("GET", fmt.Sprintf("/users/by/username/%s", username), creds.BearerToken, nil) - if err != nil { - return "", fmt.Errorf("failed to resolve username: %w", err) - } - if data, ok := userResult["data"].(map[string]interface{}); ok { - userID, _ = data["id"].(string) - } - } - - if userID == "" { - return "", fmt.Errorf("either user_id or username is required") - } - - maxResults := 10 - if mr, ok := args["max_results"].(float64); ok && mr > 0 { - maxResults = int(mr) - if maxResults < 5 { - maxResults = 5 - } - if maxResults > 100 { - maxResults = 100 - } - } - - params := url.Values{} - params.Set("max_results", strconv.Itoa(maxResults)) - params.Set("tweet.fields", "created_at,public_metrics,conversation_id") - - endpoint := fmt.Sprintf("/users/%s/tweets?%s", userID, params.Encode()) - - result, err := xBearerRequest("GET", endpoint, creds.BearerToken, nil) - if err != nil { - return "", err - } - - posts := []map[string]interface{}{} - if data, ok := result["data"].([]interface{}); ok { - for _, p := range data { - if post, ok := p.(map[string]interface{}); ok { - posts = append(posts, post) - } - } - } - - response := map[string]interface{}{ - "success": true, - "user_id": userID, - "count": len(posts), - "posts": posts, - "meta": result["meta"], - } - - jsonResult, _ := json.MarshalIndent(response, "", " ") - return string(jsonResult), nil -} - diff --git a/backend/internal/tools/zoom_tool.go b/backend/internal/tools/zoom_tool.go deleted file mode 100644 index 646fe65d..00000000 --- a/backend/internal/tools/zoom_tool.go +++ /dev/null @@ -1,772 +0,0 @@ -package tools - -import ( - "bytes" - "encoding/base64" - "encoding/json" - "fmt" - "io" - "net/http" - "net/url" - "strings" - "sync" - "time" -) - -// Zoom OAuth token cache to avoid generating new tokens on every request -var ( - zoomTokenCache = make(map[string]*zoomCachedToken) - zoomTokenCacheMu sync.RWMutex -) - -type zoomCachedToken struct { - AccessToken string - ExpiresAt time.Time -} - -// NewZoomTool creates a Zoom meeting and webinar management tool -func NewZoomTool() *Tool { - return &Tool{ - Name: "zoom_meeting", - DisplayName: "Zoom Meeting & Webinar", - Description: `Manage Zoom meetings and webinars - create, list, get details, and create registrations. - -Features: -- Create instant or scheduled meetings -- Create webinars (requires webinar add-on license) -- List user's meetings or webinars -- Get meeting/webinar details -- Create meeting/webinar registrations (for registration-required events) - -Authentication is handled automatically via configured Zoom Server-to-Server OAuth credentials. -Do NOT ask users for credentials - they configure them once in the Credentials page. - -IMPORTANT: For registration, the meeting/webinar must have registration enabled when created. -NOTE: Webinar features require a Zoom Webinar add-on license.`, - Icon: "Video", - Source: ToolSourceBuiltin, - Category: "integration", - Keywords: []string{"zoom", "meeting", "webinar", "video", "conference", "call", "schedule", "registration", "broadcast"}, - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "credential_id": map[string]interface{}{ - "type": "string", - "description": "INTERNAL: Auto-injected by system. Do not set manually.", - }, - "action": map[string]interface{}{ - "type": "string", - "enum": []string{"create", "list", "get", "register", "create_webinar", "list_webinars", "get_webinar", "register_webinar"}, - "description": "Action: 'create' (meeting), 'list' (meetings), 'get' (meeting details), 'register' (meeting registrant), 'create_webinar', 'list_webinars', 'get_webinar', 'register_webinar'", - }, - "meeting_id": map[string]interface{}{ - "type": "string", - "description": "Meeting ID (required for 'get', 'register' actions)", - }, - "webinar_id": map[string]interface{}{ - "type": "string", - "description": "Webinar ID (required for 'get_webinar', 'register_webinar' actions)", - }, - "topic": map[string]interface{}{ - "type": "string", - "description": "Meeting/Webinar topic/title (required for 'create' and 'create_webinar')", - }, - "type": map[string]interface{}{ - "type": "number", - "description": "Meeting type: 1=instant, 2=scheduled (default), 3=recurring no fixed time, 8=recurring fixed time. Webinar type: 5=webinar, 6=recurring no fixed time, 9=recurring fixed time", - }, - "start_time": map[string]interface{}{ - "type": "string", - "description": "Start time in ISO 8601 format (e.g., '2024-01-15T10:00:00Z'). Required for scheduled meetings/webinars.", - }, - "duration": map[string]interface{}{ - "type": "number", - "description": "Duration in minutes (default: 60)", - }, - "timezone": map[string]interface{}{ - "type": "string", - "description": "Timezone (e.g., 'America/New_York', 'UTC'). Default: UTC", - }, - "agenda": map[string]interface{}{ - "type": "string", - "description": "Meeting/Webinar agenda/description", - }, - "password": map[string]interface{}{ - "type": "string", - "description": "Password (auto-generated if not provided)", - }, - "registration_required": map[string]interface{}{ - "type": "boolean", - "description": "Whether registration is required to join", - }, - "registrant_email": map[string]interface{}{ - "type": "string", - "description": "Email of the person to register (required for 'register' and 'register_webinar')", - }, - "registrant_first_name": map[string]interface{}{ - "type": "string", - "description": "First name of the registrant (required for 'register' and 'register_webinar')", - }, - "registrant_last_name": map[string]interface{}{ - "type": "string", - "description": "Last name of the registrant (optional)", - }, - "user_id": map[string]interface{}{ - "type": "string", - "description": "Zoom user ID or email (default: 'me' for the authenticated user)", - }, - }, - "required": []string{"action"}, - }, - Execute: executeZoomMeeting, - } -} - -func executeZoomMeeting(args map[string]interface{}) (string, error) { - // Get credential data for Zoom - credData, err := GetCredentialData(args, "zoom") - if err != nil { - return "", fmt.Errorf("failed to get Zoom credentials: %w. Please configure Zoom credentials first.", err) - } - - // Extract required credentials - accountID, _ := credData["account_id"].(string) - clientID, _ := credData["client_id"].(string) - clientSecret, _ := credData["client_secret"].(string) - - if accountID == "" || clientID == "" || clientSecret == "" { - return "", fmt.Errorf("Zoom credentials incomplete: account_id, client_id, and client_secret are required") - } - - // Get OAuth access token - accessToken, err := getZoomAccessToken(accountID, clientID, clientSecret) - if err != nil { - return "", fmt.Errorf("failed to get Zoom access token: %w", err) - } - - // Get action - action, ok := args["action"].(string) - if !ok || action == "" { - return "", fmt.Errorf("'action' is required (create, list, get, register, create_webinar, list_webinars, get_webinar, register_webinar)") - } - - // Execute based on action - switch action { - case "create": - return createZoomMeeting(accessToken, args) - case "list": - return listZoomMeetings(accessToken, args) - case "get": - return getZoomMeeting(accessToken, args) - case "register": - return registerZoomMeeting(accessToken, args) - case "create_webinar": - return createZoomWebinar(accessToken, args) - case "list_webinars": - return listZoomWebinars(accessToken, args) - case "get_webinar": - return getZoomWebinar(accessToken, args) - case "register_webinar": - return registerZoomWebinar(accessToken, args) - default: - return "", fmt.Errorf("unknown action: %s. Valid actions: create, list, get, register, create_webinar, list_webinars, get_webinar, register_webinar", action) - } -} - -// getZoomAccessToken gets an OAuth access token using Server-to-Server OAuth -func getZoomAccessToken(accountID, clientID, clientSecret string) (string, error) { - cacheKey := accountID + ":" + clientID - - // Check cache first - zoomTokenCacheMu.RLock() - cached, exists := zoomTokenCache[cacheKey] - zoomTokenCacheMu.RUnlock() - - if exists && time.Now().Before(cached.ExpiresAt) { - return cached.AccessToken, nil - } - - // Request new token - tokenURL := "https://zoom.us/oauth/token" - data := url.Values{} - data.Set("grant_type", "account_credentials") - data.Set("account_id", accountID) - - req, err := http.NewRequest("POST", tokenURL, strings.NewReader(data.Encode())) - if err != nil { - return "", err - } - - // Basic auth with client_id:client_secret - auth := base64.StdEncoding.EncodeToString([]byte(clientID + ":" + clientSecret)) - req.Header.Set("Authorization", "Basic "+auth) - req.Header.Set("Content-Type", "application/x-www-form-urlencoded") - - client := &http.Client{Timeout: 10 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("token request failed: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - - if resp.StatusCode != 200 { - return "", fmt.Errorf("Zoom OAuth failed (status %d): %s", resp.StatusCode, string(body)) - } - - var tokenResp struct { - AccessToken string `json:"access_token"` - TokenType string `json:"token_type"` - ExpiresIn int `json:"expires_in"` - } - - if err := json.Unmarshal(body, &tokenResp); err != nil { - return "", fmt.Errorf("failed to parse token response: %w", err) - } - - // Cache the token (expire 5 minutes early to be safe) - zoomTokenCacheMu.Lock() - zoomTokenCache[cacheKey] = &zoomCachedToken{ - AccessToken: tokenResp.AccessToken, - ExpiresAt: time.Now().Add(time.Duration(tokenResp.ExpiresIn-300) * time.Second), - } - zoomTokenCacheMu.Unlock() - - return tokenResp.AccessToken, nil -} - -// createZoomMeeting creates a new Zoom meeting -func createZoomMeeting(accessToken string, args map[string]interface{}) (string, error) { - topic, _ := args["topic"].(string) - if topic == "" { - return "", fmt.Errorf("'topic' is required for creating a meeting") - } - - userID := "me" - if uid, ok := args["user_id"].(string); ok && uid != "" { - userID = uid - } - - // Build meeting request - meeting := map[string]interface{}{ - "topic": topic, - "type": 2, // Default to scheduled meeting - } - - if meetingType, ok := args["type"].(float64); ok { - meeting["type"] = int(meetingType) - } - - if startTime, ok := args["start_time"].(string); ok && startTime != "" { - meeting["start_time"] = startTime - } - - if duration, ok := args["duration"].(float64); ok { - meeting["duration"] = int(duration) - } else { - meeting["duration"] = 60 // Default 60 minutes - } - - if timezone, ok := args["timezone"].(string); ok && timezone != "" { - meeting["timezone"] = timezone - } - - if agenda, ok := args["agenda"].(string); ok && agenda != "" { - meeting["agenda"] = agenda - } - - if password, ok := args["password"].(string); ok && password != "" { - meeting["password"] = password - } - - // Meeting settings - settings := map[string]interface{}{ - "join_before_host": true, - "mute_upon_entry": true, - "waiting_room": false, - } - - if regRequired, ok := args["registration_required"].(bool); ok && regRequired { - settings["approval_type"] = 0 // Auto-approve - meeting["type"] = 2 // Must be scheduled for registration - } - - meeting["settings"] = settings - - // Make API request - apiURL := fmt.Sprintf("https://api.zoom.us/v2/users/%s/meetings", userID) - jsonBody, _ := json.Marshal(meeting) - - req, err := http.NewRequest("POST", apiURL, bytes.NewBuffer(jsonBody)) - if err != nil { - return "", err - } - - req.Header.Set("Authorization", "Bearer "+accessToken) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("API request failed: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - - if resp.StatusCode < 200 || resp.StatusCode >= 300 { - return "", fmt.Errorf("Zoom API error (status %d): %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - json.Unmarshal(body, &result) - - // Format the response nicely - output := map[string]interface{}{ - "success": true, - "meeting_id": result["id"], - "topic": result["topic"], - "join_url": result["join_url"], - "start_url": result["start_url"], - "password": result["password"], - "start_time": result["start_time"], - "duration": result["duration"], - "timezone": result["timezone"], - "host_email": result["host_email"], - } - - jsonResult, _ := json.MarshalIndent(output, "", " ") - return string(jsonResult), nil -} - -// listZoomMeetings lists meetings for a user -func listZoomMeetings(accessToken string, args map[string]interface{}) (string, error) { - userID := "me" - if uid, ok := args["user_id"].(string); ok && uid != "" { - userID = uid - } - - apiURL := fmt.Sprintf("https://api.zoom.us/v2/users/%s/meetings?type=upcoming&page_size=30", userID) - - req, err := http.NewRequest("GET", apiURL, nil) - if err != nil { - return "", err - } - - req.Header.Set("Authorization", "Bearer "+accessToken) - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("API request failed: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - - if resp.StatusCode < 200 || resp.StatusCode >= 300 { - return "", fmt.Errorf("Zoom API error (status %d): %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - json.Unmarshal(body, &result) - - output := map[string]interface{}{ - "success": true, - "total_records": result["total_records"], - "meetings": result["meetings"], - } - - jsonResult, _ := json.MarshalIndent(output, "", " ") - return string(jsonResult), nil -} - -// getZoomMeeting gets details of a specific meeting -func getZoomMeeting(accessToken string, args map[string]interface{}) (string, error) { - meetingID, ok := args["meeting_id"].(string) - if !ok || meetingID == "" { - return "", fmt.Errorf("'meeting_id' is required for getting meeting details") - } - - apiURL := fmt.Sprintf("https://api.zoom.us/v2/meetings/%s", meetingID) - - req, err := http.NewRequest("GET", apiURL, nil) - if err != nil { - return "", err - } - - req.Header.Set("Authorization", "Bearer "+accessToken) - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("API request failed: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - - if resp.StatusCode < 200 || resp.StatusCode >= 300 { - return "", fmt.Errorf("Zoom API error (status %d): %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - json.Unmarshal(body, &result) - - output := map[string]interface{}{ - "success": true, - "meeting_id": result["id"], - "topic": result["topic"], - "type": result["type"], - "status": result["status"], - "start_time": result["start_time"], - "duration": result["duration"], - "timezone": result["timezone"], - "join_url": result["join_url"], - "password": result["password"], - "host_id": result["host_id"], - "host_email": result["host_email"], - "settings": result["settings"], - } - - jsonResult, _ := json.MarshalIndent(output, "", " ") - return string(jsonResult), nil -} - -// registerZoomMeeting registers a participant for a meeting -func registerZoomMeeting(accessToken string, args map[string]interface{}) (string, error) { - meetingID, ok := args["meeting_id"].(string) - if !ok || meetingID == "" { - return "", fmt.Errorf("'meeting_id' is required for registration") - } - - email, ok := args["registrant_email"].(string) - if !ok || email == "" { - return "", fmt.Errorf("'registrant_email' is required for registration") - } - - firstName, ok := args["registrant_first_name"].(string) - if !ok || firstName == "" { - return "", fmt.Errorf("'registrant_first_name' is required for registration") - } - - registrant := map[string]interface{}{ - "email": email, - "first_name": firstName, - } - - if lastName, ok := args["registrant_last_name"].(string); ok && lastName != "" { - registrant["last_name"] = lastName - } - - apiURL := fmt.Sprintf("https://api.zoom.us/v2/meetings/%s/registrants", meetingID) - jsonBody, _ := json.Marshal(registrant) - - req, err := http.NewRequest("POST", apiURL, bytes.NewBuffer(jsonBody)) - if err != nil { - return "", err - } - - req.Header.Set("Authorization", "Bearer "+accessToken) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("API request failed: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - - if resp.StatusCode < 200 || resp.StatusCode >= 300 { - return "", fmt.Errorf("Zoom API error (status %d): %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - json.Unmarshal(body, &result) - - output := map[string]interface{}{ - "success": true, - "registrant_id": result["registrant_id"], - "meeting_id": meetingID, - "email": email, - "first_name": firstName, - "join_url": result["join_url"], - "topic": result["topic"], - "start_time": result["start_time"], - } - - jsonResult, _ := json.MarshalIndent(output, "", " ") - return string(jsonResult), nil -} - -// createZoomWebinar creates a new Zoom webinar -func createZoomWebinar(accessToken string, args map[string]interface{}) (string, error) { - topic, _ := args["topic"].(string) - if topic == "" { - return "", fmt.Errorf("'topic' is required for creating a webinar") - } - - userID := "me" - if uid, ok := args["user_id"].(string); ok && uid != "" { - userID = uid - } - - // Build webinar request - webinar := map[string]interface{}{ - "topic": topic, - "type": 5, // Default to webinar - } - - if webinarType, ok := args["type"].(float64); ok { - webinar["type"] = int(webinarType) - } - - if startTime, ok := args["start_time"].(string); ok && startTime != "" { - webinar["start_time"] = startTime - } - - if duration, ok := args["duration"].(float64); ok { - webinar["duration"] = int(duration) - } else { - webinar["duration"] = 60 - } - - if timezone, ok := args["timezone"].(string); ok && timezone != "" { - webinar["timezone"] = timezone - } - - if agenda, ok := args["agenda"].(string); ok && agenda != "" { - webinar["agenda"] = agenda - } - - if password, ok := args["password"].(string); ok && password != "" { - webinar["password"] = password - } - - // Webinar settings - settings := map[string]interface{}{ - "approval_type": 0, // Auto-approve registrants - "registration_type": 1, // Register once and attend any occurrence - "attendees_and_panelists_reminder_email_notification": map[string]interface{}{ - "enable": true, - }, - } - - if regRequired, ok := args["registration_required"].(bool); ok && regRequired { - settings["approval_type"] = 0 - } - - webinar["settings"] = settings - - // Make API request - apiURL := fmt.Sprintf("https://api.zoom.us/v2/users/%s/webinars", userID) - jsonBody, _ := json.Marshal(webinar) - - req, err := http.NewRequest("POST", apiURL, bytes.NewBuffer(jsonBody)) - if err != nil { - return "", err - } - - req.Header.Set("Authorization", "Bearer "+accessToken) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("API request failed: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - - if resp.StatusCode < 200 || resp.StatusCode >= 300 { - return "", fmt.Errorf("Zoom API error (status %d): %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - json.Unmarshal(body, &result) - - output := map[string]interface{}{ - "success": true, - "webinar_id": result["id"], - "topic": result["topic"], - "join_url": result["join_url"], - "start_url": result["start_url"], - "password": result["password"], - "start_time": result["start_time"], - "duration": result["duration"], - "timezone": result["timezone"], - "host_email": result["host_email"], - "type": "webinar", - } - - jsonResult, _ := json.MarshalIndent(output, "", " ") - return string(jsonResult), nil -} - -// listZoomWebinars lists webinars for a user -func listZoomWebinars(accessToken string, args map[string]interface{}) (string, error) { - userID := "me" - if uid, ok := args["user_id"].(string); ok && uid != "" { - userID = uid - } - - apiURL := fmt.Sprintf("https://api.zoom.us/v2/users/%s/webinars?page_size=30", userID) - - req, err := http.NewRequest("GET", apiURL, nil) - if err != nil { - return "", err - } - - req.Header.Set("Authorization", "Bearer "+accessToken) - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("API request failed: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - - if resp.StatusCode < 200 || resp.StatusCode >= 300 { - return "", fmt.Errorf("Zoom API error (status %d): %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - json.Unmarshal(body, &result) - - output := map[string]interface{}{ - "success": true, - "total_records": result["total_records"], - "webinars": result["webinars"], - "type": "webinar_list", - } - - jsonResult, _ := json.MarshalIndent(output, "", " ") - return string(jsonResult), nil -} - -// getZoomWebinar gets details of a specific webinar -func getZoomWebinar(accessToken string, args map[string]interface{}) (string, error) { - webinarID, ok := args["webinar_id"].(string) - if !ok || webinarID == "" { - return "", fmt.Errorf("'webinar_id' is required for getting webinar details") - } - - apiURL := fmt.Sprintf("https://api.zoom.us/v2/webinars/%s", webinarID) - - req, err := http.NewRequest("GET", apiURL, nil) - if err != nil { - return "", err - } - - req.Header.Set("Authorization", "Bearer "+accessToken) - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("API request failed: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - - if resp.StatusCode < 200 || resp.StatusCode >= 300 { - return "", fmt.Errorf("Zoom API error (status %d): %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - json.Unmarshal(body, &result) - - output := map[string]interface{}{ - "success": true, - "webinar_id": result["id"], - "topic": result["topic"], - "type": result["type"], - "start_time": result["start_time"], - "duration": result["duration"], - "timezone": result["timezone"], - "join_url": result["join_url"], - "password": result["password"], - "host_id": result["host_id"], - "host_email": result["host_email"], - "settings": result["settings"], - "event_type": "webinar", - } - - jsonResult, _ := json.MarshalIndent(output, "", " ") - return string(jsonResult), nil -} - -// registerZoomWebinar registers a participant for a webinar -func registerZoomWebinar(accessToken string, args map[string]interface{}) (string, error) { - webinarID, ok := args["webinar_id"].(string) - if !ok || webinarID == "" { - return "", fmt.Errorf("'webinar_id' is required for webinar registration") - } - - email, ok := args["registrant_email"].(string) - if !ok || email == "" { - return "", fmt.Errorf("'registrant_email' is required for registration") - } - - firstName, ok := args["registrant_first_name"].(string) - if !ok || firstName == "" { - return "", fmt.Errorf("'registrant_first_name' is required for registration") - } - - registrant := map[string]interface{}{ - "email": email, - "first_name": firstName, - } - - if lastName, ok := args["registrant_last_name"].(string); ok && lastName != "" { - registrant["last_name"] = lastName - } - - apiURL := fmt.Sprintf("https://api.zoom.us/v2/webinars/%s/registrants", webinarID) - jsonBody, _ := json.Marshal(registrant) - - req, err := http.NewRequest("POST", apiURL, bytes.NewBuffer(jsonBody)) - if err != nil { - return "", err - } - - req.Header.Set("Authorization", "Bearer "+accessToken) - req.Header.Set("Content-Type", "application/json") - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return "", fmt.Errorf("API request failed: %w", err) - } - defer resp.Body.Close() - - body, _ := io.ReadAll(resp.Body) - - if resp.StatusCode < 200 || resp.StatusCode >= 300 { - return "", fmt.Errorf("Zoom API error (status %d): %s", resp.StatusCode, string(body)) - } - - var result map[string]interface{} - json.Unmarshal(body, &result) - - output := map[string]interface{}{ - "success": true, - "registrant_id": result["registrant_id"], - "webinar_id": webinarID, - "email": email, - "first_name": firstName, - "join_url": result["join_url"], - "topic": result["topic"], - "start_time": result["start_time"], - "type": "webinar_registration", - } - - jsonResult, _ := json.MarshalIndent(output, "", " ") - return string(jsonResult), nil -} diff --git a/backend/internal/utils/docx.go b/backend/internal/utils/docx.go deleted file mode 100644 index d2270932..00000000 --- a/backend/internal/utils/docx.go +++ /dev/null @@ -1,187 +0,0 @@ -package utils - -import ( - "archive/zip" - "bytes" - "encoding/xml" - "fmt" - "io" - "strings" -) - -const ( - // MaxDOCXPages is a soft limit for very large documents - MaxDOCXPages = 500 -) - -// DOCXMetadata contains information about a DOCX file -type DOCXMetadata struct { - PageCount int // Estimated from content - WordCount int - Text string -} - -// ValidateDOCX checks if a file is a valid DOCX by checking ZIP structure -func ValidateDOCX(data []byte) error { - reader := bytes.NewReader(data) - zipReader, err := zip.NewReader(reader, int64(len(data))) - if err != nil { - return fmt.Errorf("invalid DOCX: not a valid ZIP file: %w", err) - } - - // Check for required DOCX files - hasContentTypes := false - hasDocument := false - - for _, file := range zipReader.File { - if file.Name == "[Content_Types].xml" { - hasContentTypes = true - } - if file.Name == "word/document.xml" { - hasDocument = true - } - } - - if !hasContentTypes { - return fmt.Errorf("invalid DOCX: missing [Content_Types].xml") - } - if !hasDocument { - return fmt.Errorf("invalid DOCX: missing word/document.xml") - } - - return nil -} - -// ExtractDOCXText extracts text from a DOCX file -func ExtractDOCXText(data []byte) (*DOCXMetadata, error) { - reader := bytes.NewReader(data) - zipReader, err := zip.NewReader(reader, int64(len(data))) - if err != nil { - return nil, fmt.Errorf("failed to open DOCX: %w", err) - } - - var textBuilder strings.Builder - - // Find and read document.xml - for _, file := range zipReader.File { - if file.Name == "word/document.xml" { - rc, err := file.Open() - if err != nil { - return nil, fmt.Errorf("failed to open document.xml: %w", err) - } - defer rc.Close() - - content, err := io.ReadAll(rc) - if err != nil { - return nil, fmt.Errorf("failed to read document.xml: %w", err) - } - - // Parse XML and extract text - text := extractTextFromDOCXML(content) - textBuilder.WriteString(text) - break - } - } - - extractedText := textBuilder.String() - extractedText = cleanDocumentText(extractedText) - - // Enforce size limit - if len(extractedText) > MaxExtractedTextSize { - extractedText = extractedText[:MaxExtractedTextSize] + "\n... [Content truncated]" - } - - wordCount := countWords(extractedText) - - // Estimate page count (roughly 500 words per page) - pageCount := (wordCount / 500) + 1 - if pageCount < 1 { - pageCount = 1 - } - - return &DOCXMetadata{ - PageCount: pageCount, - WordCount: wordCount, - Text: extractedText, - }, nil -} - -// extractTextFromDOCXML parses DOCX XML and extracts text content -func extractTextFromDOCXML(xmlContent []byte) string { - var textBuilder strings.Builder - decoder := xml.NewDecoder(bytes.NewReader(xmlContent)) - - inParagraph := false - paragraphText := strings.Builder{} - - for { - token, err := decoder.Token() - if err == io.EOF { - break - } - if err != nil { - break - } - - switch t := token.(type) { - case xml.StartElement: - // Track paragraph boundaries - if t.Name.Local == "p" && t.Name.Space == "http://schemas.openxmlformats.org/wordprocessingml/2006/main" { - inParagraph = true - paragraphText.Reset() - } - case xml.EndElement: - // End of paragraph - add newline - if t.Name.Local == "p" && t.Name.Space == "http://schemas.openxmlformats.org/wordprocessingml/2006/main" { - if inParagraph && paragraphText.Len() > 0 { - textBuilder.WriteString(paragraphText.String()) - textBuilder.WriteString("\n") - } - inParagraph = false - } - case xml.CharData: - // Collect text content - text := strings.TrimSpace(string(t)) - if text != "" && inParagraph { - if paragraphText.Len() > 0 { - paragraphText.WriteString(" ") - } - paragraphText.WriteString(text) - } - } - } - - return textBuilder.String() -} - -// cleanDocumentText cleans extracted document text -func cleanDocumentText(text string) string { - // Remove null bytes - text = strings.ReplaceAll(text, "\x00", "") - - // Normalize multiple newlines to double newlines (paragraph breaks) - for strings.Contains(text, "\n\n\n") { - text = strings.ReplaceAll(text, "\n\n\n", "\n\n") - } - - // Trim - text = strings.TrimSpace(text) - - return text -} - -// GetDOCXPreview returns the first N characters of text as a preview -func GetDOCXPreview(text string, maxChars int) string { - if len(text) <= maxChars { - return text - } - - // Try to break at a word boundary - preview := text[:maxChars] - lastSpace := strings.LastIndex(preview, " ") - if lastSpace > maxChars/2 { - preview = preview[:lastSpace] - } - - return preview + "..." -} diff --git a/backend/internal/utils/image.go b/backend/internal/utils/image.go deleted file mode 100644 index 3e07d62c..00000000 --- a/backend/internal/utils/image.go +++ /dev/null @@ -1,90 +0,0 @@ -package utils - -import ( - "encoding/base64" - "fmt" - "io" - "os" - "path/filepath" -) - -// ImageUtils provides image processing utilities -type ImageUtils struct{} - -// NewImageUtils creates a new ImageUtils instance -func NewImageUtils() *ImageUtils { - return &ImageUtils{} -} - -// EncodeToBase64 reads an image file and encodes it to base64 -func (u *ImageUtils) EncodeToBase64(filePath string) (string, error) { - // Open file - file, err := os.Open(filePath) - if err != nil { - return "", fmt.Errorf("failed to open file: %w", err) - } - defer file.Close() - - // Read file contents - data, err := io.ReadAll(file) - if err != nil { - return "", fmt.Errorf("failed to read file: %w", err) - } - - // Encode to base64 - encoded := base64.StdEncoding.EncodeToString(data) - - // Get MIME type from extension - mimeType := u.GetMimeTypeFromExtension(filepath.Ext(filePath)) - - // Return data URL format - return fmt.Sprintf("data:%s;base64,%s", mimeType, encoded), nil -} - -// GetMimeTypeFromExtension returns MIME type for a file extension -func (u *ImageUtils) GetMimeTypeFromExtension(ext string) string { - switch ext { - case ".jpg", ".jpeg": - return "image/jpeg" - case ".png": - return "image/png" - case ".gif": - return "image/gif" - case ".webp": - return "image/webp" - default: - return "application/octet-stream" - } -} - -// IsValidImageExtension checks if the file extension is a valid image type -func (u *ImageUtils) IsValidImageExtension(ext string) bool { - validExts := map[string]bool{ - ".jpg": true, - ".jpeg": true, - ".png": true, - ".gif": true, - ".webp": true, - } - return validExts[ext] -} - -// GetFileSize returns the size of a file in bytes -func (u *ImageUtils) GetFileSize(filePath string) (int64, error) { - info, err := os.Stat(filePath) - if err != nil { - return 0, err - } - return info.Size(), nil -} - -// DeleteFile removes a file from the filesystem -func (u *ImageUtils) DeleteFile(filePath string) error { - return os.Remove(filePath) -} - -// FileExists checks if a file exists -func (u *ImageUtils) FileExists(filePath string) bool { - _, err := os.Stat(filePath) - return err == nil -} diff --git a/backend/internal/utils/pdf.go b/backend/internal/utils/pdf.go deleted file mode 100644 index b9e7ae3b..00000000 --- a/backend/internal/utils/pdf.go +++ /dev/null @@ -1,184 +0,0 @@ -package utils - -import ( - "bytes" - "fmt" - "strings" - "unicode" - - "github.com/ledongthuc/pdf" -) - -const ( - // MaxPDFPages limits the number of pages to process - MaxPDFPages = 100 - - // MaxExtractedTextSize limits the extracted text size (1MB) - MaxExtractedTextSize = 1024 * 1024 -) - -// PDFMetadata contains information about a PDF -type PDFMetadata struct { - PageCount int - WordCount int - Text string -} - -// ValidatePDF checks if a file is a valid PDF by attempting to open it -func ValidatePDF(data []byte) error { - reader := bytes.NewReader(data) - _, err := pdf.NewReader(reader, int64(len(data))) - if err != nil { - return fmt.Errorf("invalid PDF: %w", err) - } - return nil -} - -// ExtractPDFText extracts text from a PDF file (provided as byte data) -func ExtractPDFText(data []byte) (*PDFMetadata, error) { - // Create reader from byte data - reader := bytes.NewReader(data) - pdfReader, err := pdf.NewReader(reader, int64(len(data))) - if err != nil { - return nil, fmt.Errorf("failed to open PDF: %w", err) - } - - // Check page count - totalPages := pdfReader.NumPage() - if totalPages == 0 { - return nil, fmt.Errorf("PDF has no pages") - } - - // Limit page count for security - if totalPages > MaxPDFPages { - return nil, fmt.Errorf("PDF has too many pages (%d), max allowed is %d", totalPages, MaxPDFPages) - } - - var textBuilder strings.Builder - wordCount := 0 - - // Extract text from each page - for pageNum := 1; pageNum <= totalPages; pageNum++ { - page := pdfReader.Page(pageNum) - if page.V.IsNull() { - continue - } - - // Get text content - text, err := page.GetPlainText(nil) - if err != nil { - // Skip pages with extraction errors, don't fail completely - continue - } - - // Clean and add text - cleaned := cleanPDFText(text) - if cleaned != "" { - textBuilder.WriteString(fmt.Sprintf("\n--- Page %d ---\n", pageNum)) - textBuilder.WriteString(cleaned) - textBuilder.WriteString("\n") - - // Count words - wordCount += countWords(cleaned) - } - - // Check size limit - if textBuilder.Len() > MaxExtractedTextSize { - textBuilder.WriteString("\n... [Content truncated - size limit reached]") - break - } - } - - extractedText := textBuilder.String() - - // Final size check - if len(extractedText) > MaxExtractedTextSize { - extractedText = extractedText[:MaxExtractedTextSize] + "\n... [Content truncated]" - } - - return &PDFMetadata{ - PageCount: totalPages, - WordCount: wordCount, - Text: extractedText, - }, nil -} - -// cleanPDFText cleans extracted PDF text -func cleanPDFText(text string) string { - // Remove null bytes - text = strings.ReplaceAll(text, "\x00", "") - - // Normalize whitespace - text = normalizeWhitespace(text) - - // Trim - text = strings.TrimSpace(text) - - return text -} - -// normalizeWhitespace normalizes whitespace in text -func normalizeWhitespace(text string) string { - var result strings.Builder - lastWasSpace := false - - for _, r := range text { - if unicode.IsSpace(r) { - if !lastWasSpace { - // Preserve newlines, convert other spaces to single space - if r == '\n' { - result.WriteRune('\n') - lastWasSpace = false - } else { - result.WriteRune(' ') - lastWasSpace = true - } - } - } else { - result.WriteRune(r) - lastWasSpace = false - } - } - - return result.String() -} - -// countWords counts the number of words in text -func countWords(text string) int { - count := 0 - inWord := false - - for _, r := range text { - if unicode.IsSpace(r) || unicode.IsPunct(r) { - if inWord { - count++ - inWord = false - } - } else { - inWord = true - } - } - - // Count last word - if inWord { - count++ - } - - return count -} - -// GetPDFPreview returns the first N characters of text as a preview -func GetPDFPreview(text string, maxChars int) string { - if len(text) <= maxChars { - return text - } - - // Try to break at a word boundary - preview := text[:maxChars] - lastSpace := strings.LastIndex(preview, " ") - if lastSpace > maxChars/2 { - preview = preview[:lastSpace] - } - - return preview + "..." -} diff --git a/backend/internal/utils/pptx.go b/backend/internal/utils/pptx.go deleted file mode 100644 index 5c2baa8f..00000000 --- a/backend/internal/utils/pptx.go +++ /dev/null @@ -1,208 +0,0 @@ -package utils - -import ( - "archive/zip" - "bytes" - "encoding/xml" - "fmt" - "io" - "path" - "sort" - "strconv" - "strings" -) - -const ( - // MaxPPTXSlides limits the number of slides to process - MaxPPTXSlides = 200 -) - -// PPTXMetadata contains information about a PPTX file -type PPTXMetadata struct { - SlideCount int - WordCount int - Text string -} - -// ValidatePPTX checks if a file is a valid PPTX by checking ZIP structure -func ValidatePPTX(data []byte) error { - reader := bytes.NewReader(data) - zipReader, err := zip.NewReader(reader, int64(len(data))) - if err != nil { - return fmt.Errorf("invalid PPTX: not a valid ZIP file: %w", err) - } - - // Check for required PPTX files - hasContentTypes := false - hasSlides := false - - for _, file := range zipReader.File { - if file.Name == "[Content_Types].xml" { - hasContentTypes = true - } - if strings.HasPrefix(file.Name, "ppt/slides/slide") && strings.HasSuffix(file.Name, ".xml") { - hasSlides = true - } - } - - if !hasContentTypes { - return fmt.Errorf("invalid PPTX: missing [Content_Types].xml") - } - if !hasSlides { - return fmt.Errorf("invalid PPTX: no slides found") - } - - return nil -} - -// ExtractPPTXText extracts text from a PPTX file -func ExtractPPTXText(data []byte) (*PPTXMetadata, error) { - reader := bytes.NewReader(data) - zipReader, err := zip.NewReader(reader, int64(len(data))) - if err != nil { - return nil, fmt.Errorf("failed to open PPTX: %w", err) - } - - // Collect slide files and sort them by slide number - type slideFile struct { - num int - file *zip.File - } - var slides []slideFile - - for _, file := range zipReader.File { - if strings.HasPrefix(file.Name, "ppt/slides/slide") && strings.HasSuffix(file.Name, ".xml") { - // Extract slide number from filename (e.g., "ppt/slides/slide1.xml" -> 1) - baseName := path.Base(file.Name) - numStr := strings.TrimPrefix(baseName, "slide") - numStr = strings.TrimSuffix(numStr, ".xml") - num, err := strconv.Atoi(numStr) - if err != nil { - continue - } - slides = append(slides, slideFile{num: num, file: file}) - } - } - - // Sort slides by number - sort.Slice(slides, func(i, j int) bool { - return slides[i].num < slides[j].num - }) - - // Limit slides for security - if len(slides) > MaxPPTXSlides { - slides = slides[:MaxPPTXSlides] - } - - var textBuilder strings.Builder - slideCount := len(slides) - - // Extract text from each slide - for _, slide := range slides { - rc, err := slide.file.Open() - if err != nil { - continue - } - - content, err := io.ReadAll(rc) - rc.Close() - if err != nil { - continue - } - - // Parse XML and extract text - slideText := extractTextFromPPTXML(content) - if slideText != "" { - textBuilder.WriteString(fmt.Sprintf("\n--- Slide %d ---\n", slide.num)) - textBuilder.WriteString(slideText) - textBuilder.WriteString("\n") - } - - // Check size limit - if textBuilder.Len() > MaxExtractedTextSize { - textBuilder.WriteString("\n... [Content truncated - size limit reached]") - break - } - } - - extractedText := textBuilder.String() - extractedText = cleanDocumentText(extractedText) - - // Final size check - if len(extractedText) > MaxExtractedTextSize { - extractedText = extractedText[:MaxExtractedTextSize] + "\n... [Content truncated]" - } - - wordCount := countWords(extractedText) - - return &PPTXMetadata{ - SlideCount: slideCount, - WordCount: wordCount, - Text: extractedText, - }, nil -} - -// extractTextFromPPTXML parses PPTX slide XML and extracts text content -func extractTextFromPPTXML(xmlContent []byte) string { - var textBuilder strings.Builder - decoder := xml.NewDecoder(bytes.NewReader(xmlContent)) - - // Track text runs within paragraphs - inTextParagraph := false - paragraphText := strings.Builder{} - - for { - token, err := decoder.Token() - if err == io.EOF { - break - } - if err != nil { - break - } - - switch t := token.(type) { - case xml.StartElement: - // Track paragraph boundaries (a:p is DrawingML paragraph) - if t.Name.Local == "p" && strings.Contains(t.Name.Space, "drawingml") { - inTextParagraph = true - paragraphText.Reset() - } - case xml.EndElement: - // End of paragraph - add newline - if t.Name.Local == "p" && strings.Contains(t.Name.Space, "drawingml") { - if inTextParagraph && paragraphText.Len() > 0 { - textBuilder.WriteString(paragraphText.String()) - textBuilder.WriteString("\n") - } - inTextParagraph = false - } - case xml.CharData: - // Collect text content (a:t is DrawingML text) - text := strings.TrimSpace(string(t)) - if text != "" && inTextParagraph { - if paragraphText.Len() > 0 { - paragraphText.WriteString(" ") - } - paragraphText.WriteString(text) - } - } - } - - return textBuilder.String() -} - -// GetPPTXPreview returns the first N characters of text as a preview -func GetPPTXPreview(text string, maxChars int) string { - if len(text) <= maxChars { - return text - } - - // Try to break at a word boundary - preview := text[:maxChars] - lastSpace := strings.LastIndex(preview, " ") - if lastSpace > maxChars/2 { - preview = preview[:lastSpace] - } - - return preview + "..." -} diff --git a/backend/internal/vision/service.go b/backend/internal/vision/service.go deleted file mode 100644 index d38f5711..00000000 --- a/backend/internal/vision/service.go +++ /dev/null @@ -1,213 +0,0 @@ -package vision - -import ( - "bytes" - "encoding/base64" - "encoding/json" - "fmt" - "io" - "log" - "net/http" - "strings" - "sync" - "time" -) - -// Provider represents a minimal provider interface for vision -type Provider struct { - ID int - Name string - BaseURL string - APIKey string - Enabled bool -} - -// ModelAlias represents a model alias with vision support info -type ModelAlias struct { - DisplayName string - ActualModel string - SupportsVision *bool -} - -// ProviderGetter is a function type to get provider by ID -type ProviderGetter func(id int) (*Provider, error) - -// VisionModelFinder is a function type to find vision-capable models -type VisionModelFinder func() (providerID int, modelName string, err error) - -// Service handles image analysis using vision-capable models -type Service struct { - httpClient *http.Client - providerGetter ProviderGetter - visionModelFinder VisionModelFinder - mu sync.RWMutex -} - -var ( - instance *Service - once sync.Once -) - -// GetService returns the singleton vision service -// Note: Must call InitService first to set up dependencies -func GetService() *Service { - return instance -} - -// InitService initializes the vision service with dependencies -func InitService(providerGetter ProviderGetter, visionModelFinder VisionModelFinder) *Service { - once.Do(func() { - instance = &Service{ - httpClient: &http.Client{ - Timeout: 60 * time.Second, - }, - providerGetter: providerGetter, - visionModelFinder: visionModelFinder, - } - }) - return instance -} - -// DescribeImageRequest contains parameters for image description -type DescribeImageRequest struct { - ImageData []byte - MimeType string - Question string // Optional question about the image - Detail string // "brief" or "detailed" -} - -// DescribeImageResponse contains the result of image description -type DescribeImageResponse struct { - Description string `json:"description"` - Model string `json:"model"` - Provider string `json:"provider"` -} - -// DescribeImage analyzes an image and returns a text description -func (s *Service) DescribeImage(req *DescribeImageRequest) (*DescribeImageResponse, error) { - s.mu.RLock() - defer s.mu.RUnlock() - - if s.visionModelFinder == nil || s.providerGetter == nil { - return nil, fmt.Errorf("vision service not properly initialized") - } - - log.Printf("🖼️ [VISION] Analyzing image (%d bytes, %s)", len(req.ImageData), req.MimeType) - - // Convert to base64 - base64Image := base64.StdEncoding.EncodeToString(req.ImageData) - dataURL := fmt.Sprintf("data:%s;base64,%s", req.MimeType, base64Image) - - // Find a vision-capable model - providerID, modelName, err := s.visionModelFinder() - if err != nil { - return nil, fmt.Errorf("no vision-capable model available: %w", err) - } - - provider, err := s.providerGetter(providerID) - if err != nil { - return nil, fmt.Errorf("failed to get provider: %w", err) - } - - // Build the prompt - prompt := "Describe this image in detail." - if req.Question != "" { - prompt = req.Question - } else if req.Detail == "brief" { - prompt = "Briefly describe this image in 1-2 sentences." - } - - // Build the API request - messages := []map[string]interface{}{ - { - "role": "user", - "content": []map[string]interface{}{ - { - "type": "text", - "text": prompt, - }, - { - "type": "image_url", - "image_url": map[string]interface{}{ - "url": dataURL, - "detail": "auto", - }, - }, - }, - }, - } - - // Detect if using OpenAI - they require max_completion_tokens instead of max_tokens - isOpenAI := strings.Contains(strings.ToLower(provider.BaseURL), "openai.com") - - requestBody := map[string]interface{}{ - "model": modelName, - "messages": messages, - } - - // Use correct token limit parameter based on provider - if isOpenAI { - requestBody["max_completion_tokens"] = 1000 - } else { - requestBody["max_tokens"] = 1000 - } - - requestJSON, err := json.Marshal(requestBody) - if err != nil { - return nil, fmt.Errorf("failed to marshal request: %w", err) - } - - // Make the API call - apiURL := fmt.Sprintf("%s/chat/completions", strings.TrimSuffix(provider.BaseURL, "/")) - httpReq, err := http.NewRequest("POST", apiURL, bytes.NewReader(requestJSON)) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - httpReq.Header.Set("Content-Type", "application/json") - httpReq.Header.Set("Authorization", fmt.Sprintf("Bearer %s", provider.APIKey)) - - log.Printf("🔄 [VISION] Calling %s with model %s", provider.Name, modelName) - - resp, err := s.httpClient.Do(httpReq) - if err != nil { - return nil, fmt.Errorf("API request failed: %w", err) - } - defer resp.Body.Close() - - body, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - if resp.StatusCode != http.StatusOK { - log.Printf("❌ [VISION] API error: %d - %s", resp.StatusCode, string(body)) - return nil, fmt.Errorf("API error: %d", resp.StatusCode) - } - - // Parse response - var apiResp struct { - Choices []struct { - Message struct { - Content string `json:"content"` - } `json:"message"` - } `json:"choices"` - } - - if err := json.Unmarshal(body, &apiResp); err != nil { - return nil, fmt.Errorf("failed to parse response: %w", err) - } - - if len(apiResp.Choices) == 0 { - return nil, fmt.Errorf("no response from vision model") - } - - description := apiResp.Choices[0].Message.Content - log.Printf("✅ [VISION] Image described successfully (%d chars)", len(description)) - - return &DescribeImageResponse{ - Description: description, - Model: modelName, - Provider: provider.Name, - }, nil -} diff --git a/backend/internal/vision/service_test.go b/backend/internal/vision/service_test.go deleted file mode 100644 index 46469703..00000000 --- a/backend/internal/vision/service_test.go +++ /dev/null @@ -1,183 +0,0 @@ -package vision - -import ( - "testing" -) - -// TestProviderStructure tests provider structure -func TestProviderStructure(t *testing.T) { - provider := &Provider{ - ID: 1, - Name: "openai", - BaseURL: "https://api.openai.com/v1", - APIKey: "test-key", - Enabled: true, - } - - if provider.Name != "openai" { - t.Errorf("Expected provider name 'openai', got %s", provider.Name) - } - if !provider.Enabled { - t.Error("Provider should be enabled") - } - if provider.BaseURL == "" { - t.Error("BaseURL should not be empty") - } -} - -// TestDescribeImageRequestStructure tests request structure -func TestDescribeImageRequestStructure(t *testing.T) { - req := &DescribeImageRequest{ - ImageData: []byte{0xFF, 0xD8, 0xFF}, // JPEG header bytes - MimeType: "image/jpeg", - Question: "What is in this image?", - Detail: "detailed", - } - - if len(req.ImageData) == 0 { - t.Error("ImageData should be set") - } - if req.MimeType == "" { - t.Error("MimeType should be set") - } - if req.Detail != "detailed" && req.Detail != "brief" && req.Detail != "auto" && req.Detail != "" { - t.Errorf("Invalid detail level: %s", req.Detail) - } -} - -// TestDescribeImageResponseStructure tests response structure -func TestDescribeImageResponseStructure(t *testing.T) { - resp := &DescribeImageResponse{ - Description: "A beautiful sunset over the ocean", - Model: "gpt-4o", - Provider: "openai", - } - - if resp.Description == "" { - t.Error("Description should not be empty") - } - if resp.Model == "" { - t.Error("Model should not be empty") - } -} - -// TestGetServiceSingleton verifies singleton pattern -func TestGetServiceSingleton(t *testing.T) { - // GetService should return nil if not initialized - svc := GetService() - // Note: This may return non-nil if InitService was called elsewhere - // The test mainly verifies no panic occurs - _ = svc -} - -// TestDetailLevelValidation tests valid detail levels -func TestDetailLevelValidation(t *testing.T) { - validLevels := []string{"brief", "detailed", "auto", ""} - - for _, level := range validLevels { - req := &DescribeImageRequest{ - ImageData: []byte{0x89, 0x50, 0x4E, 0x47}, // PNG header - MimeType: "image/png", - Detail: level, - } - // Simply verify the struct accepts all levels - if req.Detail != level { - t.Errorf("Detail level %q should be accepted", level) - } - } -} - -// TestInitServiceWithNilCallbacks verifies graceful handling -func TestInitServiceWithNilCallbacks(t *testing.T) { - // This should not panic - defer func() { - if r := recover(); r != nil { - t.Errorf("InitService should not panic with nil callbacks: %v", r) - } - }() - - // Note: We can't actually call InitService here without affecting global state - // This test documents the expected behavior -} - -// TestProviderCallbackTypes verifies callback type signatures -func TestProviderCallbackTypes(t *testing.T) { - // Test ProviderGetter signature - var providerGetter ProviderGetter = func(id int) (*Provider, error) { - return &Provider{ - ID: id, - Name: "test", - BaseURL: "https://test.com", - APIKey: "key", - Enabled: true, - }, nil - } - - provider, err := providerGetter(1) - if err != nil { - t.Errorf("ProviderGetter should not error: %v", err) - } - if provider.ID != 1 { - t.Errorf("Expected provider ID 1, got %d", provider.ID) - } - - // Test VisionModelFinder signature - var modelFinder VisionModelFinder = func() (int, string, error) { - return 1, "gpt-4o", nil - } - - providerID, modelName, err := modelFinder() - if err != nil { - t.Errorf("VisionModelFinder should not error: %v", err) - } - if providerID != 1 { - t.Errorf("Expected provider ID 1, got %d", providerID) - } - if modelName != "gpt-4o" { - t.Errorf("Expected model 'gpt-4o', got %s", modelName) - } -} - -// TestMimeTypeValidation tests various MIME types -func TestMimeTypeValidation(t *testing.T) { - mimeTypes := []string{ - "image/jpeg", - "image/png", - "image/gif", - "image/webp", - } - - for _, mimeType := range mimeTypes { - req := &DescribeImageRequest{ - ImageData: []byte{1, 2, 3}, - MimeType: mimeType, - } - if req.MimeType != mimeType { - t.Errorf("MimeType should be %s, got %s", mimeType, req.MimeType) - } - } -} - -// Benchmark tests -func BenchmarkProviderCreation(b *testing.B) { - for i := 0; i < b.N; i++ { - _ = &Provider{ - ID: 1, - Name: "openai", - BaseURL: "https://api.openai.com/v1", - APIKey: "test-key", - Enabled: true, - } - } -} - -func BenchmarkRequestCreation(b *testing.B) { - for i := 0; i < b.N; i++ { - _ = &DescribeImageRequest{ - ImageData: []byte{0xFF, 0xD8, 0xFF}, - MimeType: "image/jpeg", - Question: "What is in this image?", - Detail: "detailed", - } - } -} diff --git a/backend/mcp-bridge/bin/mcp-client-darwin-amd64 b/backend/mcp-bridge/bin/mcp-client-darwin-amd64 deleted file mode 100755 index 32f4e253..00000000 Binary files a/backend/mcp-bridge/bin/mcp-client-darwin-amd64 and /dev/null differ diff --git a/backend/mcp-bridge/bin/mcp-client-darwin-arm64 b/backend/mcp-bridge/bin/mcp-client-darwin-arm64 deleted file mode 100755 index 5680e6e7..00000000 Binary files a/backend/mcp-bridge/bin/mcp-client-darwin-arm64 and /dev/null differ diff --git a/backend/mcp-bridge/bin/mcp-client-linux-amd64 b/backend/mcp-bridge/bin/mcp-client-linux-amd64 deleted file mode 100755 index b7d46238..00000000 Binary files a/backend/mcp-bridge/bin/mcp-client-linux-amd64 and /dev/null differ diff --git a/backend/mcp-bridge/build.bat b/backend/mcp-bridge/build.bat deleted file mode 100644 index 9b61eeb1..00000000 --- a/backend/mcp-bridge/build.bat +++ /dev/null @@ -1,52 +0,0 @@ -@echo off -REM Cross-platform build script for MCP Client - -echo Building ClaraVerse MCP Client... -echo. - -REM Windows -echo [1/3] Building for Windows (amd64)... -set GOOS=windows -set GOARCH=amd64 -go build -o bin/mcp-client-windows-amd64.exe ./cmd/mcp-client -if %ERRORLEVEL% NEQ 0 ( - echo Build failed for Windows - exit /b 1 -) -echo ✓ Windows build complete - -REM Linux -echo [2/3] Building for Linux (amd64)... -set GOOS=linux -set GOARCH=amd64 -go build -o bin/mcp-client-linux-amd64 ./cmd/mcp-client -if %ERRORLEVEL% NEQ 0 ( - echo Build failed for Linux - exit /b 1 -) -echo ✓ Linux build complete - -REM macOS -echo [3/3] Building for macOS (amd64)... -set GOOS=darwin -set GOARCH=amd64 -go build -o bin/mcp-client-darwin-amd64 ./cmd/mcp-client -if %ERRORLEVEL% NEQ 0 ( - echo Build failed for macOS - exit /b 1 -) -echo ✓ macOS build complete - -echo. -echo ======================================== -echo ✓ All builds completed successfully! -echo ======================================== -echo. -echo Binaries created in bin/: -dir /B bin\mcp-client-* -echo. -echo To run: -echo Windows: bin\mcp-client-windows-amd64.exe -echo Linux: bin/mcp-client-linux-amd64 -echo macOS: bin/mcp-client-darwin-amd64 -echo. diff --git a/backend/mcp-bridge/build.sh b/backend/mcp-bridge/build.sh deleted file mode 100755 index a7394eb9..00000000 --- a/backend/mcp-bridge/build.sh +++ /dev/null @@ -1,60 +0,0 @@ -#!/bin/bash - -# Cross-platform build script for MCP Client - -echo "Building ClaraVerse MCP Client..." -echo "" - -# Create bin directory -mkdir -p bin - -# Windows -echo "[1/3] Building for Windows (amd64)..." -GOOS=windows GOARCH=amd64 go build -o bin/mcp-client-windows-amd64.exe ./cmd/mcp-client -if [ $? -ne 0 ]; then - echo "❌ Build failed for Windows" - exit 1 -fi -echo "✓ Windows build complete" - -# Linux -echo "[2/3] Building for Linux (amd64)..." -GOOS=linux GOARCH=amd64 go build -o bin/mcp-client-linux-amd64 ./cmd/mcp-client -if [ $? -ne 0 ]; then - echo "❌ Build failed for Linux" - exit 1 -fi -echo "✓ Linux build complete" - -# macOS Intel -echo "[3/3] Building for macOS (amd64)..." -GOOS=darwin GOARCH=amd64 go build -o bin/mcp-client-darwin-amd64 ./cmd/mcp-client -if [ $? -ne 0 ]; then - echo "❌ Build failed for macOS" - exit 1 -fi -echo "✓ macOS build complete" - -# macOS Apple Silicon (bonus) -echo "[4/4] Building for macOS (arm64)..." -GOOS=darwin GOARCH=arm64 go build -o bin/mcp-client-darwin-arm64 ./cmd/mcp-client -if [ $? -ne 0 ]; then - echo "⚠️ Build failed for macOS ARM (non-critical)" -else - echo "✓ macOS ARM build complete" -fi - -echo "" -echo "========================================" -echo "✓ All builds completed successfully!" -echo "========================================" -echo "" -echo "Binaries created in bin/:" -ls -lh bin/mcp-client-* -echo "" -echo "To run:" -echo " Windows: bin/mcp-client-windows-amd64.exe" -echo " Linux: bin/mcp-client-linux-amd64" -echo " macOS: bin/mcp-client-darwin-amd64" -echo " macOS (M1/M2): bin/mcp-client-darwin-arm64" -echo "" diff --git a/backend/mcp-bridge/cmd/mcp-client/main.go b/backend/mcp-bridge/cmd/mcp-client/main.go deleted file mode 100644 index c40ef415..00000000 --- a/backend/mcp-bridge/cmd/mcp-client/main.go +++ /dev/null @@ -1,43 +0,0 @@ -package main - -import ( - "fmt" - "os" - - "github.com/claraverse/mcp-client/internal/commands" - "github.com/spf13/cobra" -) - -var ( - version = "1.0.0" - verbose bool -) - -var rootCmd = &cobra.Command{ - Use: "mcp-client", - Short: "ClaraVerse MCP Client - Connect local tools to cloud chat", - Long: `ClaraVerse MCP Client allows you to connect local MCP (Model Context Protocol) -servers to your ClaraVerse cloud chat, giving the AI access to your local tools, -filesystems, databases, and custom integrations.`, - Version: version, -} - -func init() { - // Global flags - rootCmd.PersistentFlags().BoolVarP(&verbose, "verbose", "v", false, "Enable verbose logging") - - // Add all commands - rootCmd.AddCommand(commands.LoginCmd) - rootCmd.AddCommand(commands.StartCmd) - rootCmd.AddCommand(commands.AddCmd) - rootCmd.AddCommand(commands.ListCmd) - rootCmd.AddCommand(commands.RemoveCmd) - rootCmd.AddCommand(commands.StatusCmd) -} - -func main() { - if err := rootCmd.Execute(); err != nil { - fmt.Fprintf(os.Stderr, "Error: %v\n", err) - os.Exit(1) - } -} diff --git a/backend/mcp-bridge/go.mod b/backend/mcp-bridge/go.mod deleted file mode 100644 index 07eafa99..00000000 --- a/backend/mcp-bridge/go.mod +++ /dev/null @@ -1,28 +0,0 @@ -module github.com/claraverse/mcp-client - -go 1.25.5 - -require ( - github.com/google/uuid v1.6.0 - github.com/gorilla/websocket v1.5.3 - github.com/spf13/cobra v1.10.1 - github.com/spf13/viper v1.21.0 - golang.org/x/term v0.37.0 - gopkg.in/yaml.v3 v3.0.1 -) - -require ( - github.com/fsnotify/fsnotify v1.9.0 // indirect - github.com/go-viper/mapstructure/v2 v2.4.0 // indirect - github.com/inconshreveable/mousetrap v1.1.0 // indirect - github.com/pelletier/go-toml/v2 v2.2.4 // indirect - github.com/sagikazarmark/locafero v0.11.0 // indirect - github.com/sourcegraph/conc v0.3.1-0.20240121214520-5f936abd7ae8 // indirect - github.com/spf13/afero v1.15.0 // indirect - github.com/spf13/cast v1.10.0 // indirect - github.com/spf13/pflag v1.0.10 // indirect - github.com/subosito/gotenv v1.6.0 // indirect - go.yaml.in/yaml/v3 v3.0.4 // indirect - golang.org/x/sys v0.38.0 // indirect - golang.org/x/text v0.28.0 // indirect -) diff --git a/backend/mcp-bridge/go.sum b/backend/mcp-bridge/go.sum deleted file mode 100644 index d83e3159..00000000 --- a/backend/mcp-bridge/go.sum +++ /dev/null @@ -1,60 +0,0 @@ -github.com/cpuguy83/go-md2man/v2 v2.0.6/go.mod h1:oOW0eioCTA6cOiMLiUPZOpcVxMig6NIQQ7OS05n1F4g= -github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= -github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= -github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8= -github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0= -github.com/fsnotify/fsnotify v1.9.0 h1:2Ml+OJNzbYCTzsxtv8vKSFD9PbJjmhYF14k/jKC7S9k= -github.com/fsnotify/fsnotify v1.9.0/go.mod h1:8jBTzvmWwFyi3Pb8djgCCO5IBqzKJ/Jwo8TRcHyHii0= -github.com/go-viper/mapstructure/v2 v2.4.0 h1:EBsztssimR/CONLSZZ04E8qAkxNYq4Qp9LvH92wZUgs= -github.com/go-viper/mapstructure/v2 v2.4.0/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM= -github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= -github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= -github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= -github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= -github.com/gorilla/websocket v1.5.3 h1:saDtZ6Pbx/0u+bgYQ3q96pZgCzfhKXGPqt7kZ72aNNg= -github.com/gorilla/websocket v1.5.3/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= -github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8= -github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw= -github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= -github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= -github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= -github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= -github.com/pelletier/go-toml/v2 v2.2.4 h1:mye9XuhQ6gvn5h28+VilKrrPoQVanw5PMw/TB0t5Ec4= -github.com/pelletier/go-toml/v2 v2.2.4/go.mod h1:2gIqNv+qfxSVS7cM2xJQKtLSTLUE9V8t9Stt+h56mCY= -github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= -github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= -github.com/rogpeppe/go-internal v1.9.0 h1:73kH8U+JUqXU8lRuOHeVHaa/SZPifC7BkcraZVejAe8= -github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs= -github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= -github.com/sagikazarmark/locafero v0.11.0 h1:1iurJgmM9G3PA/I+wWYIOw/5SyBtxapeHDcg+AAIFXc= -github.com/sagikazarmark/locafero v0.11.0/go.mod h1:nVIGvgyzw595SUSUE6tvCp3YYTeHs15MvlmU87WwIik= -github.com/sourcegraph/conc v0.3.1-0.20240121214520-5f936abd7ae8 h1:+jumHNA0Wrelhe64i8F6HNlS8pkoyMv5sreGx2Ry5Rw= -github.com/sourcegraph/conc v0.3.1-0.20240121214520-5f936abd7ae8/go.mod h1:3n1Cwaq1E1/1lhQhtRK2ts/ZwZEhjcQeJQ1RuC6Q/8U= -github.com/spf13/afero v1.15.0 h1:b/YBCLWAJdFWJTN9cLhiXXcD7mzKn9Dm86dNnfyQw1I= -github.com/spf13/afero v1.15.0/go.mod h1:NC2ByUVxtQs4b3sIUphxK0NioZnmxgyCrfzeuq8lxMg= -github.com/spf13/cast v1.10.0 h1:h2x0u2shc1QuLHfxi+cTJvs30+ZAHOGRic8uyGTDWxY= -github.com/spf13/cast v1.10.0/go.mod h1:jNfB8QC9IA6ZuY2ZjDp0KtFO2LZZlg4S/7bzP6qqeHo= -github.com/spf13/cobra v1.10.1 h1:lJeBwCfmrnXthfAupyUTzJ/J4Nc1RsHC/mSRU2dll/s= -github.com/spf13/cobra v1.10.1/go.mod h1:7SmJGaTHFVBY0jW4NXGluQoLvhqFQM+6XSKD+P4XaB0= -github.com/spf13/pflag v1.0.9/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= -github.com/spf13/pflag v1.0.10 h1:4EBh2KAYBwaONj6b2Ye1GiHfwjqyROoF4RwYO+vPwFk= -github.com/spf13/pflag v1.0.10/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= -github.com/spf13/viper v1.21.0 h1:x5S+0EU27Lbphp4UKm1C+1oQO+rKx36vfCoaVebLFSU= -github.com/spf13/viper v1.21.0/go.mod h1:P0lhsswPGWD/1lZJ9ny3fYnVqxiegrlNrEmgLjbTCAY= -github.com/stretchr/testify v1.11.1 h1:7s2iGBzp5EwR7/aIZr8ao5+dra3wiQyKjjFuvgVKu7U= -github.com/stretchr/testify v1.11.1/go.mod h1:wZwfW3scLgRK+23gO65QZefKpKQRnfz6sD981Nm4B6U= -github.com/subosito/gotenv v1.6.0 h1:9NlTDc1FTs4qu0DDq7AEtTPNw6SVm7uBMsUCUjABIf8= -github.com/subosito/gotenv v1.6.0/go.mod h1:Dk4QP5c2W3ibzajGcXpNraDfq2IrhjMIvMSWPKKo0FU= -go.yaml.in/yaml/v3 v3.0.4 h1:tfq32ie2Jv2UxXFdLJdh3jXuOzWiL1fo0bu/FbuKpbc= -go.yaml.in/yaml/v3 v3.0.4/go.mod h1:DhzuOOF2ATzADvBadXxruRBLzYTpT36CKvDb3+aBEFg= -golang.org/x/sys v0.38.0 h1:3yZWxaJjBmCWXqhN1qh02AkOnCQ1poK6oF+a7xWL6Gc= -golang.org/x/sys v0.38.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks= -golang.org/x/term v0.37.0 h1:8EGAD0qCmHYZg6J17DvsMy9/wJ7/D/4pV/wfnld5lTU= -golang.org/x/term v0.37.0/go.mod h1:5pB4lxRNYYVZuTLmy8oR2BH8dflOR+IbTYFD8fi3254= -golang.org/x/text v0.28.0 h1:rhazDwis8INMIwQ4tpjLDzUhx6RlXqZNPEM0huQojng= -golang.org/x/text v0.28.0/go.mod h1:U8nCwOR8jO/marOQ0QbDiOngZVEBB7MAiitBuMjXiNU= -gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= -gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo= -gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= -gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= -gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= diff --git a/backend/mcp-bridge/internal/bridge/websocket.go b/backend/mcp-bridge/internal/bridge/websocket.go deleted file mode 100644 index 1bc6b3eb..00000000 --- a/backend/mcp-bridge/internal/bridge/websocket.go +++ /dev/null @@ -1,303 +0,0 @@ -package bridge - -import ( - "fmt" - "log" - "math" - "sync" - "time" - - "github.com/gorilla/websocket" -) - -// Message represents a WebSocket message -type Message struct { - Type string `json:"type"` - Payload map[string]interface{} `json:"payload"` -} - -// ToolCall represents a tool execution request from backend -type ToolCall struct { - CallID string `json:"call_id"` - ToolName string `json:"tool_name"` - Arguments map[string]interface{} `json:"arguments"` - Timeout int `json:"timeout"` -} - -// Bridge manages the WebSocket connection to the backend -type Bridge struct { - backendURL string - authToken string - conn *websocket.Conn - writeChan chan Message - stopChan chan struct{} - reconnectDelay time.Duration - maxReconnect time.Duration - connected bool - mutex sync.RWMutex - onToolCall func(ToolCall) - verbose bool -} - -// NewBridge creates a new WebSocket bridge -func NewBridge(backendURL, authToken string, verbose bool) *Bridge { - return &Bridge{ - backendURL: backendURL, - authToken: authToken, - writeChan: make(chan Message, 100), - stopChan: make(chan struct{}), - reconnectDelay: 1 * time.Second, - maxReconnect: 60 * time.Second, - verbose: verbose, - } -} - -// SetToolCallHandler sets the callback for tool call events -func (b *Bridge) SetToolCallHandler(handler func(ToolCall)) { - b.onToolCall = handler -} - -// Connect establishes the WebSocket connection -func (b *Bridge) Connect() error { - url := fmt.Sprintf("%s?token=%s", b.backendURL, b.authToken) - - if b.verbose { - log.Printf("[Bridge] Connecting to %s", b.backendURL) - } - - conn, _, err := websocket.DefaultDialer.Dial(url, nil) - if err != nil { - return fmt.Errorf("failed to connect: %w", err) - } - - b.mutex.Lock() - b.conn = conn - b.connected = true - b.reconnectDelay = 1 * time.Second // Reset reconnect delay on successful connection - b.mutex.Unlock() - - log.Println("✅ Connected to backend") - - // Start read and write loops - go b.readLoop() - go b.writeLoop() - - return nil -} - -// ConnectWithRetry connects with automatic retry and exponential backoff -func (b *Bridge) ConnectWithRetry() { - attempt := 0 - for { - select { - case <-b.stopChan: - return - default: - } - - err := b.Connect() - if err == nil { - return - } - - attempt++ - log.Printf("❌ Connection failed (attempt %d): %v", attempt, err) - log.Printf("🔄 Retrying in %v...", b.reconnectDelay) - - time.Sleep(b.reconnectDelay) - - // Exponential backoff - b.reconnectDelay = time.Duration(math.Min( - float64(b.reconnectDelay*2), - float64(b.maxReconnect), - )) - } -} - -// readLoop handles incoming messages -func (b *Bridge) readLoop() { - defer func() { - b.handleDisconnect() - }() - - for { - var msg Message - err := b.conn.ReadJSON(&msg) - if err != nil { - if b.verbose { - log.Printf("[Bridge] Read error: %v", err) - } - return - } - - b.handleMessage(msg) - } -} - -// writeLoop handles outgoing messages -func (b *Bridge) writeLoop() { - ticker := time.NewTicker(30 * time.Second) - defer ticker.Stop() - - for { - select { - case msg := <-b.writeChan: - err := b.conn.WriteJSON(msg) - if err != nil { - if b.verbose { - log.Printf("[Bridge] Write error: %v", err) - } - return - } - - case <-ticker.C: - // Send heartbeat - if err := b.SendHeartbeat(); err != nil { - return - } - - case <-b.stopChan: - return - } - } -} - -// handleMessage processes incoming messages -func (b *Bridge) handleMessage(msg Message) { - if b.verbose { - log.Printf("[Bridge] Received: %s", msg.Type) - } - - switch msg.Type { - case "ack": - log.Printf("✅ Registration acknowledged") - if status, ok := msg.Payload["status"].(string); ok { - log.Printf(" Status: %s", status) - } - if toolsReg, ok := msg.Payload["tools_registered"].(float64); ok { - log.Printf(" Tools registered: %.0f", toolsReg) - } - - case "tool_call": - // Parse tool call - callID := msg.Payload["call_id"].(string) - toolName := msg.Payload["tool_name"].(string) - args, _ := msg.Payload["arguments"].(map[string]interface{}) - timeout, _ := msg.Payload["timeout"].(float64) - - toolCall := ToolCall{ - CallID: callID, - ToolName: toolName, - Arguments: args, - Timeout: int(timeout), - } - - log.Printf("🔧 Tool call: %s (call_id: %s)", toolName, callID) - - // Call handler if set - if b.onToolCall != nil { - b.onToolCall(toolCall) - } - - case "error": - errMsg := msg.Payload["message"].(string) - log.Printf("❌ Error from backend: %s", errMsg) - - default: - if b.verbose { - log.Printf("[Bridge] Unknown message type: %s", msg.Type) - } - } -} - -// handleDisconnect handles disconnection and reconnection -func (b *Bridge) handleDisconnect() { - b.mutex.Lock() - b.connected = false - if b.conn != nil { - b.conn.Close() - } - b.mutex.Unlock() - - log.Println("🔌 Disconnected from backend") - log.Println("🔄 Attempting to reconnect...") - - // Reconnect with exponential backoff - b.ConnectWithRetry() -} - -// RegisterTools sends tool registration message -func (b *Bridge) RegisterTools(clientID, clientVersion, platform string, tools []interface{}) error { - msg := Message{ - Type: "register_tools", - Payload: map[string]interface{}{ - "client_id": clientID, - "client_version": clientVersion, - "platform": platform, - "tools": tools, - }, - } - - b.writeChan <- msg - return nil -} - -// SendToolResult sends tool execution result back to backend -func (b *Bridge) SendToolResult(callID string, success bool, result, errorMsg string) error { - msg := Message{ - Type: "tool_result", - Payload: map[string]interface{}{ - "call_id": callID, - "success": success, - "result": result, - "error": errorMsg, - }, - } - - b.writeChan <- msg - return nil -} - -// SendHeartbeat sends a heartbeat message -func (b *Bridge) SendHeartbeat() error { - msg := Message{ - Type: "heartbeat", - Payload: map[string]interface{}{ - "timestamp": time.Now().Format(time.RFC3339), - }, - } - - b.writeChan <- msg - return nil -} - -// Close gracefully closes the bridge -func (b *Bridge) Close() error { - // Send disconnect message - msg := Message{ - Type: "disconnect", - Payload: map[string]interface{}{}, - } - b.writeChan <- msg - - // Wait a bit for message to send - time.Sleep(100 * time.Millisecond) - - close(b.stopChan) - - b.mutex.Lock() - defer b.mutex.Unlock() - - if b.conn != nil { - return b.conn.Close() - } - - return nil -} - -// IsConnected returns whether the bridge is currently connected -func (b *Bridge) IsConnected() bool { - b.mutex.RLock() - defer b.mutex.RUnlock() - return b.connected -} diff --git a/backend/mcp-bridge/internal/commands/add.go b/backend/mcp-bridge/internal/commands/add.go deleted file mode 100644 index f589d74c..00000000 --- a/backend/mcp-bridge/internal/commands/add.go +++ /dev/null @@ -1,74 +0,0 @@ -package commands - -import ( - "fmt" - - "github.com/claraverse/mcp-client/internal/config" - "github.com/spf13/cobra" -) - -var ( - serverPath string - serverType string - serverDesc string -) - -var AddCmd = &cobra.Command{ - Use: "add [name]", - Short: "Add a new MCP server", - Long: `Add a new MCP server to your configuration. The server will be -enabled by default and started when you run 'mcp-client start'. - -Examples: - mcp-client add filesystem --path /usr/local/bin/mcp-server-filesystem - mcp-client add database --path ./mcp-server-sqlite --type stdio`, - Args: cobra.ExactArgs(1), - RunE: runAdd, -} - -func init() { - AddCmd.Flags().StringVar(&serverPath, "path", "", "Path to MCP server executable (required)") - AddCmd.Flags().StringVar(&serverType, "type", "stdio", "Server type: stdio or sse") - AddCmd.Flags().StringVar(&serverDesc, "description", "", "Server description") - AddCmd.MarkFlagRequired("path") -} - -func runAdd(cmd *cobra.Command, args []string) error { - name := args[0] - - // Load config - cfg, err := config.Load() - if err != nil { - return fmt.Errorf("failed to load config: %w", err) - } - - // Create server config - server := config.MCPServer{ - Name: name, - Path: serverPath, - Type: serverType, - Description: serverDesc, - Enabled: true, - } - - // Add server - if err := cfg.AddServer(server); err != nil { - return fmt.Errorf("failed to add server: %w", err) - } - - // Save config - if err := config.Save(cfg); err != nil { - return fmt.Errorf("failed to save config: %w", err) - } - - fmt.Printf("✅ Added MCP server: %s\n", name) - fmt.Printf("📁 Path: %s\n", serverPath) - fmt.Printf("📝 Type: %s\n", serverType) - if serverDesc != "" { - fmt.Printf("💬 Description: %s\n", serverDesc) - } - fmt.Println() - fmt.Println("Server will be started automatically when you run 'mcp-client start'") - - return nil -} diff --git a/backend/mcp-bridge/internal/commands/list.go b/backend/mcp-bridge/internal/commands/list.go deleted file mode 100644 index 3f6e3062..00000000 --- a/backend/mcp-bridge/internal/commands/list.go +++ /dev/null @@ -1,58 +0,0 @@ -package commands - -import ( - "fmt" - - "github.com/claraverse/mcp-client/internal/config" - "github.com/spf13/cobra" -) - -var ListCmd = &cobra.Command{ - Use: "list", - Short: "List all configured MCP servers", - Long: `Display all MCP servers in your configuration, including their status and paths.`, - RunE: runList, -} - -func runList(cmd *cobra.Command, args []string) error { - // Load config - cfg, err := config.Load() - if err != nil { - return fmt.Errorf("failed to load config: %w", err) - } - - if len(cfg.MCPServers) == 0 { - fmt.Println("📋 No MCP servers configured") - fmt.Println() - fmt.Println("Add a server with: mcp-client add --path ") - return nil - } - - fmt.Println("📋 Configured MCP Servers:") - fmt.Println() - - for i, server := range cfg.MCPServers { - status := "🟢 Enabled" - if !server.Enabled { - status = "🔴 Disabled" - } - - fmt.Printf("%d. %s %s\n", i+1, server.Name, status) - fmt.Printf(" Type: %s\n", server.Type) - if server.Path != "" { - fmt.Printf(" Path: %s\n", server.Path) - } - if server.URL != "" { - fmt.Printf(" URL: %s\n", server.URL) - } - if server.Description != "" { - fmt.Printf(" Description: %s\n", server.Description) - } - fmt.Println() - } - - enabledCount := len(cfg.GetEnabledServers()) - fmt.Printf("Total: %d servers (%d enabled, %d disabled)\n", len(cfg.MCPServers), enabledCount, len(cfg.MCPServers)-enabledCount) - - return nil -} diff --git a/backend/mcp-bridge/internal/commands/login.go b/backend/mcp-bridge/internal/commands/login.go deleted file mode 100644 index 57cf2c35..00000000 --- a/backend/mcp-bridge/internal/commands/login.go +++ /dev/null @@ -1,141 +0,0 @@ -package commands - -import ( - "bufio" - "bytes" - "encoding/json" - "fmt" - "io" - "net/http" - "os" - "strings" - "syscall" - - "github.com/claraverse/mcp-client/internal/config" - "github.com/spf13/cobra" - "golang.org/x/term" -) - -var LoginCmd = &cobra.Command{ - Use: "login", - Short: "Authenticate with ClaraVerse", - Long: `Authenticate with your ClaraVerse account using email and password. -Your credentials will be used to obtain a JWT token from Supabase.`, - RunE: runLogin, -} - -type SupabaseAuthResponse struct { - AccessToken string `json:"access_token"` - User struct { - ID string `json:"id"` - Email string `json:"email"` - } `json:"user"` -} - -func runLogin(cmd *cobra.Command, args []string) error { - fmt.Println("🔐 ClaraVerse Authentication") - fmt.Println() - - // Get email - reader := bufio.NewReader(os.Stdin) - fmt.Print("Email: ") - email, err := reader.ReadString('\n') - if err != nil { - return fmt.Errorf("failed to read email: %w", err) - } - email = strings.TrimSpace(email) - - if email == "" { - return fmt.Errorf("email cannot be empty") - } - - // Get password (hidden input) - fmt.Print("Password: ") - passwordBytes, err := term.ReadPassword(int(syscall.Stdin)) - if err != nil { - return fmt.Errorf("failed to read password: %w", err) - } - password := string(passwordBytes) - fmt.Println() // New line after password input - - if password == "" { - return fmt.Errorf("password cannot be empty") - } - - // Authenticate with Supabase - fmt.Println("🔄 Authenticating...") - - supabaseURL := "https://ocqoqjafmjuiywsppwkw.supabase.co" - supabaseKey := "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZSIsInJlZiI6Im9jcW9xamFmbWp1aXl3c3Bwd2t3Iiwicm9sZSI6ImFub24iLCJpYXQiOjE3NjI5Njk1NTQsImV4cCI6MjA3ODU0NTU1NH0.LwM-n70KvdPpU6-lnMMgphGUPQIk62otNreXpsplYeA" - - // Create auth request - authData := map[string]string{ - "email": email, - "password": password, - } - - jsonData, err := json.Marshal(authData) - if err != nil { - return fmt.Errorf("failed to create request: %w", err) - } - - // Send auth request to Supabase - req, err := http.NewRequest("POST", supabaseURL+"/auth/v1/token?grant_type=password", bytes.NewBuffer(jsonData)) - if err != nil { - return fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("apikey", supabaseKey) - - client := &http.Client{} - resp, err := client.Do(req) - if err != nil { - return fmt.Errorf("authentication request failed: %w", err) - } - defer resp.Body.Close() - - body, err := io.ReadAll(resp.Body) - if err != nil { - return fmt.Errorf("failed to read response: %w", err) - } - - if resp.StatusCode != http.StatusOK { - return fmt.Errorf("authentication failed: %s (status: %d)", string(body), resp.StatusCode) - } - - // Parse response - var authResp SupabaseAuthResponse - if err := json.Unmarshal(body, &authResp); err != nil { - return fmt.Errorf("failed to parse response: %w", err) - } - - if authResp.AccessToken == "" { - return fmt.Errorf("no access token received") - } - - // Load or create config - cfg, err := config.Load() - if err != nil { - return fmt.Errorf("failed to load config: %w", err) - } - - // Save token and user info - cfg.AuthToken = authResp.AccessToken - cfg.UserID = authResp.User.ID - if err := config.Save(cfg); err != nil { - return fmt.Errorf("failed to save config: %w", err) - } - - fmt.Println() - fmt.Println("✅ Authentication successful!") - fmt.Printf("📧 Logged in as: %s\n", authResp.User.Email) - fmt.Printf("👤 User ID: %s\n", authResp.User.ID) - fmt.Printf("📁 Config saved to: %s\n", config.GetConfigPath()) - fmt.Println() - fmt.Println("Next steps:") - fmt.Println("1. Add MCP servers: mcp-client add --path ") - fmt.Println("2. Start client: mcp-client start") - - return nil -} diff --git a/backend/mcp-bridge/internal/commands/remove.go b/backend/mcp-bridge/internal/commands/remove.go deleted file mode 100644 index 1099f63a..00000000 --- a/backend/mcp-bridge/internal/commands/remove.go +++ /dev/null @@ -1,39 +0,0 @@ -package commands - -import ( - "fmt" - - "github.com/claraverse/mcp-client/internal/config" - "github.com/spf13/cobra" -) - -var RemoveCmd = &cobra.Command{ - Use: "remove [name]", - Short: "Remove an MCP server", - Long: `Remove an MCP server from your configuration.`, - Args: cobra.ExactArgs(1), - RunE: runRemove, -} - -func runRemove(cmd *cobra.Command, args []string) error { - name := args[0] - - // Load config - cfg, err := config.Load() - if err != nil { - return fmt.Errorf("failed to load config: %w", err) - } - - // Remove server - if err := cfg.RemoveServer(name); err != nil { - return fmt.Errorf("failed to remove server: %w", err) - } - - // Save config - if err := config.Save(cfg); err != nil { - return fmt.Errorf("failed to save config: %w", err) - } - - fmt.Printf("✅ Removed MCP server: %s\n", name) - return nil -} diff --git a/backend/mcp-bridge/internal/commands/start.go b/backend/mcp-bridge/internal/commands/start.go deleted file mode 100644 index 2e02ca7c..00000000 --- a/backend/mcp-bridge/internal/commands/start.go +++ /dev/null @@ -1,129 +0,0 @@ -package commands - -import ( - "fmt" - "log" - "os" - "os/signal" - "runtime" - "syscall" - - "github.com/claraverse/mcp-client/internal/bridge" - "github.com/claraverse/mcp-client/internal/config" - "github.com/claraverse/mcp-client/internal/registry" - "github.com/google/uuid" - "github.com/spf13/cobra" -) - -var StartCmd = &cobra.Command{ - Use: "start", - Short: "Start the MCP client and connect to backend", - Long: `Starts the MCP client daemon, connects to the ClaraVerse backend, -and registers all enabled MCP servers. The client will run in the foreground -and handle tool execution requests from the backend.`, - RunE: runStart, -} - -func runStart(cmd *cobra.Command, args []string) error { - // Load configuration - cfg, err := config.Load() - if err != nil { - return fmt.Errorf("failed to load config: %w", err) - } - - // Check if authenticated - if cfg.AuthToken == "" { - return fmt.Errorf("not authenticated. Please run 'mcp-client login' first") - } - - verbose, _ := cmd.Flags().GetBool("verbose") - - log.Println("🚀 Starting ClaraVerse MCP Client") - log.Printf("📍 Config: %s", config.GetConfigPath()) - log.Printf("🌐 Backend: %s", cfg.BackendURL) - - // Create server registry - reg := registry.NewRegistry(verbose) - - // Start all enabled MCP servers - enabledServers := cfg.GetEnabledServers() - if len(enabledServers) == 0 { - log.Println("⚠️ No MCP servers configured. Add servers with 'mcp-client add'") - } - - for _, server := range enabledServers { - if err := reg.StartServer(server); err != nil { - log.Printf("❌ Failed to start %s: %v", server.Name, err) - continue - } - } - - if reg.GetServerCount() == 0 { - return fmt.Errorf("no MCP servers started successfully") - } - - log.Printf("✅ Started %d MCP servers with %d total tools", reg.GetServerCount(), reg.GetToolCount()) - - // Create WebSocket bridge - b := bridge.NewBridge(cfg.BackendURL, cfg.AuthToken, verbose) - - // Set tool call handler - b.SetToolCallHandler(func(tc bridge.ToolCall) { - handleToolCall(reg, b, tc) - }) - - // Connect to backend - log.Println("🔌 Connecting to backend...") - if err := b.Connect(); err != nil { - return fmt.Errorf("failed to connect to backend: %w", err) - } - - // Register tools - clientID := uuid.New().String() - tools := reg.GetAllTools() - - log.Printf("📦 Registering %d tools...", len(tools)) - if err := b.RegisterTools(clientID, "1.0.0", runtime.GOOS, convertTools(tools)); err != nil { - return fmt.Errorf("failed to register tools: %w", err) - } - - log.Println("✅ MCP client running. Press Ctrl+C to exit.") - log.Println("💡 Tools are now available in your web chat!") - - // Handle graceful shutdown - sigChan := make(chan os.Signal, 1) - signal.Notify(sigChan, os.Interrupt, syscall.SIGTERM) - - <-sigChan - - log.Println("\n🛑 Shutting down...") - b.Close() - reg.StopAll() - log.Println("✅ Goodbye!") - - return nil -} - -func handleToolCall(reg *registry.Registry, b *bridge.Bridge, tc bridge.ToolCall) { - log.Printf("🔧 Executing tool: %s (call_id: %s)", tc.ToolName, tc.CallID) - - // Execute the tool - result, err := reg.ExecuteTool(tc.ToolName, tc.Arguments) - - if err != nil { - log.Printf("❌ Tool execution failed: %v", err) - b.SendToolResult(tc.CallID, false, "", err.Error()) - return - } - - log.Printf("✅ Tool executed successfully: %s", tc.ToolName) - b.SendToolResult(tc.CallID, true, result, "") -} - -func convertTools(tools []map[string]interface{}) []interface{} { - result := make([]interface{}, len(tools)) - for i, tool := range tools { - result[i] = tool - } - return result -} diff --git a/backend/mcp-bridge/internal/commands/status.go b/backend/mcp-bridge/internal/commands/status.go deleted file mode 100644 index 7497b05b..00000000 --- a/backend/mcp-bridge/internal/commands/status.go +++ /dev/null @@ -1,75 +0,0 @@ -package commands - -import ( - "fmt" - - "github.com/claraverse/mcp-client/internal/config" - "github.com/spf13/cobra" -) - -var StatusCmd = &cobra.Command{ - Use: "status", - Short: "Show client status and configuration", - Long: `Display the current status of the MCP client, including authentication and server configuration.`, - RunE: runStatus, -} - -func runStatus(cmd *cobra.Command, args []string) error { - // Load config - cfg, err := config.Load() - if err != nil { - return fmt.Errorf("failed to load config: %w", err) - } - - fmt.Println("📊 ClaraVerse MCP Client Status") - fmt.Println() - - // Authentication status - if cfg.AuthToken != "" { - fmt.Println("🔐 Authentication: ✅ Logged in") - if cfg.UserID != "" { - fmt.Printf(" User ID: %s\n", cfg.UserID) - } - } else { - fmt.Println("🔐 Authentication: ❌ Not logged in") - fmt.Println(" Run 'mcp-client login' to authenticate") - } - fmt.Println() - - // Backend configuration - fmt.Printf("🌐 Backend: %s\n", cfg.BackendURL) - fmt.Println() - - // Server configuration - enabledServers := cfg.GetEnabledServers() - fmt.Printf("📦 MCP Servers: %d configured (%d enabled)\n", len(cfg.MCPServers), len(enabledServers)) - - if len(enabledServers) > 0 { - fmt.Println() - fmt.Println("Enabled servers:") - for _, server := range enabledServers { - fmt.Printf(" • %s (%s)\n", server.Name, server.Type) - } - } - fmt.Println() - - // Configuration file - fmt.Printf("📁 Config file: %s\n", config.GetConfigPath()) - - // Next steps - if cfg.AuthToken == "" { - fmt.Println() - fmt.Println("Next steps:") - fmt.Println("1. Login: mcp-client login") - } else if len(enabledServers) == 0 { - fmt.Println() - fmt.Println("Next steps:") - fmt.Println("1. Add servers: mcp-client add --path ") - } else { - fmt.Println() - fmt.Println("✅ Ready to start!") - fmt.Println(" Run: mcp-client start") - } - - return nil -} diff --git a/backend/mcp-bridge/internal/config/config.go b/backend/mcp-bridge/internal/config/config.go deleted file mode 100644 index 5ae714f9..00000000 --- a/backend/mcp-bridge/internal/config/config.go +++ /dev/null @@ -1,159 +0,0 @@ -package config - -import ( - "fmt" - "os" - "path/filepath" - - "github.com/spf13/viper" - "gopkg.in/yaml.v3" -) - -// Config represents the application configuration -type Config struct { - BackendURL string `yaml:"backend_url" mapstructure:"backend_url"` - AuthToken string `yaml:"auth_token" mapstructure:"auth_token"` - UserID string `yaml:"user_id" mapstructure:"user_id"` - MCPServers []MCPServer `yaml:"mcp_servers" mapstructure:"mcp_servers"` -} - -// MCPServer represents a configured MCP server -type MCPServer struct { - Name string `yaml:"name" mapstructure:"name"` - Path string `yaml:"path,omitempty" mapstructure:"path"` // For executable path - Command string `yaml:"command,omitempty" mapstructure:"command"` // For command-based (e.g., "npx") - Args []string `yaml:"args,omitempty" mapstructure:"args"` // Command arguments - URL string `yaml:"url,omitempty" mapstructure:"url"` - Type string `yaml:"type" mapstructure:"type"` // "stdio" or "sse" - Config map[string]interface{} `yaml:"config,omitempty" mapstructure:"config"` - Enabled bool `yaml:"enabled" mapstructure:"enabled"` - Description string `yaml:"description,omitempty" mapstructure:"description"` -} - -var ( - configPath string - configDir string -) - -func init() { - home, err := os.UserHomeDir() - if err != nil { - panic(fmt.Sprintf("failed to get home directory: %v", err)) - } - - configDir = filepath.Join(home, ".claraverse") - configPath = filepath.Join(configDir, "mcp-config.yaml") -} - -// GetConfigPath returns the path to the config file -func GetConfigPath() string { - return configPath -} - -// GetConfigDir returns the config directory -func GetConfigDir() string { - return configDir -} - -// Load loads the configuration from file -func Load() (*Config, error) { - // Ensure config directory exists - if err := os.MkdirAll(configDir, 0755); err != nil { - return nil, fmt.Errorf("failed to create config directory: %w", err) - } - - // Check if config file exists - if _, err := os.Stat(configPath); os.IsNotExist(err) { - // Create default config - defaultConfig := &Config{ - BackendURL: "ws://localhost:3001/mcp/connect", - MCPServers: []MCPServer{}, - } - if err := Save(defaultConfig); err != nil { - return nil, fmt.Errorf("failed to create default config: %w", err) - } - return defaultConfig, nil - } - - // Read config file - viper.SetConfigFile(configPath) - if err := viper.ReadInConfig(); err != nil { - return nil, fmt.Errorf("failed to read config: %w", err) - } - - var cfg Config - if err := viper.Unmarshal(&cfg); err != nil { - return nil, fmt.Errorf("failed to unmarshal config: %w", err) - } - - return &cfg, nil -} - -// Save saves the configuration to file -func Save(cfg *Config) error { - // Ensure config directory exists - if err := os.MkdirAll(configDir, 0755); err != nil { - return fmt.Errorf("failed to create config directory: %w", err) - } - - // Marshal to YAML - data, err := yaml.Marshal(cfg) - if err != nil { - return fmt.Errorf("failed to marshal config: %w", err) - } - - // Write to file with secure permissions - if err := os.WriteFile(configPath, data, 0600); err != nil { - return fmt.Errorf("failed to write config: %w", err) - } - - return nil -} - -// AddServer adds a new MCP server to the configuration -func (c *Config) AddServer(server MCPServer) error { - // Check if server already exists - for i, s := range c.MCPServers { - if s.Name == server.Name { - // Update existing server - c.MCPServers[i] = server - return nil - } - } - - // Add new server - c.MCPServers = append(c.MCPServers, server) - return nil -} - -// RemoveServer removes an MCP server by name -func (c *Config) RemoveServer(name string) error { - for i, s := range c.MCPServers { - if s.Name == name { - c.MCPServers = append(c.MCPServers[:i], c.MCPServers[i+1:]...) - return nil - } - } - return fmt.Errorf("server %s not found", name) -} - -// GetServer retrieves a server by name -func (c *Config) GetServer(name string) (*MCPServer, error) { - for _, s := range c.MCPServers { - if s.Name == name { - return &s, nil - } - } - return nil, fmt.Errorf("server %s not found", name) -} - -// GetEnabledServers returns only enabled servers -func (c *Config) GetEnabledServers() []MCPServer { - var enabled []MCPServer - for _, s := range c.MCPServers { - if s.Enabled { - enabled = append(enabled, s) - } - } - return enabled -} diff --git a/backend/mcp-bridge/internal/mcp/executor.go b/backend/mcp-bridge/internal/mcp/executor.go deleted file mode 100644 index 74b946d7..00000000 --- a/backend/mcp-bridge/internal/mcp/executor.go +++ /dev/null @@ -1,338 +0,0 @@ -package mcp - -import ( - "bufio" - "encoding/json" - "fmt" - "io" - "log" - "os/exec" - "strings" - "sync" -) - -// JSONRPCRequest represents a JSON-RPC 2.0 request -type JSONRPCRequest struct { - JSONRPC string `json:"jsonrpc"` - ID int `json:"id"` - Method string `json:"method"` - Params map[string]interface{} `json:"params,omitempty"` -} - -// JSONRPCResponse represents a JSON-RPC 2.0 response -type JSONRPCResponse struct { - JSONRPC string `json:"jsonrpc"` - ID int `json:"id"` - Result map[string]interface{} `json:"result,omitempty"` - Error *JSONRPCError `json:"error,omitempty"` -} - -// JSONRPCError represents a JSON-RPC 2.0 error -type JSONRPCError struct { - Code int `json:"code"` - Message string `json:"message"` - Data interface{} `json:"data,omitempty"` -} - -// Tool represents an MCP tool definition -type Tool struct { - Name string `json:"name"` - Description string `json:"description"` - InputSchema map[string]interface{} `json:"inputSchema"` -} - -// Executor manages communication with an MCP server -type Executor struct { - serverPath string - cmd *exec.Cmd - stdin io.WriteCloser - stdout io.ReadCloser - stderr io.ReadCloser - reader *bufio.Reader - writer *bufio.Writer - requestID int - mutex sync.Mutex - verbose bool -} - -// NewExecutor creates a new MCP executor for a stdio server (path-based) -func NewExecutor(serverPath string, verbose bool) (*Executor, error) { - return NewExecutorWithCommand("", serverPath, nil, verbose) -} - -// NewExecutorWithCommand creates a new MCP executor with command and args support -func NewExecutorWithCommand(name, command string, args []string, verbose bool) (*Executor, error) { - var cmd *exec.Cmd - - if len(args) > 0 { - // Command with arguments (e.g., npx @browsermcp/mcp@latest) - cmd = exec.Command(command, args...) - if verbose { - log.Printf("[MCP] Starting server with command: %s %v", command, args) - } - } else { - // Single executable path - cmd = exec.Command(command) - if verbose { - log.Printf("[MCP] Starting server: %s", command) - } - } - - stdin, err := cmd.StdinPipe() - if err != nil { - return nil, fmt.Errorf("failed to create stdin pipe: %w", err) - } - - stdout, err := cmd.StdoutPipe() - if err != nil { - return nil, fmt.Errorf("failed to create stdout pipe: %w", err) - } - - stderr, err := cmd.StderrPipe() - if err != nil { - return nil, fmt.Errorf("failed to create stderr pipe: %w", err) - } - - // Start the server - if err := cmd.Start(); err != nil { - return nil, fmt.Errorf("failed to start server: %w", err) - } - - executor := &Executor{ - serverPath: command, - cmd: cmd, - stdin: stdin, - stdout: stdout, - stderr: stderr, - reader: bufio.NewReader(stdout), - writer: bufio.NewWriter(stdin), - requestID: 0, - verbose: verbose, - } - - // Start stderr reader - go executor.readStderr() - - // Initialize the server - if err := executor.initialize(); err != nil { - executor.Close() - return nil, fmt.Errorf("failed to initialize server: %w", err) - } - - return executor, nil -} - -// readStderr logs stderr output -func (e *Executor) readStderr() { - scanner := bufio.NewScanner(e.stderr) - for scanner.Scan() { - if e.verbose { - log.Printf("[MCP stderr] %s", scanner.Text()) - } - } -} - -// initialize sends the initialize request to the MCP server -func (e *Executor) initialize() error { - req := JSONRPCRequest{ - JSONRPC: "2.0", - ID: e.nextID(), - Method: "initialize", - Params: map[string]interface{}{ - "protocolVersion": "2024-11-05", - "capabilities": map[string]interface{}{ - "roots": map[string]interface{}{ - "listChanged": true, - }, - }, - "clientInfo": map[string]interface{}{ - "name": "claraverse-mcp-client", - "version": "1.0.0", - }, - }, - } - - resp, err := e.sendRequest(req) - if err != nil { - return fmt.Errorf("initialize failed: %w", err) - } - - if resp.Error != nil { - return fmt.Errorf("initialize error: %s", resp.Error.Message) - } - - if e.verbose { - log.Printf("[MCP] Server initialized") - } - - return nil -} - -// ListTools retrieves all available tools from the MCP server -func (e *Executor) ListTools() ([]Tool, error) { - req := JSONRPCRequest{ - JSONRPC: "2.0", - ID: e.nextID(), - Method: "tools/list", - } - - resp, err := e.sendRequest(req) - if err != nil { - return nil, fmt.Errorf("tools/list failed: %w", err) - } - - if resp.Error != nil { - return nil, fmt.Errorf("tools/list error: %s", resp.Error.Message) - } - - // Parse tools from result - toolsData, ok := resp.Result["tools"].([]interface{}) - if !ok { - return nil, fmt.Errorf("invalid tools response format") - } - - var tools []Tool - for _, t := range toolsData { - toolMap, ok := t.(map[string]interface{}) - if !ok { - continue - } - - tool := Tool{ - Name: toolMap["name"].(string), - Description: toolMap["description"].(string), - } - - if schema, ok := toolMap["inputSchema"].(map[string]interface{}); ok { - tool.InputSchema = schema - } - - tools = append(tools, tool) - } - - return tools, nil -} - -// CallTool executes a tool on the MCP server -func (e *Executor) CallTool(toolName string, arguments map[string]interface{}) (string, error) { - req := JSONRPCRequest{ - JSONRPC: "2.0", - ID: e.nextID(), - Method: "tools/call", - Params: map[string]interface{}{ - "name": toolName, - "arguments": arguments, - }, - } - - resp, err := e.sendRequest(req) - if err != nil { - return "", fmt.Errorf("tools/call failed: %w", err) - } - - if resp.Error != nil { - return "", fmt.Errorf("tool error: %s", resp.Error.Message) - } - - // Extract result content - content, ok := resp.Result["content"].([]interface{}) - if !ok || len(content) == 0 { - return "", fmt.Errorf("no content in tool result") - } - - // Get text from first content item - firstContent, ok := content[0].(map[string]interface{}) - if !ok { - return "", fmt.Errorf("invalid content format") - } - - text, ok := firstContent["text"].(string) - if !ok { - return "", fmt.Errorf("no text in content") - } - - return text, nil -} - -// sendRequest sends a JSON-RPC request and waits for response -func (e *Executor) sendRequest(req JSONRPCRequest) (*JSONRPCResponse, error) { - e.mutex.Lock() - defer e.mutex.Unlock() - - // Marshal request - data, err := json.Marshal(req) - if err != nil { - return nil, fmt.Errorf("failed to marshal request: %w", err) - } - - if e.verbose { - log.Printf("[MCP →] %s", string(data)) - } - - // Write request - if _, err := e.writer.Write(data); err != nil { - return nil, fmt.Errorf("failed to write request: %w", err) - } - if _, err := e.writer.WriteString("\n"); err != nil { - return nil, fmt.Errorf("failed to write newline: %w", err) - } - if err := e.writer.Flush(); err != nil { - return nil, fmt.Errorf("failed to flush: %w", err) - } - - // Read response - skip non-JSON lines (logs, etc.) - var resp JSONRPCResponse - maxAttempts := 100 // Prevent infinite loop - for attempt := 0; attempt < maxAttempts; attempt++ { - line, err := e.reader.ReadString('\n') - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - line = strings.TrimSpace(line) - if line == "" { - continue // Skip empty lines - } - - if e.verbose { - log.Printf("[MCP ←] %s", line) - } - - // Try to parse as JSON-RPC response - if err := json.Unmarshal([]byte(line), &resp); err != nil { - // Not valid JSON-RPC, might be a log line - skip it - if e.verbose { - log.Printf("[MCP] Skipping non-JSON line: %s", line) - } - continue - } - - // Successfully parsed JSON-RPC response - return &resp, nil - } - - return nil, fmt.Errorf("no valid JSON-RPC response found after %d lines", maxAttempts) -} - -// nextID returns the next request ID -func (e *Executor) nextID() int { - e.requestID++ - return e.requestID -} - -// Close terminates the MCP server -func (e *Executor) Close() error { - if e.stdin != nil { - e.stdin.Close() - } - if e.stdout != nil { - e.stdout.Close() - } - if e.stderr != nil { - e.stderr.Close() - } - if e.cmd != nil && e.cmd.Process != nil { - return e.cmd.Process.Kill() - } - return nil -} diff --git a/backend/mcp-bridge/internal/registry/registry.go b/backend/mcp-bridge/internal/registry/registry.go deleted file mode 100644 index dd4cc759..00000000 --- a/backend/mcp-bridge/internal/registry/registry.go +++ /dev/null @@ -1,208 +0,0 @@ -package registry - -import ( - "fmt" - "log" - "sync" - - "github.com/claraverse/mcp-client/internal/config" - "github.com/claraverse/mcp-client/internal/mcp" -) - -// ServerInstance represents a running MCP server -type ServerInstance struct { - Config config.MCPServer - Executor *mcp.Executor - Tools []mcp.Tool -} - -// Registry manages all MCP server instances -type Registry struct { - servers map[string]*ServerInstance - mutex sync.RWMutex - verbose bool -} - -// NewRegistry creates a new server registry -func NewRegistry(verbose bool) *Registry { - return &Registry{ - servers: make(map[string]*ServerInstance), - verbose: verbose, - } -} - -// StartServer starts an MCP server -func (r *Registry) StartServer(cfg config.MCPServer) error { - r.mutex.Lock() - defer r.mutex.Unlock() - - // Check if already running - if _, exists := r.servers[cfg.Name]; exists { - return fmt.Errorf("server %s is already running", cfg.Name) - } - - // Only support stdio for now - if cfg.Type != "stdio" { - return fmt.Errorf("only stdio servers are supported (server %s uses %s)", cfg.Name, cfg.Type) - } - - log.Printf("🚀 Starting MCP server: %s", cfg.Name) - - // Create executor - check if command-based or path-based - var executor *mcp.Executor - var err error - - if cfg.Command != "" { - // Command-based server (e.g., npx @browsermcp/mcp@latest) - executor, err = mcp.NewExecutorWithCommand(cfg.Name, cfg.Command, cfg.Args, r.verbose) - } else if cfg.Path != "" { - // Path-based server (e.g., /path/to/server.exe) - executor, err = mcp.NewExecutor(cfg.Path, r.verbose) - } else { - return fmt.Errorf("server %s must have either 'path' or 'command' configured", cfg.Name) - } - - if err != nil { - return fmt.Errorf("failed to start server %s: %w", cfg.Name, err) - } - - // List tools - tools, err := executor.ListTools() - if err != nil { - executor.Close() - return fmt.Errorf("failed to list tools from %s: %w", cfg.Name, err) - } - - instance := &ServerInstance{ - Config: cfg, - Executor: executor, - Tools: tools, - } - - r.servers[cfg.Name] = instance - - log.Printf("✅ Server %s started with %d tools", cfg.Name, len(tools)) - for _, tool := range tools { - log.Printf(" - %s: %s", tool.Name, tool.Description) - } - - return nil -} - -// StopServer stops an MCP server -func (r *Registry) StopServer(name string) error { - r.mutex.Lock() - defer r.mutex.Unlock() - - instance, exists := r.servers[name] - if !exists { - return fmt.Errorf("server %s is not running", name) - } - - log.Printf("🛑 Stopping MCP server: %s", name) - - if err := instance.Executor.Close(); err != nil { - log.Printf("Warning: error closing executor for %s: %v", name, err) - } - - delete(r.servers, name) - - log.Printf("✅ Server %s stopped", name) - return nil -} - -// StopAll stops all running servers -func (r *Registry) StopAll() { - r.mutex.Lock() - defer r.mutex.Unlock() - - for name, instance := range r.servers { - log.Printf("🛑 Stopping server: %s", name) - instance.Executor.Close() - } - - r.servers = make(map[string]*ServerInstance) -} - -// GetAllTools returns all tools from all running servers -func (r *Registry) GetAllTools() []map[string]interface{} { - r.mutex.RLock() - defer r.mutex.RUnlock() - - var allTools []map[string]interface{} - - for _, instance := range r.servers { - for _, tool := range instance.Tools { - // Convert MCP tool to OpenAI format - toolDef := map[string]interface{}{ - "name": tool.Name, - "description": tool.Description, - "parameters": tool.InputSchema, - } - allTools = append(allTools, toolDef) - } - } - - return allTools -} - -// ExecuteTool executes a tool by finding which server provides it -func (r *Registry) ExecuteTool(toolName string, arguments map[string]interface{}) (string, error) { - r.mutex.RLock() - defer r.mutex.RUnlock() - - // Find which server has this tool - for serverName, instance := range r.servers { - for _, tool := range instance.Tools { - if tool.Name == toolName { - log.Printf("🔧 Executing %s on server %s", toolName, serverName) - return instance.Executor.CallTool(toolName, arguments) - } - } - } - - return "", fmt.Errorf("tool %s not found in any running server", toolName) -} - -// GetServerCount returns the number of running servers -func (r *Registry) GetServerCount() int { - r.mutex.RLock() - defer r.mutex.RUnlock() - return len(r.servers) -} - -// GetToolCount returns the total number of tools across all servers -func (r *Registry) GetToolCount() int { - r.mutex.RLock() - defer r.mutex.RUnlock() - - count := 0 - for _, instance := range r.servers { - count += len(instance.Tools) - } - return count -} - -// GetServerNames returns names of all running servers -func (r *Registry) GetServerNames() []string { - r.mutex.RLock() - defer r.mutex.RUnlock() - - names := make([]string, 0, len(r.servers)) - for name := range r.servers { - names = append(names, name) - } - return names -} - -// GetServer returns a server instance by name -func (r *Registry) GetServer(name string) (*ServerInstance, error) { - r.mutex.RLock() - defer r.mutex.RUnlock() - - instance, exists := r.servers[name] - if !exists { - return nil, fmt.Errorf("server %s not found", name) - } - return instance, nil -} diff --git a/backend/mcp-bridge/mcp-client.exe~ b/backend/mcp-bridge/mcp-client.exe~ deleted file mode 100644 index 5125e66a..00000000 Binary files a/backend/mcp-bridge/mcp-client.exe~ and /dev/null differ diff --git a/backend/mcp-bridge/test/mock_client.go b/backend/mcp-bridge/test/mock_client.go deleted file mode 100644 index cad1e645..00000000 --- a/backend/mcp-bridge/test/mock_client.go +++ /dev/null @@ -1,226 +0,0 @@ -package main - -import ( - "flag" - "fmt" - "log" - "os" - "os/signal" - "time" - - "github.com/gorilla/websocket" -) - -// Message types matching backend protocol -type Message struct { - Type string `json:"type"` - Payload map[string]interface{} `json:"payload"` -} - -type Tool struct { - Name string `json:"name"` - Description string `json:"description"` - Parameters map[string]interface{} `json:"parameters"` -} - -func main() { - // Command line flags - backendURL := flag.String("url", "ws://localhost:3001/mcp/connect", "Backend WebSocket URL") - token := flag.String("token", "", "JWT authentication token (required)") - flag.Parse() - - if *token == "" { - log.Fatal("❌ Error: --token is required\n\nUsage: go run mock_client.go --token YOUR_JWT_TOKEN\n") - } - - // Build connection URL with token - url := fmt.Sprintf("%s?token=%s", *backendURL, *token) - - log.Printf("🔌 Connecting to backend: %s", *backendURL) - - // Connect to WebSocket - conn, resp, err := websocket.DefaultDialer.Dial(url, nil) - if err != nil { - if resp != nil { - log.Fatalf("❌ Connection failed: %v (HTTP %d)", err, resp.StatusCode) - } - log.Fatalf("❌ Connection failed: %v", err) - } - defer conn.Close() - - log.Println("✅ Connected to backend!") - - // Set up signal handling for graceful shutdown - interrupt := make(chan os.Signal, 1) - signal.Notify(interrupt, os.Interrupt) - - done := make(chan struct{}) - - // Start read loop - go func() { - defer close(done) - for { - var msg Message - err := conn.ReadJSON(&msg) - if err != nil { - log.Printf("❌ Read error: %v", err) - return - } - - handleMessage(msg) - } - }() - - // Register test tools - log.Println("📦 Registering test tools...") - err = registerTools(conn) - if err != nil { - log.Fatalf("❌ Registration failed: %v", err) - } - - // Send heartbeat every 15 seconds - ticker := time.NewTicker(15 * time.Second) - defer ticker.Stop() - - log.Println("✅ Mock client running. Press Ctrl+C to exit.") - log.Println("💡 Now try chatting in the web browser and check if tools are visible!") - - for { - select { - case <-done: - return - - case <-ticker.C: - // Send heartbeat - err := sendHeartbeat(conn) - if err != nil { - log.Printf("❌ Heartbeat failed: %v", err) - return - } - log.Println("💓 Heartbeat sent") - - case <-interrupt: - log.Println("\n🛑 Interrupt received, closing connection...") - - // Send disconnect message - sendDisconnect(conn) - - // Close connection gracefully - err := conn.WriteMessage(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseNormalClosure, "")) - if err != nil { - log.Printf("Write close error: %v", err) - } - - select { - case <-done: - case <-time.After(time.Second): - } - return - } - } -} - -func registerTools(conn *websocket.Conn) error { - // Create test tools - tools := []Tool{ - { - Name: "mock_echo", - Description: "A test tool that echoes back the input message", - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "message": map[string]interface{}{ - "type": "string", - "description": "Message to echo back", - }, - }, - "required": []string{"message"}, - }, - }, - { - Name: "mock_timestamp", - Description: "A test tool that returns current timestamp", - Parameters: map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{}, - }, - }, - } - - // Create registration message - msg := Message{ - Type: "register_tools", - Payload: map[string]interface{}{ - "client_id": "mock-client-" + fmt.Sprintf("%d", time.Now().Unix()), - "client_version": "1.0.0-test", - "platform": "test", - "tools": tools, - }, - } - - log.Printf("📤 Sending registration: %d tools", len(tools)) - err := conn.WriteJSON(msg) - if err != nil { - return fmt.Errorf("failed to send registration: %w", err) - } - - return nil -} - -func sendHeartbeat(conn *websocket.Conn) error { - msg := Message{ - Type: "heartbeat", - Payload: map[string]interface{}{ - "timestamp": time.Now().Format(time.RFC3339), - }, - } - - return conn.WriteJSON(msg) -} - -func sendDisconnect(conn *websocket.Conn) error { - msg := Message{ - Type: "disconnect", - Payload: map[string]interface{}{}, - } - - return conn.WriteJSON(msg) -} - -func handleMessage(msg Message) { - switch msg.Type { - case "ack": - log.Printf("✅ ACK received: %+v", msg.Payload) - if status, ok := msg.Payload["status"].(string); ok { - log.Printf(" Status: %s", status) - } - if toolsReg, ok := msg.Payload["tools_registered"].(float64); ok { - log.Printf(" Tools registered: %.0f", toolsReg) - } - - case "tool_call": - log.Printf("🔧 TOOL CALL received: %+v", msg.Payload) - callID := msg.Payload["call_id"].(string) - toolName := msg.Payload["tool_name"].(string) - args := msg.Payload["arguments"] - - log.Printf(" Call ID: %s", callID) - log.Printf(" Tool: %s", toolName) - log.Printf(" Arguments: %+v", args) - - // TODO: In a real client, this would execute the MCP tool - // For now, we just log it - log.Println(" ⚠️ Note: This mock client cannot execute tools yet") - log.Println(" 💡 To test tool execution, build the full MCP client") - - case "error": - log.Printf("❌ ERROR received: %+v", msg.Payload) - if errMsg, ok := msg.Payload["message"].(string); ok { - log.Printf(" Message: %s", errMsg) - } - - default: - log.Printf("📨 Unknown message type: %s", msg.Type) - log.Printf(" Payload: %+v", msg.Payload) - } -} diff --git a/backend/mcp-bridge/test/quick-test.bat b/backend/mcp-bridge/test/quick-test.bat deleted file mode 100644 index 6de0b9c0..00000000 --- a/backend/mcp-bridge/test/quick-test.bat +++ /dev/null @@ -1,93 +0,0 @@ -@echo off -REM Quick MCP Bridge Test Script for Windows -REM This script helps you test the MCP Bridge end-to-end - -echo. -echo ======================================== -echo MCP Bridge Quick Test (Windows) -echo ======================================== -echo. - -REM Step 1: Build the client -echo [1/5] Building MCP client... -cd .. -if exist "mcp-client.exe" del mcp-client.exe -go build -o mcp-client.exe .\cmd\mcp-client -if %ERRORLEVEL% NEQ 0 ( - echo [X] Build failed! - pause - exit /b 1 -) -echo [OK] Client built successfully -echo. - -REM Step 2: Check version -echo [2/5] Checking client version... -.\mcp-client.exe --version -if %ERRORLEVEL% NEQ 0 ( - echo [X] Client won't run! - pause - exit /b 1 -) -echo. - -REM Step 3: Instructions for backend -echo [3/5] Backend Check -echo. -echo Is your backend running on port 3001? -echo If not, open a NEW terminal and run: -echo cd backend -echo go run .\cmd\server -echo. -pause - -REM Step 4: Get JWT Token -echo. -echo [4/5] Getting JWT Token -echo. -echo Open your browser and follow these steps: -echo 1. Go to http://localhost:5173 -echo 2. Login to ClaraVerse -echo 3. Press F12 (open DevTools) -echo 4. Go to Console tab -echo 5. Run: localStorage.getItem('supabase.auth.token') -echo 6. Copy the token (starts with ey...) -echo. -echo When ready, run the login command: -echo .\mcp-client.exe login -echo. -pause - -REM Step 5: Instructions for running -echo. -echo [5/5] Testing Instructions -echo. -echo Now you can test the client! Run these commands: -echo. -echo 1. Login (if you haven't already): -echo .\mcp-client.exe login -echo. -echo 2. Check status: -echo .\mcp-client.exe status -echo. -echo 3. Start the client: -echo .\mcp-client.exe start -echo. -echo 4. Open browser and chat: -echo - Go to http://localhost:5173 -echo - Start a NEW conversation -echo - Ask: "What tools do you have?" -echo - You should see any MCP tools you've added -echo. -echo ======================================== -echo Setup Complete! -echo ======================================== -echo. -echo Client binary: %cd%\mcp-client.exe -echo Config will be: %USERPROFILE%\.claraverse\mcp-config.yaml -echo. -echo For mock testing (without real MCP servers): -echo cd test -echo go run mock_client.go --token YOUR_TOKEN -echo. -pause diff --git a/backend/mcp-bridge/test/verify.bat b/backend/mcp-bridge/test/verify.bat deleted file mode 100644 index c89694f8..00000000 --- a/backend/mcp-bridge/test/verify.bat +++ /dev/null @@ -1,154 +0,0 @@ -@echo off -REM MCP Bridge Backend Verification Script (Windows) - -echo. -echo MCP Bridge Backend Verification -echo ================================== -echo. - -REM Check if we're in the right directory -if not exist "..\..\go.mod" ( - echo [X] Error: Must run from mcp-bridge\test directory - echo cd mcp-bridge\test - echo verify.bat - exit /b 1 -) - -echo [OK] Directory check passed - -REM Check if backend can be built -echo. -echo Building backend to verify compilation... -cd ..\.. -go build -o temp_test.exe .\cmd\server >nul 2>&1 -if %ERRORLEVEL% EQU 0 ( - echo [OK] Backend compiles successfully - del temp_test.exe >nul 2>&1 -) else ( - echo [X] Backend compilation failed - echo Run: go build .\cmd\server - exit /b 1 -) - -REM Check database schema -echo. -echo Checking database schema... -findstr /C:"mcp_connections" internal\database\database.go >nul -if %ERRORLEVEL% EQU 0 ( - echo [OK] mcp_connections table defined -) else ( - echo [X] mcp_connections table missing - exit /b 1 -) - -findstr /C:"mcp_tools" internal\database\database.go >nul -if %ERRORLEVEL% EQU 0 ( - echo [OK] mcp_tools table defined -) else ( - echo [X] mcp_tools table missing - exit /b 1 -) - -findstr /C:"mcp_audit_log" internal\database\database.go >nul -if %ERRORLEVEL% EQU 0 ( - echo [OK] mcp_audit_log table defined -) else ( - echo [X] mcp_audit_log table missing - exit /b 1 -) - -REM Check MCP models -echo. -echo Checking MCP models... -if exist "internal\models\mcp.go" ( - echo [OK] MCP models file exists -) else ( - echo [X] internal\models\mcp.go missing - exit /b 1 -) - -REM Check MCP service -echo. -echo Checking MCP service... -if exist "internal\services\mcp_bridge_service.go" ( - echo [OK] MCP bridge service exists -) else ( - echo [X] internal\services\mcp_bridge_service.go missing - exit /b 1 -) - -REM Check MCP handler -echo. -echo Checking MCP handler... -if exist "internal\handlers\mcp_websocket.go" ( - echo [OK] MCP WebSocket handler exists -) else ( - echo [X] internal\handlers\mcp_websocket.go missing - exit /b 1 -) - -REM Check tool registry -echo. -echo Checking tool registry extensions... -findstr /C:"RegisterUserTool" internal\tools\registry.go >nul -if %ERRORLEVEL% EQU 0 ( - echo [OK] RegisterUserTool method exists -) else ( - echo [X] RegisterUserTool method missing - exit /b 1 -) - -findstr /C:"GetUserTools" internal\tools\registry.go >nul -if %ERRORLEVEL% EQU 0 ( - echo [OK] GetUserTools method exists -) else ( - echo [X] GetUserTools method missing - exit /b 1 -) - -REM Check MCP endpoint -echo. -echo Checking MCP endpoint... -findstr /C:"/mcp/connect" cmd\server\main.go >nul -if %ERRORLEVEL% EQU 0 ( - echo [OK] /mcp/connect endpoint registered -) else ( - echo [X] /mcp/connect endpoint missing - exit /b 1 -) - -REM Check dependencies -echo. -echo Checking mock client dependencies... -cd mcp-bridge -go list -m github.com/gorilla/websocket >nul 2>&1 -if %ERRORLEVEL% EQU 0 ( - echo [OK] gorilla/websocket dependency installed -) else ( - echo [!] gorilla/websocket not installed - echo Run: go get github.com/gorilla/websocket -) - -REM Final summary -echo. -echo ================================== -echo [OK] All checks passed! -echo. -echo Next steps: -echo 1. Start backend: -echo cd backend -echo go run .\cmd\server -echo. -echo 2. Get JWT token from browser (see QUICKSTART.md) -echo. -echo 3. Run mock client: -echo cd mcp-bridge\test -echo go run mock_client.go --token YOUR_TOKEN -echo. -echo Documentation: -echo - Quick test: mcp-bridge\test\QUICKSTART.md -echo - Full guide: mcp-bridge\test\README.md -echo - Architecture: mcp-bridge\README.md -echo. -echo Backend is ready for testing! -echo. diff --git a/backend/mcp-bridge/test/verify.sh b/backend/mcp-bridge/test/verify.sh deleted file mode 100644 index 0856fc48..00000000 --- a/backend/mcp-bridge/test/verify.sh +++ /dev/null @@ -1,138 +0,0 @@ -#!/bin/bash - -# MCP Bridge Backend Verification Script -# This script checks if the backend is properly configured for MCP - -echo "🔍 MCP Bridge Backend Verification" -echo "===================================" -echo "" - -# Check if we're in the right directory -if [ ! -f "../../go.mod" ]; then - echo "❌ Error: Must run from mcp-bridge/test directory" - echo " cd mcp-bridge/test && bash verify.sh" - exit 1 -fi - -echo "✅ Directory check passed" - -# Check if backend can be built -echo "" -echo "🔨 Checking if backend compiles..." -cd ../../ -if go build -o /tmp/claraverse-test ./cmd/server > /dev/null 2>&1; then - echo "✅ Backend compiles successfully" - rm -f /tmp/claraverse-test -else - echo "❌ Backend compilation failed" - echo " Run: go build ./cmd/server" - exit 1 -fi - -# Check if MCP tables exist in database schema -echo "" -echo "📋 Checking database schema..." -if grep -q "mcp_connections" internal/database/database.go; then - echo "✅ mcp_connections table defined" -else - echo "❌ mcp_connections table missing" - exit 1 -fi - -if grep -q "mcp_tools" internal/database/database.go; then - echo "✅ mcp_tools table defined" -else - echo "❌ mcp_tools table missing" - exit 1 -fi - -if grep -q "mcp_audit_log" internal/database/database.go; then - echo "✅ mcp_audit_log table defined" -else - echo "❌ mcp_audit_log table missing" - exit 1 -fi - -# Check if MCP models exist -echo "" -echo "📦 Checking MCP models..." -if [ -f "internal/models/mcp.go" ]; then - echo "✅ MCP models file exists" -else - echo "❌ internal/models/mcp.go missing" - exit 1 -fi - -# Check if MCP service exists -echo "" -echo "⚙️ Checking MCP service..." -if [ -f "internal/services/mcp_bridge_service.go" ]; then - echo "✅ MCP bridge service exists" -else - echo "❌ internal/services/mcp_bridge_service.go missing" - exit 1 -fi - -# Check if MCP handler exists -echo "" -echo "🔌 Checking MCP handler..." -if [ -f "internal/handlers/mcp_websocket.go" ]; then - echo "✅ MCP WebSocket handler exists" -else - echo "❌ internal/handlers/mcp_websocket.go missing" - exit 1 -fi - -# Check if tool registry has user tool support -echo "" -echo "🛠️ Checking tool registry extensions..." -if grep -q "RegisterUserTool" internal/tools/registry.go; then - echo "✅ RegisterUserTool method exists" -else - echo "❌ RegisterUserTool method missing" - exit 1 -fi - -if grep -q "GetUserTools" internal/tools/registry.go; then - echo "✅ GetUserTools method exists" -else - echo "❌ GetUserTools method missing" - exit 1 -fi - -# Check if main.go has MCP endpoint -echo "" -echo "🌐 Checking MCP endpoint..." -if grep -q "/mcp/connect" cmd/server/main.go; then - echo "✅ /mcp/connect endpoint registered" -else - echo "❌ /mcp/connect endpoint missing" - exit 1 -fi - -# Check if mock client dependencies are available -echo "" -echo "📚 Checking mock client dependencies..." -cd mcp-bridge -if go list -m github.com/gorilla/websocket > /dev/null 2>&1; then - echo "✅ gorilla/websocket dependency installed" -else - echo "⚠️ gorilla/websocket not installed (run: go get github.com/gorilla/websocket)" -fi - -# Final summary -echo "" -echo "==================================" -echo "✅ All checks passed!" -echo "" -echo "📝 Next steps:" -echo " 1. Start backend: cd backend && go run ./cmd/server" -echo " 2. Get JWT token from browser (see QUICKSTART.md)" -echo " 3. Run mock client: cd mcp-bridge/test && go run mock_client.go --token YOUR_TOKEN" -echo "" -echo "📖 Documentation:" -echo " - Quick test: mcp-bridge/test/QUICKSTART.md" -echo " - Full guide: mcp-bridge/test/README.md" -echo " - Architecture: mcp-bridge/README.md" -echo "" -echo "🎉 Backend is ready for testing!" diff --git a/backend/migrations/001_initial_schema.sql b/backend/migrations/001_initial_schema.sql deleted file mode 100644 index 8117cf94..00000000 --- a/backend/migrations/001_initial_schema.sql +++ /dev/null @@ -1,251 +0,0 @@ --- ClaraVerse MySQL Schema Migration --- Date: 2026-01-17 --- Purpose: Initial schema for provider/model management - --- Drop tables if they exist (for clean re-runs) -DROP TABLE IF EXISTS mcp_audit_log; -DROP TABLE IF EXISTS mcp_tools; -DROP TABLE IF EXISTS mcp_connections; -DROP TABLE IF EXISTS model_refresh_log; -DROP TABLE IF EXISTS model_capabilities; -DROP TABLE IF EXISTS provider_model_filters; -DROP TABLE IF EXISTS model_aliases; -DROP TABLE IF EXISTS recommended_models; -DROP TABLE IF EXISTS models; -DROP TABLE IF EXISTS providers; - --- ============================================================================= --- PROVIDERS TABLE --- Stores AI API providers (OpenAI, Anthropic, custom providers, etc.) --- ============================================================================= -CREATE TABLE providers ( - id INT AUTO_INCREMENT PRIMARY KEY, - name VARCHAR(255) NOT NULL UNIQUE COMMENT 'Provider name (unique identifier)', - base_url VARCHAR(512) NOT NULL COMMENT 'API endpoint URL', - api_key TEXT COMMENT 'API authentication key (encrypted)', - enabled BOOLEAN DEFAULT TRUE COMMENT 'Is provider active', - audio_only BOOLEAN DEFAULT FALSE COMMENT 'Audio-only provider (e.g., Groq)', - image_only BOOLEAN DEFAULT FALSE COMMENT 'Image generation only', - image_edit_only BOOLEAN DEFAULT FALSE COMMENT 'Image editing only', - secure BOOLEAN DEFAULT FALSE COMMENT 'Privacy-focused provider (no data storage)', - default_model VARCHAR(255) COMMENT 'Default model for this provider', - system_prompt TEXT COMMENT 'Provider-level system prompt', - favicon VARCHAR(512) COMMENT 'Provider icon URL', - created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, - updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, - - INDEX idx_enabled (enabled), - INDEX idx_name (name) -) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci -COMMENT='AI API providers'; - --- ============================================================================= --- MODELS TABLE --- Stores LLM models available from providers --- ============================================================================= -CREATE TABLE models ( - id VARCHAR(512) PRIMARY KEY COMMENT 'Unique model identifier', - provider_id INT NOT NULL COMMENT 'Foreign key to providers', - name VARCHAR(255) NOT NULL COMMENT 'Model name (API identifier)', - display_name VARCHAR(255) COMMENT 'UI display name', - description TEXT COMMENT 'Model description', - context_length INT COMMENT 'Token context window size', - supports_tools BOOLEAN DEFAULT FALSE COMMENT 'Function calling support', - supports_streaming BOOLEAN DEFAULT FALSE COMMENT 'SSE streaming support', - supports_vision BOOLEAN DEFAULT FALSE COMMENT 'Image/vision support', - smart_tool_router BOOLEAN DEFAULT FALSE COMMENT 'Can predict tool usage for context optimization', - agents_enabled BOOLEAN DEFAULT FALSE COMMENT 'Available in agent builder', - is_visible BOOLEAN DEFAULT TRUE COMMENT 'Show in UI', - system_prompt TEXT COMMENT 'Model-specific system prompt', - fetched_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP COMMENT 'When fetched from provider API', - created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, - updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, - - FOREIGN KEY (provider_id) REFERENCES providers(id) ON DELETE CASCADE, - INDEX idx_provider (provider_id), - INDEX idx_visible (is_visible), - INDEX idx_agents (agents_enabled), - INDEX idx_tool_router (smart_tool_router), - INDEX idx_capabilities (supports_tools, supports_vision, supports_streaming) -) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci -COMMENT='LLM models from providers'; - --- ============================================================================= --- MODEL_ALIASES TABLE --- Maps frontend display names to actual model names --- ============================================================================= -CREATE TABLE model_aliases ( - id INT AUTO_INCREMENT PRIMARY KEY, - alias_name VARCHAR(255) NOT NULL COMMENT 'Frontend display name', - model_id VARCHAR(512) NOT NULL COMMENT 'Actual model ID (foreign key)', - provider_id INT NOT NULL COMMENT 'Provider ID (foreign key)', - display_name VARCHAR(255) NOT NULL COMMENT 'UI display name', - description TEXT COMMENT 'Model description', - supports_vision BOOLEAN COMMENT 'Vision support override', - agents_enabled BOOLEAN DEFAULT FALSE COMMENT 'Available in agent builder', - smart_tool_router BOOLEAN DEFAULT FALSE COMMENT 'Can be used as tool predictor', - free_tier BOOLEAN DEFAULT FALSE COMMENT 'Available on free tier', - structured_output_support ENUM('excellent', 'good', 'fair', 'poor', 'unknown') COMMENT 'Structured output quality', - structured_output_compliance INT COMMENT '0-100 percentage compliance', - structured_output_warning TEXT COMMENT 'Warning message for structured output', - structured_output_speed_ms INT COMMENT 'Average latency for structured outputs', - structured_output_badge VARCHAR(50) COMMENT 'UI badge (e.g., "FASTEST")', - memory_extractor BOOLEAN DEFAULT FALSE COMMENT 'Can extract memories from conversations', - memory_selector BOOLEAN DEFAULT FALSE COMMENT 'Can select relevant memories', - created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, - updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, - - UNIQUE KEY unique_alias_provider (alias_name, provider_id), - FOREIGN KEY (model_id) REFERENCES models(id) ON DELETE CASCADE, - FOREIGN KEY (provider_id) REFERENCES providers(id) ON DELETE CASCADE, - INDEX idx_alias (alias_name), - INDEX idx_agents (agents_enabled), - INDEX idx_memory (memory_extractor, memory_selector) -) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci -COMMENT='Frontend model name mappings'; - --- ============================================================================= --- PROVIDER_MODEL_FILTERS TABLE --- Include/exclude patterns for model visibility --- ============================================================================= -CREATE TABLE provider_model_filters ( - id INT AUTO_INCREMENT PRIMARY KEY, - provider_id INT NOT NULL COMMENT 'Provider ID (foreign key)', - model_pattern VARCHAR(255) NOT NULL COMMENT 'Regex/glob pattern (e.g., "gpt-4*")', - action ENUM('include', 'exclude') NOT NULL COMMENT 'Filter action', - priority INT DEFAULT 0 COMMENT 'Processing priority (higher = processed first)', - created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, - - FOREIGN KEY (provider_id) REFERENCES providers(id) ON DELETE CASCADE, - INDEX idx_provider (provider_id), - INDEX idx_priority (priority DESC) -) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci -COMMENT='Model visibility filters'; - --- ============================================================================= --- MODEL_CAPABILITIES TABLE --- Test results and benchmark data for models --- ============================================================================= -CREATE TABLE model_capabilities ( - id INT AUTO_INCREMENT PRIMARY KEY, - model_id VARCHAR(512) NOT NULL COMMENT 'Model ID (foreign key)', - provider_id INT NOT NULL COMMENT 'Provider ID (foreign key)', - connection_test_passed BOOLEAN COMMENT 'Connection test result', - connection_test_latency_ms INT COMMENT 'Connection latency in milliseconds', - connection_test_error TEXT COMMENT 'Connection test error message', - capability_test_passed BOOLEAN COMMENT 'Capability test result', - tools_test_passed BOOLEAN COMMENT 'Function calling test', - vision_test_passed BOOLEAN COMMENT 'Vision/image test', - streaming_test_passed BOOLEAN COMMENT 'Streaming test', - structured_output_compliance INT COMMENT '0-100 structured output compliance', - structured_output_speed_ms INT COMMENT 'Structured output latency', - structured_output_quality ENUM('excellent', 'good', 'fair', 'poor') COMMENT 'Output quality rating', - benchmark_tokens_per_second FLOAT COMMENT 'Benchmark throughput', - benchmark_avg_latency_ms INT COMMENT 'Average benchmark latency', - benchmark_date TIMESTAMP COMMENT 'When benchmark was run', - tested_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, - updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, - - FOREIGN KEY (model_id) REFERENCES models(id) ON DELETE CASCADE, - FOREIGN KEY (provider_id) REFERENCES providers(id) ON DELETE CASCADE, - UNIQUE KEY unique_model_capability (model_id), - INDEX idx_quality (structured_output_quality), - INDEX idx_performance (benchmark_tokens_per_second) -) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci -COMMENT='Model test results and benchmarks'; - --- ============================================================================= --- MODEL_REFRESH_LOG TABLE --- Audit trail for model fetch operations --- ============================================================================= -CREATE TABLE model_refresh_log ( - id INT AUTO_INCREMENT PRIMARY KEY, - provider_id INT NOT NULL COMMENT 'Provider ID (foreign key)', - models_fetched INT NOT NULL COMMENT 'Number of models fetched', - success BOOLEAN DEFAULT TRUE COMMENT 'Was fetch successful', - error_message TEXT COMMENT 'Error message if failed', - refreshed_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, - - FOREIGN KEY (provider_id) REFERENCES providers(id) ON DELETE CASCADE, - INDEX idx_provider_date (provider_id, refreshed_at DESC) -) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci -COMMENT='Model fetch audit trail'; - --- ============================================================================= --- RECOMMENDED_MODELS TABLE --- Tier-based model recommendations per provider --- ============================================================================= -CREATE TABLE recommended_models ( - id INT AUTO_INCREMENT PRIMARY KEY, - provider_id INT NOT NULL COMMENT 'Provider ID (foreign key)', - tier ENUM('top', 'medium', 'fastest', 'new') NOT NULL COMMENT 'Recommendation tier', - model_alias VARCHAR(255) NOT NULL COMMENT 'Model alias name', - created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, - updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, - - UNIQUE KEY unique_provider_tier (provider_id, tier), - FOREIGN KEY (provider_id) REFERENCES providers(id) ON DELETE CASCADE, - INDEX idx_tier (tier) -) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci -COMMENT='Recommended models by tier'; - --- ============================================================================= --- MCP (Model Context Protocol) TABLES (Optional) --- Only used if MCP integration is enabled --- ============================================================================= - -CREATE TABLE mcp_connections ( - id INT AUTO_INCREMENT PRIMARY KEY, - user_id VARCHAR(255) NOT NULL COMMENT 'Supabase user ID', - client_id VARCHAR(255) NOT NULL COMMENT 'MCP client identifier', - connection_name VARCHAR(255) COMMENT 'Friendly connection name', - is_active BOOLEAN DEFAULT TRUE COMMENT 'Is connection currently active', - connected_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, - disconnected_at TIMESTAMP NULL COMMENT 'When connection was closed', - - INDEX idx_user (user_id), - INDEX idx_active (is_active) -) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci -COMMENT='MCP client connections'; - -CREATE TABLE mcp_tools ( - id INT AUTO_INCREMENT PRIMARY KEY, - user_id VARCHAR(255) NOT NULL COMMENT 'Supabase user ID', - connection_id INT NOT NULL COMMENT 'MCP connection ID (foreign key)', - tool_name VARCHAR(255) NOT NULL COMMENT 'Tool name', - tool_definition JSON NOT NULL COMMENT 'Tool schema definition', - created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, - - FOREIGN KEY (connection_id) REFERENCES mcp_connections(id) ON DELETE CASCADE, - INDEX idx_user (user_id), - INDEX idx_connection (connection_id) -) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci -COMMENT='MCP tool definitions cache'; - -CREATE TABLE mcp_audit_log ( - id INT AUTO_INCREMENT PRIMARY KEY, - user_id VARCHAR(255) NOT NULL COMMENT 'Supabase user ID', - tool_name VARCHAR(255) NOT NULL COMMENT 'Tool that was executed', - conversation_id VARCHAR(255) COMMENT 'Associated conversation ID', - success BOOLEAN NOT NULL COMMENT 'Was execution successful', - error_message TEXT COMMENT 'Error message if failed', - executed_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, - - INDEX idx_user_date (user_id, executed_at DESC), - INDEX idx_tool (tool_name) -) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci -COMMENT='MCP tool execution audit log'; - --- ============================================================================= --- SCHEMA VERSION TRACKING --- ============================================================================= -CREATE TABLE schema_version ( - version INT PRIMARY KEY COMMENT 'Schema version number', - applied_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP COMMENT 'When migration was applied', - description VARCHAR(255) COMMENT 'Migration description' -) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci -COMMENT='Schema version tracking'; - --- Insert initial version -INSERT INTO schema_version (version, description) VALUES - (1, 'Initial schema - providers, models, aliases, capabilities'); diff --git a/backend/migrations/002_global_tiers.sql b/backend/migrations/002_global_tiers.sql deleted file mode 100644 index c4f8e37b..00000000 --- a/backend/migrations/002_global_tiers.sql +++ /dev/null @@ -1,65 +0,0 @@ --- Migration: Convert from per-provider tiers to 5 global tiers --- This migration updates the recommended_models table to support a global tier system --- where only 5 models (one per tier) can be recommended across all providers. - --- Step 1: Drop foreign key constraint (required before dropping index) -ALTER TABLE recommended_models - DROP FOREIGN KEY recommended_models_ibfk_1; - --- Step 2: Drop old per-provider unique constraint -ALTER TABLE recommended_models - DROP INDEX unique_provider_tier; - --- Step 3: Clear existing tier data (will need to be reassigned via admin UI) -TRUNCATE TABLE recommended_models; - --- Step 4: Update tier enum to use tier1-tier5 naming -ALTER TABLE recommended_models - MODIFY COLUMN tier ENUM('tier1', 'tier2', 'tier3', 'tier4', 'tier5') NOT NULL - COMMENT 'Global tier assignment'; - --- Step 5: Add new global tier uniqueness constraint --- This ensures only ONE model can occupy each of the 5 global tier slots -ALTER TABLE recommended_models - ADD UNIQUE KEY unique_global_tier (tier); - --- Step 6: Re-add index on provider_id for foreign key -ALTER TABLE recommended_models - ADD KEY idx_provider (provider_id); - --- Step 7: Recreate foreign key constraint -ALTER TABLE recommended_models - ADD CONSTRAINT recommended_models_ibfk_1 - FOREIGN KEY (provider_id) REFERENCES providers(id) ON DELETE CASCADE; - --- Step 8: Create tier_labels table for UI display customization -CREATE TABLE IF NOT EXISTS tier_labels ( - tier ENUM('tier1', 'tier2', 'tier3', 'tier4', 'tier5') PRIMARY KEY - COMMENT 'Tier identifier', - label VARCHAR(50) NOT NULL - COMMENT 'Display label (e.g., "Elite", "Premium")', - description TEXT - COMMENT 'Tier description for admin UI', - icon VARCHAR(20) - COMMENT 'Icon/emoji for UI display', - display_order INT NOT NULL - COMMENT 'Sort order for UI display', - - INDEX idx_display_order (display_order) -) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci -COMMENT='Customizable tier labels for UI display'; - --- Step 9: Insert default tier labels -INSERT INTO tier_labels (tier, label, description, icon, display_order) VALUES -('tier1', 'Elite', 'Most powerful and capable models', '⭐', 1), -('tier2', 'Premium', 'High-quality professional models', '💎', 2), -('tier3', 'Standard', 'Balanced performance and cost', '🎯', 3), -('tier4', 'Fast', 'Speed-optimized models', '⚡', 4), -('tier5', 'New', 'Latest model additions', '✨', 5); - --- Migration Notes: --- ================ --- 1. All existing tier assignments have been cleared --- 2. Admins must reassign models to the 5 global tiers via the admin UI --- 3. The tier system is now global across all providers (only 5 models total can be in tiers) --- 4. Tier labels can be customized by updating the tier_labels table diff --git a/backend/pkg/auth/local_jwt.go b/backend/pkg/auth/local_jwt.go deleted file mode 100644 index 0b622e61..00000000 --- a/backend/pkg/auth/local_jwt.go +++ /dev/null @@ -1,279 +0,0 @@ -package auth - -import ( - "crypto/rand" - "encoding/base64" - "errors" - "fmt" - "time" - - "github.com/golang-jwt/jwt/v5" - "golang.org/x/crypto/argon2" -) - -// LocalJWTAuth handles local JWT-based authentication -type LocalJWTAuth struct { - SecretKey []byte - AccessTokenExpiry time.Duration // Default: 15 minutes - RefreshTokenExpiry time.Duration // Default: 7 days -} - -// NewLocalJWTAuth creates a new local JWT auth instance -func NewLocalJWTAuth(secretKey string, accessExpiry, refreshExpiry time.Duration) (*LocalJWTAuth, error) { - if secretKey == "" { - return nil, errors.New("JWT secret key cannot be empty") - } - - if accessExpiry == 0 { - accessExpiry = 15 * time.Minute - } - - if refreshExpiry == 0 { - refreshExpiry = 7 * 24 * time.Hour - } - - return &LocalJWTAuth{ - SecretKey: []byte(secretKey), - AccessTokenExpiry: accessExpiry, - RefreshTokenExpiry: refreshExpiry, - }, nil -} - -// JWTClaims represents the JWT token claims -type JWTClaims struct { - UserID string `json:"sub"` - Email string `json:"email"` - Role string `json:"role"` - TokenID string `json:"jti"` // For refresh token tracking - jwt.RegisteredClaims -} - -// GenerateTokens generates both access and refresh tokens -func (a *LocalJWTAuth) GenerateTokens(userID, email, role string) (accessToken, refreshToken string, err error) { - // Generate unique token ID for refresh token - tokenID, err := generateTokenID() - if err != nil { - return "", "", fmt.Errorf("failed to generate token ID: %w", err) - } - - // Access token (short-lived) - accessClaims := JWTClaims{ - UserID: userID, - Email: email, - Role: role, - RegisteredClaims: jwt.RegisteredClaims{ - ExpiresAt: jwt.NewNumericDate(time.Now().Add(a.AccessTokenExpiry)), - IssuedAt: jwt.NewNumericDate(time.Now()), - NotBefore: jwt.NewNumericDate(time.Now()), - Issuer: "claraverse-local", - }, - } - - accessTokenObj := jwt.NewWithClaims(jwt.SigningMethodHS256, accessClaims) - accessToken, err = accessTokenObj.SignedString(a.SecretKey) - if err != nil { - return "", "", fmt.Errorf("failed to sign access token: %w", err) - } - - // Refresh token (long-lived) - refreshClaims := JWTClaims{ - UserID: userID, - Email: email, - Role: role, - TokenID: tokenID, - RegisteredClaims: jwt.RegisteredClaims{ - ExpiresAt: jwt.NewNumericDate(time.Now().Add(a.RefreshTokenExpiry)), - IssuedAt: jwt.NewNumericDate(time.Now()), - NotBefore: jwt.NewNumericDate(time.Now()), - Issuer: "claraverse-local", - }, - } - - refreshTokenObj := jwt.NewWithClaims(jwt.SigningMethodHS256, refreshClaims) - refreshToken, err = refreshTokenObj.SignedString(a.SecretKey) - if err != nil { - return "", "", fmt.Errorf("failed to sign refresh token: %w", err) - } - - return accessToken, refreshToken, nil -} - -// VerifyAccessToken verifies an access token and returns the user -func (a *LocalJWTAuth) VerifyAccessToken(tokenString string) (*User, error) { - token, err := jwt.ParseWithClaims(tokenString, &JWTClaims{}, func(token *jwt.Token) (interface{}, error) { - // Verify signing method - if _, ok := token.Method.(*jwt.SigningMethodHMAC); !ok { - return nil, fmt.Errorf("unexpected signing method: %v", token.Header["alg"]) - } - return a.SecretKey, nil - }) - - if err != nil { - return nil, fmt.Errorf("failed to parse token: %w", err) - } - - if claims, ok := token.Claims.(*JWTClaims); ok && token.Valid { - return &User{ - ID: claims.UserID, - Email: claims.Email, - Role: claims.Role, - }, nil - } - - return nil, errors.New("invalid token") -} - -// VerifyRefreshToken verifies a refresh token and returns claims -func (a *LocalJWTAuth) VerifyRefreshToken(tokenString string) (*JWTClaims, error) { - token, err := jwt.ParseWithClaims(tokenString, &JWTClaims{}, func(token *jwt.Token) (interface{}, error) { - if _, ok := token.Method.(*jwt.SigningMethodHMAC); !ok { - return nil, fmt.Errorf("unexpected signing method: %v", token.Header["alg"]) - } - return a.SecretKey, nil - }) - - if err != nil { - return nil, fmt.Errorf("failed to parse refresh token: %w", err) - } - - if claims, ok := token.Claims.(*JWTClaims); ok && token.Valid { - return claims, nil - } - - return nil, errors.New("invalid refresh token") -} - -// Argon2 password hashing parameters (OWASP recommended) -const ( - argon2Time = 3 // Number of iterations - argon2Memory = 64 * 1024 // 64MB - argon2Threads = 4 // Parallelism - argon2KeyLength = 32 // 32 bytes (256 bits) - saltLength = 16 // 16 bytes salt -) - -// HashPassword hashes a password using Argon2id -func (a *LocalJWTAuth) HashPassword(password string) (string, error) { - // Generate random salt - salt := make([]byte, saltLength) - if _, err := rand.Read(salt); err != nil { - return "", fmt.Errorf("failed to generate salt: %w", err) - } - - // Hash password with Argon2id - hash := argon2.IDKey([]byte(password), salt, argon2Time, argon2Memory, argon2Threads, argon2KeyLength) - - // Encode salt and hash to base64 - saltEncoded := base64.RawStdEncoding.EncodeToString(salt) - hashEncoded := base64.RawStdEncoding.EncodeToString(hash) - - // Format: argon2id$salt$hash - return fmt.Sprintf("argon2id$%s$%s", saltEncoded, hashEncoded), nil -} - -// VerifyPassword verifies a password against an Argon2id hash -func (a *LocalJWTAuth) VerifyPassword(hashedPassword, password string) (bool, error) { - // Parse hash format: argon2id$salt$hash - parts := []byte(hashedPassword) - if len(parts) < 10 || string(parts[:9]) != "argon2id$" { - return false, fmt.Errorf("invalid hash format: missing argon2id prefix") - } - - // Split by $ delimiter - hashParts := []string{} - start := 9 // Skip "argon2id$" - for i := start; i < len(parts); i++ { - if parts[i] == '$' { - hashParts = append(hashParts, string(parts[start:i])) - start = i + 1 - } - } - // Add the last part - if start < len(parts) { - hashParts = append(hashParts, string(parts[start:])) - } - - if len(hashParts) != 2 { - return false, fmt.Errorf("invalid hash format: expected 2 parts, got %d", len(hashParts)) - } - - saltEncoded := hashParts[0] - hashEncoded := hashParts[1] - - // Decode salt and hash from base64 - salt, err := base64.RawStdEncoding.DecodeString(saltEncoded) - if err != nil { - return false, fmt.Errorf("failed to decode salt: %w", err) - } - - expectedHash, err := base64.RawStdEncoding.DecodeString(hashEncoded) - if err != nil { - return false, fmt.Errorf("failed to decode hash: %w", err) - } - - // Hash provided password with same salt - actualHash := argon2.IDKey([]byte(password), salt, argon2Time, argon2Memory, argon2Threads, argon2KeyLength) - - // Constant-time comparison - if len(actualHash) != len(expectedHash) { - return false, nil - } - - var equal byte = 0 - for i := 0; i < len(actualHash); i++ { - equal |= actualHash[i] ^ expectedHash[i] - } - - return equal == 0, nil -} - -// generateTokenID generates a random token ID for refresh tokens -func generateTokenID() (string, error) { - b := make([]byte, 32) - if _, err := rand.Read(b); err != nil { - return "", err - } - return base64.URLEncoding.EncodeToString(b), nil -} - -// ValidatePassword checks if password meets requirements -func ValidatePassword(password string) error { - if len(password) < 8 { - return errors.New("password must be at least 8 characters long") - } - - var ( - hasUpper bool - hasLower bool - hasNumber bool - hasSpecial bool - ) - - for _, char := range password { - switch { - case 'A' <= char && char <= 'Z': - hasUpper = true - case 'a' <= char && char <= 'z': - hasLower = true - case '0' <= char && char <= '9': - hasNumber = true - case char == '!' || char == '@' || char == '#' || char == '$' || char == '%' || char == '^' || char == '&' || char == '*': - hasSpecial = true - } - } - - if !hasUpper { - return errors.New("password must contain at least one uppercase letter") - } - if !hasLower { - return errors.New("password must contain at least one lowercase letter") - } - if !hasNumber { - return errors.New("password must contain at least one number") - } - if !hasSpecial { - return errors.New("password must contain at least one special character (!@#$%^&*)") - } - - return nil -} diff --git a/backend/pkg/auth/supabase.go b/backend/pkg/auth/supabase.go deleted file mode 100644 index 7f7aa6c5..00000000 --- a/backend/pkg/auth/supabase.go +++ /dev/null @@ -1,79 +0,0 @@ -package auth - -import ( - "encoding/json" - "fmt" - "io" - "net/http" - "strings" -) - -// SupabaseAuth handles Supabase authentication -type SupabaseAuth struct { - URL string - Key string -} - -// NewSupabaseAuth creates a new Supabase auth instance -func NewSupabaseAuth(url, key string) *SupabaseAuth { - return &SupabaseAuth{ - URL: url, - Key: key, - } -} - -// User represents an authenticated user -type User struct { - ID string `json:"id"` - Email string `json:"email"` - Role string `json:"role"` -} - -// VerifyToken verifies a Supabase JWT token and returns the user -func (s *SupabaseAuth) VerifyToken(token string) (*User, error) { - if s.URL == "" || s.Key == "" { - return nil, fmt.Errorf("supabase not configured") - } - - // Call Supabase API to verify token - req, err := http.NewRequest("GET", s.URL+"/auth/v1/user", nil) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Authorization", "Bearer "+token) - req.Header.Set("apikey", s.Key) - - client := &http.Client{} - resp, err := client.Do(req) - if err != nil { - return nil, fmt.Errorf("failed to verify token: %w", err) - } - defer resp.Body.Close() - - if resp.StatusCode != http.StatusOK { - body, _ := io.ReadAll(resp.Body) - return nil, fmt.Errorf("token verification failed: %s", string(body)) - } - - var user User - if err := json.NewDecoder(resp.Body).Decode(&user); err != nil { - return nil, fmt.Errorf("failed to decode user: %w", err) - } - - return &user, nil -} - -// ExtractToken extracts the bearer token from Authorization header -func ExtractToken(authHeader string) (string, error) { - if authHeader == "" { - return "", fmt.Errorf("authorization header is empty") - } - - parts := strings.Split(authHeader, " ") - if len(parts) != 2 || parts[0] != "Bearer" { - return "", fmt.Errorf("invalid authorization header format") - } - - return parts[1], nil -} diff --git a/backend/providers.example.json b/backend/providers.example.json deleted file mode 100644 index 51414668..00000000 --- a/backend/providers.example.json +++ /dev/null @@ -1,102 +0,0 @@ -{ - "providers": [ - { - "name": "OpenAI", - "base_url": "https://api.openai.com/v1", - "api_key": "sk-your-openai-api-key-here", - "enabled": true, - "filters": [ - { - "pattern": "gpt-4o", - "action": "include", - "priority": 20 - }, - { - "pattern": "gpt-4o-mini", - "action": "include", - "priority": 15 - }, - { - "pattern": "gpt-3.5-turbo", - "action": "include", - "priority": 10 - } - ] - }, - { - "name": "Anthropic", - "base_url": "https://api.anthropic.com/v1", - "api_key": "your-anthropic-api-key-here", - "enabled": true, - "filters": [ - { - "pattern": "claude-3-*", - "action": "include", - "priority": 10 - } - ] - }, - { - "name": "Z.AI", - "base_url": "https://api.z.ai/api/coding/paas/v4", - "api_key": "your-z-ai-api-key-here", - "enabled": true, - "filters": [ - { - "pattern": "glm-4*", - "action": "include", - "priority": 10 - } - ] - }, - { - "name": "Custom Provider", - "base_url": "https://your-custom-openai-compatible-api.com/v1", - "api_key": "your-api-key-here", - "enabled": false, - "filters": [ - { - "pattern": "*", - "action": "include", - "priority": 0 - } - ] - }, - { - "name": "Groq", - "base_url": "https://api.groq.com/openai/v1", - "api_key": "gsk_your-groq-api-key-here", - "enabled": true, - "audio_only": true, - "filters": [] - }, - { - "name": "OpenAI DALL-E", - "base_url": "https://api.openai.com", - "api_key": "sk-your-openai-api-key-here", - "enabled": false, - "image_only": true, - "default_model": "dall-e-3", - "filters": [] - }, - { - "name": "Together AI Images", - "base_url": "https://api.together.xyz", - "api_key": "your-together-api-key-here", - "enabled": false, - "image_only": true, - "default_model": "black-forest-labs/FLUX.1-schnell-Free", - "filters": [] - }, - { - "name": "Replicate Images", - "base_url": "https://api.replicate.com", - "api_key": "r8_your-replicate-api-key-here", - "enabled": false, - "image_only": true, - "default_model": "stability-ai/sdxl", - "filters": [] - } - ] -} - diff --git a/backend/run-tests.sh b/backend/run-tests.sh deleted file mode 100644 index 8a2189e9..00000000 --- a/backend/run-tests.sh +++ /dev/null @@ -1,89 +0,0 @@ -#!/bin/sh -# ClaraVerse Backend Test Runner -# This script runs all unit tests and integration tests -# Exit code 0 = all tests passed, non-zero = failure - -set -e - -echo "==========================================" -echo " ClaraVerse Backend Test Suite" -echo "==========================================" -echo "" - -# Track test failures -FAILURES=0 - -# Function to run tests for a package -run_package_tests() { - local package=$1 - local description=$2 - - echo "----------------------------------------" - echo "Testing: $description" - echo "Package: $package" - echo "----------------------------------------" - - if go test -v -race -timeout 60s "$package"; then - echo "✅ PASSED: $description" - else - echo "❌ FAILED: $description" - FAILURES=$((FAILURES + 1)) - fi - echo "" -} - -# Core packages -echo "=== Core Package Tests ===" -run_package_tests "./internal/database" "Database" -run_package_tests "./internal/models/..." "Models" - -# Service tests -echo "=== Service Tests ===" -run_package_tests "./internal/services/..." "Services" - -# Handler tests -echo "=== Handler Tests ===" -run_package_tests "./internal/handlers/..." "Handlers" - -# Tool tests (includes all file tools) -echo "=== Tool Tests ===" -run_package_tests "./internal/tools/..." "Tools" - -# Execution tests (workflow, variable executor) -echo "=== Execution Tests ===" -run_package_tests "./internal/execution/..." "Execution" - -# File cache tests -echo "=== File Cache Tests ===" -run_package_tests "./internal/filecache/..." "File Cache" - -# Audio service tests -echo "=== Audio Service Tests ===" -run_package_tests "./internal/audio/..." "Audio Service" - -# Vision service tests -echo "=== Vision Service Tests ===" -run_package_tests "./internal/vision/..." "Vision Service" - -# Preflight tests -echo "=== Preflight Tests ===" -run_package_tests "./internal/preflight/..." "Preflight Checks" - -# Integration tests -echo "=== Integration Tests ===" -run_package_tests "./tests/..." "Integration Tests" - -# Summary -echo "==========================================" -echo " Test Summary" -echo "==========================================" - -if [ $FAILURES -eq 0 ]; then - echo "✅ ALL TESTS PASSED" - echo "" - exit 0 -else - echo "❌ $FAILURES TEST SUITE(S) FAILED" - echo "" - exit 1 -fi diff --git a/backend/scripts/benchmark_test.py b/backend/scripts/benchmark_test.py deleted file mode 100755 index 04c42471..00000000 --- a/backend/scripts/benchmark_test.py +++ /dev/null @@ -1,637 +0,0 @@ -#!/usr/bin/env python3 - -""" -Comprehensive Benchmark Test Suite for Workflow Generator - -This script tests the workflow generator with increasingly complex scenarios -to evaluate quality, schema compliance, and handling of edge cases. - -Usage: - python3 benchmark_test.py # Run all benchmark tests - python3 benchmark_test.py --category complexity # Run specific category - python3 benchmark_test.py --quick # Run quick subset -""" - -import json -import requests -import time -import argparse -from typing import Dict, List, Any, Tuple -from datetime import datetime -import os - -# Configuration -API_BASE_URL = "http://localhost:3001" -USE_DEV_ENDPOINTS = True -DEV_USER_ID = "dev-test-user" - -# Available models -AVAILABLE_MODELS = [ - {"id": "Qwen/Qwen3-Coder-480B-A35B-Instruct-FP8-TEE", "name": "Qwen3 Coder 480B (Recommended)"}, - {"id": "Qwen/Qwen3-235B-A22B", "name": "Qwen3 235B (Advanced reasoning)"}, - {"id": "deepseek-ai/DeepSeek-V3.2-TEE", "name": "DeepSeek V3.2"}, - {"id": "zai-org/GLM-4.7-TEE", "name": "GLM 4.7"}, - {"id": "Qwen/Qwen3-VL-235B-A22B-Instruct", "name": "Qwen3 VL 235B"}, - {"id": "MiniMaxAI/MiniMax-M2.1-TEE", "name": "MiniMax M2.1"}, -] - -SELECTED_MODEL = None # Will be set by CLI arg or auto - -# ANSI color codes -class Colors: - RED = '\033[0;31m' - GREEN = '\033[0;32m' - YELLOW = '\033[1;33m' - BLUE = '\033[0;34m' - CYAN = '\033[0;36m' - MAGENTA = '\033[0;35m' - NC = '\033[0m' - -# Benchmark Test Categories -BENCHMARK_TESTS = { - "simple": { - "name": "Simple Workflows (1-2 blocks)", - "tests": [ - { - "name": "Single Search", - "prompt": "Create a workflow that searches for AI news", - "expected_blocks": 2, - "expected_structured": 1, - "complexity": 1 - }, - { - "name": "Text Generation", - "prompt": "Write a blog post about renewable energy", - "expected_blocks": 2, - "expected_structured": 0, - "complexity": 1 - } - ] - }, - "moderate": { - "name": "Moderate Workflows (3-5 blocks)", - "tests": [ - { - "name": "Search and Summarize", - "prompt": "Search for AI news, extract the top 3 articles with titles and URLs, then write a summary", - "expected_blocks": 4, - "expected_structured": 1, - "complexity": 2 - }, - { - "name": "Data Extraction Pipeline", - "prompt": "Search for Python tutorials, extract titles, URLs, and difficulty levels, then filter only beginner tutorials", - "expected_blocks": 4, - "expected_structured": 2, - "complexity": 2 - }, - { - "name": "Multi-Tool Workflow", - "prompt": "Get current time, search for today's tech news, and extract article summaries with timestamps", - "expected_blocks": 4, - "expected_structured": 1, - "complexity": 2 - } - ] - }, - "complex": { - "name": "Complex Workflows (6-10 blocks)", - "tests": [ - { - "name": "Research & Report Generator", - "prompt": "Create a comprehensive research workflow: 1) Search for AI breakthroughs in 2026, 2) Extract key findings with sources, 3) Search for expert opinions, 4) Combine all data into structured format, 5) Generate executive summary, 6) Create detailed report", - "expected_blocks": 8, - "expected_structured": 3, - "complexity": 3 - }, - { - "name": "Multi-Source Aggregator", - "prompt": "Build a news aggregator that: searches GitHub for trending repos, searches web for tech news, searches for AI research papers, extracts structured data from each source, combines all results, ranks by relevance, and generates a daily digest", - "expected_blocks": 10, - "expected_structured": 4, - "complexity": 3 - }, - { - "name": "Conditional Branching", - "prompt": "Create a weather alert system: get current weather, check if temperature is above 30C OR if it's raining, if hot then search for heat safety tips, if raining then search for indoor activities, combine results and send summary", - "expected_blocks": 7, - "expected_structured": 2, - "complexity": 3 - } - ] - }, - "edge_cases": { - "name": "Edge Cases & Schema Challenges", - "tests": [ - { - "name": "Array Root Schema", - "prompt": "Return a simple array of product IDs directly, not wrapped in an object", - "expected_blocks": 2, - "expected_structured": 1, - "complexity": 2, - "expects_array_schema": True - }, - { - "name": "Nested Arrays", - "prompt": "Search for restaurants and return a list where each restaurant has an array of reviews, each review has an array of tags", - "expected_blocks": 3, - "expected_structured": 1, - "complexity": 2 - }, - { - "name": "Mixed Data Types", - "prompt": "Create a workflow that returns structured data with strings, numbers, booleans, arrays, and nested objects for a product catalog", - "expected_blocks": 3, - "expected_structured": 1, - "complexity": 2 - }, - { - "name": "Large Schema", - "prompt": "Extract detailed user profiles with 15+ fields including: name, email, age, address (street, city, state, zip), preferences (array), social media links (object), purchase history (array of objects with date, amount, items), subscription status (boolean), and account metadata", - "expected_blocks": 3, - "expected_structured": 1, - "complexity": 2 - }, - { - "name": "Multiple Structured Outputs", - "prompt": "Search for tech companies, extract structured company data (name, founded, employees), then for each company search for recent news and extract structured article data (title, date, summary), then aggregate everything", - "expected_blocks": 5, - "expected_structured": 3, - "complexity": 3 - } - ] - }, - "performance": { - "name": "Performance & Scale Tests", - "tests": [ - { - "name": "10-Block Chain", - "prompt": "Create a 10-step workflow: 1) search topic A, 2) extract data from A, 3) search topic B, 4) extract data from B, 5) search topic C, 6) extract data from C, 7) merge A+B, 8) merge result with C, 9) analyze trends, 10) generate final report", - "expected_blocks": 12, - "expected_structured": 5, - "complexity": 4 - }, - { - "name": "Parallel Processing", - "prompt": "Create parallel workflows that simultaneously: search news, search GitHub, search YouTube, search Twitter, search Reddit, then combine all results", - "expected_blocks": 7, - "expected_structured": 5, - "complexity": 3 - } - ] - }, - "real_world": { - "name": "Real-World Use Cases", - "tests": [ - { - "name": "Content Curation Bot", - "prompt": "Build a content curation bot that searches for AI/ML articles, filters by quality indicators (source credibility, engagement), extracts key points, generates summaries, categorizes by topic, and creates a newsletter format", - "expected_blocks": 8, - "expected_structured": 3, - "complexity": 3 - }, - { - "name": "Market Research Tool", - "prompt": "Create a market research workflow: search for competitor products, extract pricing and features, search for customer reviews, analyze sentiment, compare with our product, identify gaps, generate competitive analysis report", - "expected_blocks": 9, - "expected_structured": 4, - "complexity": 3 - }, - { - "name": "Automated Report Generator", - "prompt": "Build an automated weekly report: get current date, search for industry news from past 7 days, extract key events with dates, search for market data, calculate trends, generate charts description, write executive summary, compile full report with sections", - "expected_blocks": 10, - "expected_structured": 3, - "complexity": 3 - }, - { - "name": "E-commerce Analytics", - "prompt": "Create product analytics workflow: search for product reviews, extract ratings and feedback, calculate average rating, identify common complaints, search for competitor products, compare features and prices, generate improvement recommendations", - "expected_blocks": 9, - "expected_structured": 4, - "complexity": 3 - } - ] - } -} - -def print_header(): - print(f"{Colors.MAGENTA}╔════════════════════════════════════════════════════════════════╗{Colors.NC}") - print(f"{Colors.MAGENTA}║{Colors.NC} 🏁 ClaraVerse Workflow Generator Benchmark Suite {Colors.MAGENTA}║{Colors.NC}") - print(f"{Colors.MAGENTA}╚════════════════════════════════════════════════════════════════╝{Colors.NC}") - print() - -def print_section(title: str): - print(f"\n{Colors.YELLOW}▶ {title}{Colors.NC}") - print("━" * 64) - -def select_model() -> str: - """Interactive model selection.""" - global SELECTED_MODEL - - print(f"\n{Colors.YELLOW}▶ Model Selection{Colors.NC}") - print("━" * 64) - print(f"{Colors.CYAN}Available models:{Colors.NC}\n") - - for i, model in enumerate(AVAILABLE_MODELS, 1): - print(f" {Colors.YELLOW}{i}.{Colors.NC} {model['name']}") - print(f" {Colors.BLUE}{model['id']}{Colors.NC}") - print() - - print(f" {Colors.YELLOW}0.{Colors.NC} Auto (let the generator choose)") - print() - - while True: - try: - choice = input(f"{Colors.CYAN}Select a model (0-{len(AVAILABLE_MODELS)}): {Colors.NC}").strip() - choice_num = int(choice) - - if choice_num == 0: - SELECTED_MODEL = None - print(f"{Colors.GREEN}✅ Using auto model selection{Colors.NC}") - return None - elif 1 <= choice_num <= len(AVAILABLE_MODELS): - SELECTED_MODEL = AVAILABLE_MODELS[choice_num - 1]['id'] - print(f"{Colors.GREEN}✅ Selected: {AVAILABLE_MODELS[choice_num - 1]['name']}{Colors.NC}") - return SELECTED_MODEL - else: - print(f"{Colors.RED}Invalid choice. Please enter 0-{len(AVAILABLE_MODELS)}{Colors.NC}") - except ValueError: - print(f"{Colors.RED}Invalid input. Please enter a number{Colors.NC}") - except KeyboardInterrupt: - print(f"\n{Colors.YELLOW}Using auto selection{Colors.NC}") - return None - -def check_backend() -> bool: - """Check if backend is reachable.""" - try: - response = requests.get(f"{API_BASE_URL}/health", timeout=5) - return response.status_code == 200 - except: - return False - -def create_test_agent(name: str) -> str: - """Create a test agent and return its ID.""" - try: - response = requests.post( - f"{API_BASE_URL}/api/dev-test/agents" if USE_DEV_ENDPOINTS else f"{API_BASE_URL}/api/agents", - headers={"Content-Type": "application/json"}, - json={"name": name, "description": "Benchmark test agent"}, - timeout=10 - ) - if response.status_code in [200, 201]: - return response.json().get('id') - except: - pass - return None - -def generate_workflow(agent_id: str, prompt: str) -> Dict: - """Generate a workflow for the given prompt.""" - try: - # Build request body - request_body = { - "user_message": prompt, - "conversation_id": f"benchmark-{int(time.time())}" - } - - # Add model_id if one was selected - if SELECTED_MODEL: - request_body["model_id"] = SELECTED_MODEL - - response = requests.post( - f"{API_BASE_URL}/api/dev-test/agents/{agent_id}/generate-workflow" if USE_DEV_ENDPOINTS else f"{API_BASE_URL}/api/agents/{agent_id}/generate-workflow", - headers={"Content-Type": "application/json"}, - json=request_body, - timeout=120 # Allow up to 2 minutes for complex workflows - ) - - if response.status_code == 200: - return response.json() - else: - return {"error": f"Status {response.status_code}: {response.text}"} - except Exception as e: - return {"error": str(e)} - -def cleanup_agent(agent_id: str): - """Delete the test agent.""" - try: - if USE_DEV_ENDPOINTS: - requests.delete(f"{API_BASE_URL}/api/dev-test/agents/{agent_id}", timeout=5) - else: - requests.delete(f"{API_BASE_URL}/api/agents/{agent_id}", timeout=5) - except: - pass - -def analyze_workflow_quality(workflow_data: Dict, test_config: Dict) -> Dict: - """Comprehensive quality analysis with scoring.""" - - if not workflow_data or 'workflow' not in workflow_data: - return { - "score": 0, - "errors": ["Invalid workflow data"], - "metrics": {} - } - - workflow = workflow_data['workflow'] - blocks = workflow.get('blocks', []) - - # Basic metrics - metrics = { - "total_blocks": len(blocks), - "llm_blocks": len([b for b in blocks if b.get('type') == 'llm_inference']), - "variable_blocks": len([b for b in blocks if b.get('type') == 'variable']), - "structured_blocks": len([b for b in blocks if b.get('type') == 'llm_inference' and b.get('config', {}).get('outputFormat') == 'json']), - "connections": len(workflow.get('connections', [])), - } - - # Schema analysis - schema_issues = [] - complete_required_arrays = True - missing_additional_props = 0 - has_array_schema = False - - for i, block in enumerate(blocks): - if block.get('type') == 'llm_inference': - config = block.get('config', {}) - if config.get('outputFormat') == 'json': - schema = config.get('outputSchema', {}) - schema_type = schema.get('type', 'object') - - if schema_type == 'array': - has_array_schema = True - items = schema.get('items', {}) - if items.get('type') == 'object': - props = items.get('properties', {}) - required = items.get('required', []) - for prop in props.keys(): - if prop not in required: - complete_required_arrays = False - schema_issues.append(f"Block {i}: Array item property '{prop}' not in required") - - elif schema_type == 'object': - props = schema.get('properties', {}) - required = schema.get('required', []) - for prop in props.keys(): - if prop not in required: - complete_required_arrays = False - schema_issues.append(f"Block {i}: Property '{prop}' not in required") - - if schema.get('additionalProperties') != False: - missing_additional_props += 1 - - # Calculate score - score = 100 - - # Block count accuracy (20 points) - expected_blocks = test_config.get('expected_blocks', 0) - if expected_blocks > 0: - block_diff = abs(metrics['total_blocks'] - expected_blocks) - if block_diff == 0: - score += 0 # Perfect - elif block_diff <= 2: - score -= 5 - else: - score -= 15 - - # Structured output usage (20 points) - expected_structured = test_config.get('expected_structured', 0) - if expected_structured > 0: - structured_diff = abs(metrics['structured_blocks'] - expected_structured) - if structured_diff == 0: - score += 0 - elif structured_diff <= 1: - score -= 5 - else: - score -= 15 - - # Schema compliance (40 points) - if not complete_required_arrays: - score -= 20 - - if missing_additional_props > 0: - score -= min(20, missing_additional_props * 5) - - # Array schema handling (20 points bonus for edge cases) - if test_config.get('expects_array_schema', False): - if has_array_schema: - score += 10 - else: - score -= 20 - schema_issues.append("Expected array schema at root level but got object") - - score = max(0, min(100, score)) - - return { - "score": score, - "metrics": metrics, - "schema_issues": schema_issues, - "complete_required_arrays": complete_required_arrays, - "missing_additional_props": missing_additional_props, - "has_array_schema": has_array_schema - } - -def run_benchmark_test(category: str, test: Dict, test_num: int, total: int) -> Tuple[Dict, float]: - """Run a single benchmark test and return results.""" - - print(f"\n{Colors.CYAN}{'═' * 64}{Colors.NC}") - print(f"{Colors.CYAN}Test {test_num}/{total}: {test['name']}{Colors.NC}") - print(f"{Colors.CYAN}Complexity: {'⭐' * test['complexity']}{Colors.NC}") - print(f"{Colors.CYAN}{'═' * 64}{Colors.NC}") - print(f"\n{Colors.BLUE}Prompt:{Colors.NC} {test['prompt'][:100]}...") - - # Create agent - agent_id = create_test_agent(f"Benchmark {test['name']}") - if not agent_id: - return {"error": "Failed to create agent", "score": 0}, 0.0 - - # Generate workflow (timed) - start_time = time.time() - workflow_data = generate_workflow(agent_id, test['prompt']) - generation_time = time.time() - start_time - - # Clean up - cleanup_agent(agent_id) - - # Check for errors - if 'error' in workflow_data: - print(f"{Colors.RED}❌ Generation failed: {workflow_data['error'][:100]}{Colors.NC}") - return {"error": workflow_data['error'], "score": 0}, generation_time - - # Analyze quality - analysis = analyze_workflow_quality(workflow_data, test) - - # Print results - print(f"\n{Colors.YELLOW}Results:{Colors.NC}") - print(f" ⏱️ Generation time: {generation_time:.2f}s") - print(f" 📊 Blocks: {analysis['metrics']['total_blocks']} (expected: {test.get('expected_blocks', '?')})") - print(f" 🔧 Structured outputs: {analysis['metrics']['structured_blocks']} (expected: {test.get('expected_structured', '?')})") - - if analysis['schema_issues']: - print(f" {Colors.YELLOW}⚠️ Schema issues:{Colors.NC}") - for issue in analysis['schema_issues'][:3]: - print(f" • {issue}") - if len(analysis['schema_issues']) > 3: - print(f" • ... and {len(analysis['schema_issues']) - 3} more") - - score = analysis['score'] - if score >= 90: - print(f" {Colors.GREEN}🏆 Score: {score}/100 (Excellent){Colors.NC}") - elif score >= 70: - print(f" {Colors.YELLOW}⭐ Score: {score}/100 (Good){Colors.NC}") - else: - print(f" {Colors.RED}⚠️ Score: {score}/100 (Needs Improvement){Colors.NC}") - - return analysis, generation_time - -def run_benchmark_suite(categories: List[str] = None, quick: bool = False): - """Run the full benchmark suite.""" - - print_header() - - if not check_backend(): - print(f"{Colors.RED}❌ Backend not reachable at {API_BASE_URL}{Colors.NC}") - return - - print(f"{Colors.GREEN}✅ Backend is running{Colors.NC}") - - # Let user select model (if not already set via CLI) - if SELECTED_MODEL is None: - select_model() - - # Select categories - if categories: - test_categories = {k: v for k, v in BENCHMARK_TESTS.items() if k in categories} - else: - test_categories = BENCHMARK_TESTS - - # Quick mode: only run first test from each category - if quick: - print(f"{Colors.YELLOW}🚀 Quick mode: Running first test from each category{Colors.NC}") - for cat in test_categories.values(): - cat['tests'] = cat['tests'][:1] - - # Count total tests - total_tests = sum(len(cat['tests']) for cat in test_categories.values()) - - print(f"\n{Colors.BLUE}📋 Running {total_tests} benchmark tests across {len(test_categories)} categories{Colors.NC}") - - # Results tracking - all_results = {} - test_counter = 0 - - # Create output directory - os.makedirs("benchmark-results", exist_ok=True) - timestamp = datetime.now().strftime("%Y%m%d_%H%M%S") - - # Run tests - for cat_key, category in test_categories.items(): - print(f"\n{Colors.MAGENTA}{'═' * 64}{Colors.NC}") - print(f"{Colors.MAGENTA}Category: {category['name']}{Colors.NC}") - print(f"{Colors.MAGENTA}{'═' * 64}{Colors.NC}") - - cat_results = [] - - for test in category['tests']: - test_counter += 1 - result, gen_time = run_benchmark_test(cat_key, test, test_counter, total_tests) - - cat_results.append({ - "name": test['name'], - "prompt": test['prompt'], - "complexity": test['complexity'], - "generation_time": gen_time, - "score": result.get('score', 0), - "metrics": result.get('metrics', {}), - "issues": result.get('schema_issues', []) - }) - - time.sleep(2) # Avoid rate limiting - - all_results[cat_key] = { - "category_name": category['name'], - "results": cat_results - } - - # Generate summary - print(f"\n{Colors.MAGENTA}{'═' * 64}{Colors.NC}") - print(f"{Colors.MAGENTA}Benchmark Summary{Colors.NC}") - print(f"{Colors.MAGENTA}{'═' * 64}{Colors.NC}\n") - - total_score = 0 - total_time = 0 - test_count = 0 - - for cat_key, cat_data in all_results.items(): - cat_scores = [r['score'] for r in cat_data['results'] if 'score' in r] - cat_times = [r['generation_time'] for r in cat_data['results'] if 'generation_time' in r] - - if cat_scores: - avg_score = sum(cat_scores) / len(cat_scores) - avg_time = sum(cat_times) / len(cat_times) - - print(f"{Colors.CYAN}{cat_data['category_name']}:{Colors.NC}") - print(f" Tests: {len(cat_data['results'])}") - print(f" Avg Score: {avg_score:.1f}/100") - print(f" Avg Time: {avg_time:.2f}s") - print() - - total_score += sum(cat_scores) - total_time += sum(cat_times) - test_count += len(cat_scores) - - if test_count > 0: - overall_avg = total_score / test_count - overall_time = total_time / test_count - - print(f"{Colors.BLUE}Overall Results:{Colors.NC}") - print(f" Total tests: {test_count}") - print(f" Average score: {overall_avg:.1f}/100") - print(f" Average generation time: {overall_time:.2f}s") - print() - - if overall_avg >= 90: - print(f"{Colors.GREEN}🏆 Overall Grade: EXCELLENT{Colors.NC}") - print(" Workflow generator is production-ready!") - elif overall_avg >= 75: - print(f"{Colors.YELLOW}⭐ Overall Grade: GOOD{Colors.NC}") - print(" Workflow generator performs well with minor issues.") - elif overall_avg >= 60: - print(f"{Colors.YELLOW}📊 Overall Grade: FAIR{Colors.NC}") - print(" Workflow generator needs improvements in some areas.") - else: - print(f"{Colors.RED}⚠️ Overall Grade: NEEDS WORK{Colors.NC}") - print(" Significant improvements needed.") - - # Save results - output_file = f"benchmark-results/benchmark_{timestamp}.json" - with open(output_file, 'w') as f: - json.dump({ - "timestamp": timestamp, - "total_tests": test_count, - "average_score": overall_avg if test_count > 0 else 0, - "average_time": overall_time if test_count > 0 else 0, - "categories": all_results - }, f, indent=2) - - print(f"\n{Colors.GREEN}✅ Detailed results saved to: {output_file}{Colors.NC}") - -def main(): - global SELECTED_MODEL - - parser = argparse.ArgumentParser(description="Benchmark the workflow generator") - parser.add_argument("--category", type=str, help="Run specific category (simple, moderate, complex, edge_cases, performance, real_world)") - parser.add_argument("--quick", action="store_true", help="Quick mode: Run only first test from each category") - parser.add_argument("--model", type=str, help="Model ID to use (e.g., Qwen/Qwen3-Coder-480B-A35B-Instruct-FP8-TEE)") - - args = parser.parse_args() - - # Set model if provided - if args.model: - SELECTED_MODEL = args.model - print(f"{Colors.CYAN}Using model: {args.model}{Colors.NC}\n") - - categories = [args.category] if args.category else None - run_benchmark_suite(categories=categories, quick=args.quick) - -if __name__ == "__main__": - main() diff --git a/backend/scripts/migrate_legacy_users.go b/backend/scripts/migrate_legacy_users.go deleted file mode 100644 index d9b2e4f1..00000000 --- a/backend/scripts/migrate_legacy_users.go +++ /dev/null @@ -1,314 +0,0 @@ -package main - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "context" - "encoding/json" - "flag" - "fmt" - "io" - "log" - "net/http" - "os" - "time" - - "github.com/joho/godotenv" - "go.mongodb.org/mongo-driver/bson" -) - -// SupabaseUser represents a user from Supabase Auth -type SupabaseUser struct { - ID string `json:"id"` - Email string `json:"email"` - CreatedAt time.Time `json:"created_at"` -} - -// SupabaseUsersResponse represents the response from Supabase Admin API -type SupabaseUsersResponse struct { - Users []SupabaseUser `json:"users"` -} - -// MigrationStats tracks migration statistics -type MigrationStats struct { - TotalSupabaseUsers int - AlreadyMigrated int - UpdatedToLegacy int - CreatedAsLegacy int - Errors int -} - -func main() { - // Parse command line flags - dryRun := flag.Bool("dry-run", false, "Run in dry-run mode (no changes made)") - flag.Parse() - - fmt.Println("==============================================") - fmt.Println("🔄 Legacy User Migration Script v2") - fmt.Println("==============================================") - fmt.Println() - if *dryRun { - fmt.Println("🔍 DRY RUN MODE - No changes will be made") - fmt.Println() - } - fmt.Println("This script will:") - fmt.Println(" 1. Fetch all users from Supabase Auth") - fmt.Println(" 2. For each user:") - fmt.Println(" - If exists in MongoDB → update to legacy_unlimited") - fmt.Println(" - If not exists → create new doc with legacy_unlimited") - fmt.Println() - - // Load environment - if err := godotenv.Load("../.env"); err != nil { - log.Printf("⚠️ No .env file found: %v (using environment variables)", err) - } - - // Get required environment variables - mongoURI := os.Getenv("MONGODB_URI") - if mongoURI == "" { - log.Fatal("❌ MONGODB_URI environment variable is required") - } - - supabaseURL := os.Getenv("SUPABASE_URL") - if supabaseURL == "" { - log.Fatal("❌ SUPABASE_URL environment variable is required") - } - - // Need service role key for admin API - supabaseServiceKey := os.Getenv("SUPABASE_SERVICE_ROLE_KEY") - if supabaseServiceKey == "" { - // Fall back to regular key (might work if it's actually the service key) - supabaseServiceKey = os.Getenv("SUPABASE_KEY") - if supabaseServiceKey == "" { - log.Fatal("❌ SUPABASE_SERVICE_ROLE_KEY environment variable is required") - } - fmt.Println("⚠️ Using SUPABASE_KEY (ensure it's the service role key for admin access)") - } - - fmt.Println("📋 Configuration:") - fmt.Printf(" MongoDB URI: %s\n", maskURI(mongoURI)) - fmt.Printf(" Supabase URL: %s\n", supabaseURL) - fmt.Println() - - // Connect to MongoDB - fmt.Println("🔗 Connecting to MongoDB...") - db, err := database.NewMongoDB(mongoURI) - if err != nil { - log.Fatalf("❌ Failed to connect to MongoDB: %v", err) - } - defer db.Close(context.Background()) - fmt.Println("✅ Connected to MongoDB") - fmt.Println() - - ctx := context.Background() - - // Fetch all users from Supabase - fmt.Println("🔗 Fetching users from Supabase Auth...") - supabaseUsers, err := fetchSupabaseUsers(supabaseURL, supabaseServiceKey) - if err != nil { - log.Fatalf("❌ Failed to fetch Supabase users: %v", err) - } - fmt.Printf("✅ Found %d users in Supabase Auth\n", len(supabaseUsers)) - fmt.Println() - - if len(supabaseUsers) == 0 { - fmt.Println("✅ No users to migrate!") - return - } - - // Show sample of users - fmt.Println("📋 Sample of users (first 5):") - for i, user := range supabaseUsers { - if i >= 5 { - break - } - fmt.Printf(" - %s (Created: %s)\n", user.Email, user.CreatedAt.Format("2006-01-02 15:04:05")) - } - fmt.Println() - - // Process migration - stats := MigrationStats{TotalSupabaseUsers: len(supabaseUsers)} - usersCollection := db.Database().Collection("users") - now := time.Now() - - fmt.Println("🔨 Processing users...") - fmt.Println() - - for _, supabaseUser := range supabaseUsers { - // Check if user exists in MongoDB - var existingUser models.User - err := usersCollection.FindOne(ctx, bson.M{"supabaseUserId": supabaseUser.ID}).Decode(&existingUser) - - if err == nil { - // User exists in MongoDB - if existingUser.SubscriptionTier == models.TierLegacyUnlimited { - // Already migrated - stats.AlreadyMigrated++ - continue - } - - // Update existing user to legacy_unlimited - if *dryRun { - fmt.Printf(" [DRY RUN] Would UPDATE: %s → legacy_unlimited (current: %s)\n", - supabaseUser.Email, existingUser.SubscriptionTier) - } else { - update := bson.M{ - "$set": bson.M{ - "subscriptionTier": models.TierLegacyUnlimited, - "subscriptionStatus": models.SubStatusActive, - "migratedToLegacyAt": now, - }, - } - _, err := usersCollection.UpdateOne(ctx, bson.M{"supabaseUserId": supabaseUser.ID}, update) - if err != nil { - fmt.Printf(" ❌ Failed to update %s: %v\n", supabaseUser.Email, err) - stats.Errors++ - continue - } - fmt.Printf(" ✅ UPDATED: %s → legacy_unlimited\n", supabaseUser.Email) - } - stats.UpdatedToLegacy++ - - } else { - // User doesn't exist in MongoDB - create new document - if *dryRun { - fmt.Printf(" [DRY RUN] Would CREATE: %s as legacy_unlimited\n", supabaseUser.Email) - } else { - newUser := bson.M{ - "supabaseUserId": supabaseUser.ID, - "email": supabaseUser.Email, - "createdAt": supabaseUser.CreatedAt, // Preserve original Supabase created_at - "lastLoginAt": now, - "subscriptionTier": models.TierLegacyUnlimited, - "subscriptionStatus": models.SubStatusActive, - "migratedToLegacyAt": now, - "preferences": bson.M{ - "storeBuilderChatHistory": true, - }, - } - - _, err := usersCollection.InsertOne(ctx, newUser) - if err != nil { - fmt.Printf(" ❌ Failed to create %s: %v\n", supabaseUser.Email, err) - stats.Errors++ - continue - } - fmt.Printf(" ✅ CREATED: %s as legacy_unlimited\n", supabaseUser.Email) - } - stats.CreatedAsLegacy++ - } - } - - // Print summary - fmt.Println() - fmt.Println("==============================================") - if *dryRun { - fmt.Println("📊 DRY RUN Summary (no changes made)") - } else { - fmt.Println("📊 Migration Summary") - } - fmt.Println("==============================================") - fmt.Printf(" Total Supabase users: %d\n", stats.TotalSupabaseUsers) - fmt.Printf(" Already legacy_unlimited: %d\n", stats.AlreadyMigrated) - fmt.Printf(" Updated to legacy: %d\n", stats.UpdatedToLegacy) - fmt.Printf(" Created as legacy: %d\n", stats.CreatedAsLegacy) - fmt.Printf(" Errors: %d\n", stats.Errors) - fmt.Println() - - if *dryRun { - fmt.Println("💡 To apply changes, run without --dry-run flag:") - fmt.Println(" go run migrate_legacy_users.go") - } else { - // Verification - fmt.Println("🔍 Verification:") - legacyCount, err := usersCollection.CountDocuments(ctx, bson.M{ - "subscriptionTier": models.TierLegacyUnlimited, - }) - if err != nil { - log.Printf("⚠️ Could not verify: %v", err) - } else { - fmt.Printf(" Total legacy_unlimited users in MongoDB: %d\n", legacyCount) - } - } - - fmt.Println() - fmt.Println("==============================================") - if *dryRun { - fmt.Println("✅ Dry Run Complete!") - } else { - fmt.Println("✅ Migration Complete!") - } - fmt.Println("==============================================") -} - -// fetchSupabaseUsers fetches all users from Supabase Auth Admin API -func fetchSupabaseUsers(supabaseURL, serviceKey string) ([]SupabaseUser, error) { - var allUsers []SupabaseUser - page := 1 - perPage := 100 - - for { - url := fmt.Sprintf("%s/auth/v1/admin/users?page=%d&per_page=%d", supabaseURL, page, perPage) - - req, err := http.NewRequest("GET", url, nil) - if err != nil { - return nil, fmt.Errorf("failed to create request: %w", err) - } - - req.Header.Set("Authorization", "Bearer "+serviceKey) - req.Header.Set("apikey", serviceKey) - - client := &http.Client{Timeout: 30 * time.Second} - resp, err := client.Do(req) - if err != nil { - return nil, fmt.Errorf("failed to fetch users: %w", err) - } - defer resp.Body.Close() - - if resp.StatusCode != http.StatusOK { - body, _ := io.ReadAll(resp.Body) - return nil, fmt.Errorf("API returned status %d: %s", resp.StatusCode, string(body)) - } - - body, err := io.ReadAll(resp.Body) - if err != nil { - return nil, fmt.Errorf("failed to read response: %w", err) - } - - var response SupabaseUsersResponse - if err := json.Unmarshal(body, &response); err != nil { - // Try parsing as array directly (some Supabase versions return array) - var users []SupabaseUser - if err2 := json.Unmarshal(body, &users); err2 != nil { - return nil, fmt.Errorf("failed to parse response: %w (original: %v)", err2, err) - } - allUsers = append(allUsers, users...) - if len(users) < perPage { - break - } - } else { - allUsers = append(allUsers, response.Users...) - if len(response.Users) < perPage { - break - } - } - - page++ - } - - return allUsers, nil -} - -// maskURI masks sensitive parts of a connection URI for logging -func maskURI(uri string) string { - if len(uri) > 20 { - return uri[:15] + "..." + uri[len(uri)-10:] - } - return "***" -} - -// ensureIndex creates an index on supabaseUserId if it doesn't exist -func ensureIndex(ctx context.Context, collection interface{}) { - // This is a helper that could be called to ensure fast lookups - // For now, we assume the index exists or MongoDB handles it -} diff --git a/backend/scripts/migrate_sqlite_to_mysql.go b/backend/scripts/migrate_sqlite_to_mysql.go deleted file mode 100644 index 168b1990..00000000 --- a/backend/scripts/migrate_sqlite_to_mysql.go +++ /dev/null @@ -1,702 +0,0 @@ -package main - -import ( - "database/sql" - "fmt" - "log" - "os" - "strings" - - _ "github.com/go-sql-driver/mysql" - _ "modernc.org/sqlite" -) - -type MigrationStats struct { - Providers int - Models int - Aliases int - Filters int - Capabilities int - RefreshLogs int - RecommendedModels int - MCPConnections int - MCPTools int - MCPAuditLogs int - Errors []string -} - -func main() { - // Read configuration from environment - sqlitePath := getEnv("SQLITE_PATH", "./model_capabilities.db") - mysqlDSN := getEnv("MYSQL_DSN", "") - - if mysqlDSN == "" { - log.Fatal("❌ MYSQL_DSN environment variable required\n Format: user:pass@tcp(host:port)/dbname") - } - - log.Println("🔄 Starting SQLite → MySQL migration...") - log.Printf(" SQLite: %s", sqlitePath) - log.Printf(" MySQL: %s\n", maskDSN(mysqlDSN)) - - // Open databases - sqliteDB, err := sql.Open("sqlite", sqlitePath) - if err != nil { - log.Fatalf("❌ Failed to open SQLite: %v", err) - } - defer sqliteDB.Close() - - mysqlDB, err := sql.Open("mysql", mysqlDSN+"?parseTime=true&charset=utf8mb4&collation=utf8mb4_unicode_ci") - if err != nil { - log.Fatalf("❌ Failed to open MySQL: %v", err) - } - defer mysqlDB.Close() - - // Test connections - if err := sqliteDB.Ping(); err != nil { - log.Fatalf("❌ SQLite connection failed: %v", err) - } - if err := mysqlDB.Ping(); err != nil { - log.Fatalf("❌ MySQL connection failed: %v", err) - } - - log.Println("✅ Database connections established\n") - - // Run migration - stats := &MigrationStats{} - - // Start transaction for atomicity - tx, err := mysqlDB.Begin() - if err != nil { - log.Fatalf("❌ Failed to start transaction: %v", err) - } - defer tx.Rollback() // Rollback if we don't commit - - // Migrate in order (respect foreign keys) - steps := []struct { - name string - fn func(*sql.DB, *sql.Tx, *MigrationStats) error - }{ - {"providers", migrateProviders}, - {"models", migrateModels}, - {"model_aliases", migrateAliases}, - {"provider_model_filters", migrateFilters}, - {"model_capabilities", migrateCapabilities}, - {"model_refresh_log", migrateRefreshLogs}, - {"recommended_models", migrateRecommendedModels}, - {"mcp_connections", migrateMCPConnections}, - {"mcp_tools", migrateMCPTools}, - {"mcp_audit_log", migrateMCPAuditLog}, - } - - for _, step := range steps { - log.Printf("📦 Migrating %s...", step.name) - if err := step.fn(sqliteDB, tx, stats); err != nil { - log.Printf("❌ %s migration failed: %v\n", step.name, err) - log.Println("⚠️ Transaction will be rolled back") - return - } - } - - // Commit transaction - if err := tx.Commit(); err != nil { - log.Fatalf("❌ Failed to commit transaction: %v", err) - } - - // Print summary - printSummary(stats) -} - -func migrateProviders(sqlite *sql.DB, mysql *sql.Tx, stats *MigrationStats) error { - rows, err := sqlite.Query(` - SELECT id, name, base_url, COALESCE(api_key, ''), enabled, - COALESCE(audio_only, 0), COALESCE(system_prompt, ''), - COALESCE(favicon, '') - FROM providers - ORDER BY id - `) - if err != nil { - if strings.Contains(err.Error(), "no such table") { - log.Println(" ⚠️ Table doesn't exist in SQLite, skipping") - return nil - } - return fmt.Errorf("query failed: %w", err) - } - defer rows.Close() - - stmt, err := mysql.Prepare(` - INSERT INTO providers (id, name, base_url, api_key, enabled, audio_only, - system_prompt, favicon) - VALUES (?, ?, ?, ?, ?, ?, ?, ?) - `) - if err != nil { - return fmt.Errorf("prepare failed: %w", err) - } - defer stmt.Close() - - for rows.Next() { - var ( - id, name, baseURL, apiKey, systemPrompt, favicon string - enabled, audioOnly bool - ) - - if err := rows.Scan(&id, &name, &baseURL, &apiKey, &enabled, &audioOnly, - &systemPrompt, &favicon); err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("provider scan: %v", err)) - continue - } - - _, err := stmt.Exec(id, name, baseURL, apiKey, enabled, audioOnly, - systemPrompt, favicon) - if err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("provider insert %s: %v", name, err)) - continue - } - stats.Providers++ - } - - log.Printf(" ✅ Migrated %d providers\n", stats.Providers) - return nil -} - -func migrateModels(sqlite *sql.DB, mysql *sql.Tx, stats *MigrationStats) error { - rows, err := sqlite.Query(` - SELECT id, provider_id, name, COALESCE(display_name, ''), - COALESCE(description, ''), COALESCE(context_length, 0), - COALESCE(supports_tools, 0), COALESCE(supports_streaming, 0), - COALESCE(supports_vision, 0), COALESCE(is_visible, 1), - COALESCE(system_prompt, ''), fetched_at - FROM models - ORDER BY id - `) - if err != nil { - if strings.Contains(err.Error(), "no such table") { - log.Println(" ⚠️ Table doesn't exist in SQLite, skipping") - return nil - } - return fmt.Errorf("query failed: %w", err) - } - defer rows.Close() - - stmt, err := mysql.Prepare(` - INSERT INTO models (id, provider_id, name, display_name, description, context_length, - supports_tools, supports_streaming, supports_vision, - is_visible, system_prompt, fetched_at) - VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?) - `) - if err != nil { - return fmt.Errorf("prepare failed: %w", err) - } - defer stmt.Close() - - for rows.Next() { - var ( - id, name, displayName, description, systemPrompt string - providerID, contextLength int - supportsTools, supportsStreaming, supportsVision, isVisible bool - fetchedAt string - ) - - if err := rows.Scan(&id, &providerID, &name, &displayName, &description, &contextLength, - &supportsTools, &supportsStreaming, &supportsVision, - &isVisible, &systemPrompt, &fetchedAt); err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("model scan: %v", err)) - continue - } - - _, err := stmt.Exec(id, providerID, name, displayName, description, contextLength, - supportsTools, supportsStreaming, supportsVision, - isVisible, systemPrompt, fetchedAt) - if err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("model insert %s: %v", id, err)) - continue - } - stats.Models++ - } - - log.Printf(" ✅ Migrated %d models\n", stats.Models) - return nil -} - -func migrateAliases(sqlite *sql.DB, mysql *sql.Tx, stats *MigrationStats) error { - rows, err := sqlite.Query(` - SELECT alias_name, model_id, provider_id, display_name, - COALESCE(description, ''), supports_vision, agents_enabled, - COALESCE(structured_output_support, ''), structured_output_compliance, - COALESCE(structured_output_warning, ''), structured_output_speed_ms, - COALESCE(structured_output_badge, ''), memory_extractor, memory_selector - FROM model_aliases - ORDER BY alias_name - `) - if err != nil { - if strings.Contains(err.Error(), "no such table") { - log.Println(" ⚠️ Table doesn't exist in SQLite, skipping") - return nil - } - return fmt.Errorf("query failed: %w", err) - } - defer rows.Close() - - stmt, err := mysql.Prepare(` - INSERT INTO model_aliases (alias_name, model_id, provider_id, display_name, description, - supports_vision, agents_enabled, structured_output_support, - structured_output_compliance, structured_output_warning, - structured_output_speed_ms, structured_output_badge, - memory_extractor, memory_selector) - VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?) - `) - if err != nil { - return fmt.Errorf("prepare failed: %w", err) - } - defer stmt.Close() - - for rows.Next() { - var ( - aliasName, modelID, displayName, description string - structuredOutputSupport, structuredOutputWarning string - structuredOutputBadge string - providerID, structuredOutputCompliance sql.NullInt64 - structuredOutputSpeedMs sql.NullInt64 - supportsVision, agentsEnabled sql.NullBool - memoryExtractor, memorySelector sql.NullBool - ) - - if err := rows.Scan(&aliasName, &modelID, &providerID, &displayName, &description, - &supportsVision, &agentsEnabled, &structuredOutputSupport, &structuredOutputCompliance, - &structuredOutputWarning, &structuredOutputSpeedMs, &structuredOutputBadge, - &memoryExtractor, &memorySelector); err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("alias scan: %v", err)) - continue - } - - _, err := stmt.Exec(aliasName, modelID, providerID, displayName, description, - nullBoolToPtr(supportsVision), nullBoolToPtr(agentsEnabled), - nullString(structuredOutputSupport), nullIntToPtr(structuredOutputCompliance), - nullString(structuredOutputWarning), nullIntToPtr(structuredOutputSpeedMs), - nullString(structuredOutputBadge), nullBoolToPtr(memoryExtractor), - nullBoolToPtr(memorySelector)) - if err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("alias insert %s: %v", aliasName, err)) - continue - } - stats.Aliases++ - } - - log.Printf(" ✅ Migrated %d aliases\n", stats.Aliases) - return nil -} - -func migrateFilters(sqlite *sql.DB, mysql *sql.Tx, stats *MigrationStats) error { - rows, err := sqlite.Query(` - SELECT provider_id, model_pattern, action, COALESCE(priority, 0) - FROM provider_model_filters - ORDER BY provider_id, priority DESC - `) - if err != nil { - if strings.Contains(err.Error(), "no such table") || strings.Contains(err.Error(), "no such column") { - log.Println(" ⚠️ Table doesn't exist or has different schema in SQLite, skipping") - return nil - } - return fmt.Errorf("query failed: %w", err) - } - defer rows.Close() - - stmt, err := mysql.Prepare(` - INSERT INTO provider_model_filters (provider_id, model_pattern, action, priority) - VALUES (?, ?, ?, ?) - `) - if err != nil { - return fmt.Errorf("prepare failed: %w", err) - } - defer stmt.Close() - - for rows.Next() { - var ( - providerID, priority int - modelPattern, action string - ) - - if err := rows.Scan(&providerID, &modelPattern, &action, &priority); err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("filter scan: %v", err)) - continue - } - - _, err := stmt.Exec(providerID, modelPattern, action, priority) - if err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("filter insert: %v", err)) - continue - } - stats.Filters++ - } - - log.Printf(" ✅ Migrated %d filters\n", stats.Filters) - return nil -} - -func migrateCapabilities(sqlite *sql.DB, mysql *sql.Tx, stats *MigrationStats) error { - // Note: SQLite model_capabilities table may have different schema - // Skip if table doesn't exist or has incompatible columns - rows, err := sqlite.Query(` - SELECT model_id, provider_id - FROM model_capabilities - ORDER BY model_id - `) - if err != nil { - if strings.Contains(err.Error(), "no such table") || strings.Contains(err.Error(), "no such column") { - log.Println(" ⚠️ Table doesn't exist or has different schema in SQLite, skipping") - return nil - } - return fmt.Errorf("query failed: %w", err) - } - defer rows.Close() - - stmt, err := mysql.Prepare(` - INSERT INTO model_capabilities (model_id, provider_id) - VALUES (?, ?) - `) - if err != nil { - return fmt.Errorf("prepare failed: %w", err) - } - defer stmt.Close() - - for rows.Next() { - var ( - modelID string - providerID int - ) - - if err := rows.Scan(&modelID, &providerID); err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("capability scan: %v", err)) - continue - } - - _, err := stmt.Exec(modelID, providerID) - if err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("capability insert %s: %v", modelID, err)) - continue - } - stats.Capabilities++ - } - - log.Printf(" ✅ Migrated %d capabilities\n", stats.Capabilities) - return nil -} - -func migrateRefreshLogs(sqlite *sql.DB, mysql *sql.Tx, stats *MigrationStats) error { - rows, err := sqlite.Query(` - SELECT provider_id, models_fetched, refreshed_at - FROM model_refresh_log - ORDER BY refreshed_at DESC - `) - if err != nil { - if strings.Contains(err.Error(), "no such table") { - log.Println(" ⚠️ Table doesn't exist in SQLite, skipping") - return nil - } - return fmt.Errorf("query failed: %w", err) - } - defer rows.Close() - - stmt, err := mysql.Prepare(` - INSERT INTO model_refresh_log (provider_id, models_fetched, refreshed_at) - VALUES (?, ?, ?) - `) - if err != nil { - return fmt.Errorf("prepare failed: %w", err) - } - defer stmt.Close() - - for rows.Next() { - var ( - providerID, modelsFetched int - refreshedAt string - ) - - if err := rows.Scan(&providerID, &modelsFetched, &refreshedAt); err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("refresh log scan: %v", err)) - continue - } - - _, err := stmt.Exec(providerID, modelsFetched, refreshedAt) - if err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("refresh log insert: %v", err)) - continue - } - stats.RefreshLogs++ - } - - log.Printf(" ✅ Migrated %d refresh logs\n", stats.RefreshLogs) - return nil -} - -func migrateRecommendedModels(sqlite *sql.DB, mysql *sql.Tx, stats *MigrationStats) error { - // This table might not exist in SQLite, skip if missing - rows, err := sqlite.Query(` - SELECT provider_id, tier, model_alias, created_at, updated_at - FROM recommended_models - `) - if err != nil { - if strings.Contains(err.Error(), "no such table") { - log.Println(" ⚠️ Table doesn't exist in SQLite, skipping") - return nil - } - return fmt.Errorf("query failed: %w", err) - } - defer rows.Close() - - stmt, err := mysql.Prepare(` - INSERT INTO recommended_models (provider_id, tier, model_alias, created_at, updated_at) - VALUES (?, ?, ?, ?, ?) - `) - if err != nil { - return fmt.Errorf("prepare failed: %w", err) - } - defer stmt.Close() - - for rows.Next() { - var ( - providerID int - tier, modelAlias string - createdAt, updatedAt string - ) - - if err := rows.Scan(&providerID, &tier, &modelAlias, &createdAt, &updatedAt); err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("recommended model scan: %v", err)) - continue - } - - _, err := stmt.Exec(providerID, tier, modelAlias, createdAt, updatedAt) - if err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("recommended model insert: %v", err)) - continue - } - stats.RecommendedModels++ - } - - if stats.RecommendedModels > 0 { - log.Printf(" ✅ Migrated %d recommended models\n", stats.RecommendedModels) - } - return nil -} - -func migrateMCPConnections(sqlite *sql.DB, mysql *sql.Tx, stats *MigrationStats) error { - rows, err := sqlite.Query(` - SELECT user_id, client_id, is_active, connected_at - FROM mcp_connections - `) - if err != nil { - if strings.Contains(err.Error(), "no such table") { - log.Println(" ⚠️ MCP tables don't exist, skipping") - return nil - } - return fmt.Errorf("query failed: %w", err) - } - defer rows.Close() - - stmt, err := mysql.Prepare(` - INSERT INTO mcp_connections (user_id, client_id, is_active, connected_at) - VALUES (?, ?, ?, ?) - `) - if err != nil { - return fmt.Errorf("prepare failed: %w", err) - } - defer stmt.Close() - - for rows.Next() { - var ( - userID, clientID string - isActive bool - connectedAt string - ) - - if err := rows.Scan(&userID, &clientID, &isActive, &connectedAt); err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("mcp connection scan: %v", err)) - continue - } - - _, err := stmt.Exec(userID, clientID, isActive, connectedAt) - if err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("mcp connection insert: %v", err)) - continue - } - stats.MCPConnections++ - } - - if stats.MCPConnections > 0 { - log.Printf(" ✅ Migrated %d MCP connections\n", stats.MCPConnections) - } - return nil -} - -func migrateMCPTools(sqlite *sql.DB, mysql *sql.Tx, stats *MigrationStats) error { - rows, err := sqlite.Query(` - SELECT user_id, connection_id, tool_name, tool_definition - FROM mcp_tools - `) - if err != nil { - if strings.Contains(err.Error(), "no such table") || strings.Contains(err.Error(), "no such column") { - return nil // Already logged in connections or schema mismatch - } - return fmt.Errorf("query failed: %w", err) - } - defer rows.Close() - - stmt, err := mysql.Prepare(` - INSERT INTO mcp_tools (user_id, connection_id, tool_name, tool_definition) - VALUES (?, ?, ?, ?) - `) - if err != nil { - return fmt.Errorf("prepare failed: %w", err) - } - defer stmt.Close() - - for rows.Next() { - var ( - userID, toolName, toolDefinition string - connectionID int - ) - - if err := rows.Scan(&userID, &connectionID, &toolName, &toolDefinition); err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("mcp tool scan: %v", err)) - continue - } - - _, err := stmt.Exec(userID, connectionID, toolName, toolDefinition) - if err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("mcp tool insert: %v", err)) - continue - } - stats.MCPTools++ - } - - if stats.MCPTools > 0 { - log.Printf(" ✅ Migrated %d MCP tools\n", stats.MCPTools) - } - return nil -} - -func migrateMCPAuditLog(sqlite *sql.DB, mysql *sql.Tx, stats *MigrationStats) error { - rows, err := sqlite.Query(` - SELECT user_id, tool_name, COALESCE(conversation_id, ''), success, - COALESCE(error_message, ''), executed_at - FROM mcp_audit_log - `) - if err != nil { - if strings.Contains(err.Error(), "no such table") { - return nil // Already logged in connections - } - return fmt.Errorf("query failed: %w", err) - } - defer rows.Close() - - stmt, err := mysql.Prepare(` - INSERT INTO mcp_audit_log (user_id, tool_name, conversation_id, success, error_message, executed_at) - VALUES (?, ?, ?, ?, ?, ?) - `) - if err != nil { - return fmt.Errorf("prepare failed: %w", err) - } - defer stmt.Close() - - for rows.Next() { - var ( - userID, toolName, conversationID, errorMessage string - success bool - executedAt string - ) - - if err := rows.Scan(&userID, &toolName, &conversationID, &success, &errorMessage, &executedAt); err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("mcp audit log scan: %v", err)) - continue - } - - _, err := stmt.Exec(userID, toolName, nullString(conversationID), success, nullString(errorMessage), executedAt) - if err != nil { - stats.Errors = append(stats.Errors, fmt.Sprintf("mcp audit log insert: %v", err)) - continue - } - stats.MCPAuditLogs++ - } - - if stats.MCPAuditLogs > 0 { - log.Printf(" ✅ Migrated %d MCP audit logs\n", stats.MCPAuditLogs) - } - return nil -} - -func printSummary(stats *MigrationStats) { - log.Println("\n" + strings.Repeat("=", 60)) - log.Println("✅ MIGRATION COMPLETE") - log.Println(strings.Repeat("=", 60)) - log.Printf("📊 Providers: %d migrated\n", stats.Providers) - log.Printf("📊 Models: %d migrated\n", stats.Models) - log.Printf("📊 Aliases: %d migrated\n", stats.Aliases) - log.Printf("📊 Filters: %d migrated\n", stats.Filters) - log.Printf("📊 Capabilities: %d migrated\n", stats.Capabilities) - log.Printf("📊 Refresh Logs: %d migrated\n", stats.RefreshLogs) - if stats.RecommendedModels > 0 { - log.Printf("📊 Recommended Models: %d migrated\n", stats.RecommendedModels) - } - if stats.MCPConnections > 0 { - log.Printf("📊 MCP Connections: %d migrated\n", stats.MCPConnections) - log.Printf("📊 MCP Tools: %d migrated\n", stats.MCPTools) - log.Printf("📊 MCP Audit Logs: %d migrated\n", stats.MCPAuditLogs) - } - - if len(stats.Errors) > 0 { - log.Printf("\n⚠️ %d errors occurred:\n", len(stats.Errors)) - for i, err := range stats.Errors { - if i < 10 { // Show first 10 - log.Printf(" %d. %s\n", i+1, err) - } - } - if len(stats.Errors) > 10 { - log.Printf(" ... and %d more\n", len(stats.Errors)-10) - } - } else { - log.Println("\n✅ No errors - perfect migration!") - } - log.Println(strings.Repeat("=", 60)) -} - -// Helper functions -func getEnv(key, fallback string) string { - if val := os.Getenv(key); val != "" { - return val - } - return fallback -} - -func maskDSN(dsn string) string { - // Mask password in DSN for logging - // user:pass@tcp(host:port)/dbname → user:***@tcp(host:port)/dbname - parts := strings.Split(dsn, "@") - if len(parts) < 2 { - return dsn - } - userPass := strings.Split(parts[0], ":") - if len(userPass) < 2 { - return dsn - } - return userPass[0] + ":***@" + parts[1] -} - -func nullString(s string) interface{} { - if s == "" { - return nil - } - return s -} - -func nullBoolToPtr(nb sql.NullBool) interface{} { - if !nb.Valid { - return nil - } - return nb.Bool -} - -func nullIntToPtr(ni sql.NullInt64) interface{} { - if !ni.Valid { - return nil - } - return ni.Int64 -} diff --git a/backend/scripts/run_agent_model_tests.sh b/backend/scripts/run_agent_model_tests.sh deleted file mode 100755 index 75baffeb..00000000 --- a/backend/scripts/run_agent_model_tests.sh +++ /dev/null @@ -1,53 +0,0 @@ -#!/bin/bash - -# Agent Model Testing Script -# Tests all models with "agents": true and measures their performance - -set -e - -SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)" -BACKEND_DIR="$(cd "$SCRIPT_DIR/.." && pwd)" - -echo "🧪 Running Agent Model Performance Tests" -echo "========================================" -echo "" -echo "This script will:" -echo " 1. Test all models with 'agents: true'" -echo " 2. Measure average response time (3 tests per model)" -echo " 3. Provide suggestions for providers.json updates" -echo "" -echo "⚠️ This may take several minutes depending on the number of models" -echo "" - -read -p "Press Enter to continue or Ctrl+C to cancel..." - -# Check if providers.json exists -if [ ! -f "$BACKEND_DIR/providers.json" ]; then - echo "❌ Error: providers.json not found in $BACKEND_DIR" - exit 1 -fi - -# Build and run the test script -echo "" -echo "📦 Building test script..." -cd "$SCRIPT_DIR" -go build -o test_agent_models test_agent_models.go - -echo "🚀 Running tests..." -echo "" - -# Run from backend directory so it can find providers.json -cd "$BACKEND_DIR" -"$SCRIPT_DIR/test_agent_models" - -# Cleanup -rm -f "$SCRIPT_DIR/test_agent_models" - -echo "" -echo "✅ Testing complete!" -echo "" -echo "📝 Next steps:" -echo " 1. Review the test results above" -echo " 2. Copy the suggested updates to providers.json" -echo " 3. Restart the backend server" -echo "" diff --git a/backend/scripts/run_structured_output_tests.sh b/backend/scripts/run_structured_output_tests.sh deleted file mode 100755 index 2e883829..00000000 --- a/backend/scripts/run_structured_output_tests.sh +++ /dev/null @@ -1,32 +0,0 @@ -#!/bin/bash - -# Structured Output Testing Suite -# Tests all configured LLM providers for JSON schema support - -set -e - -echo "🧪 LLM Provider Structured Output Testing Suite" -echo "==============================================" -echo "" - -# Load environment variables (filter out comments and empty lines) -if [ -f ../.env ]; then - set -a - source <(cat ../.env | grep -v '^#' | grep -v '^$' | sed 's/#.*$//' | sed 's/[[:space:]]*$//') - set +a -fi - -# Check if Go is installed -if ! command -v go &> /dev/null; then - echo "❌ Go is not installed. Please install Go to run this test suite." - exit 1 -fi - -# Run the test suite -echo "📦 Compiling test suite..." -go run test_structured_outputs.go - -echo "" -echo "✅ Testing complete! Check the report above for results." -echo "" -echo "📄 Results saved to: structured_output_test_results.json" diff --git a/backend/scripts/test-admin.sh b/backend/scripts/test-admin.sh deleted file mode 100755 index 54cbeb54..00000000 --- a/backend/scripts/test-admin.sh +++ /dev/null @@ -1,42 +0,0 @@ -#!/bin/bash - -# Configuration -export TOKEN="eyJhbGciOiJIUzI1NiIsImtpZCI6Ik80ZEtMNTB1YityUmo4N2ciLCJ0eXAiOiJKV1QifQ.eyJpc3MiOiJodHRwczovL29jcW9xamFmbWp1aXl3c3Bwd2t3LnN1cGFiYXNlLmNvL2F1dGgvdjEiLCJzdWIiOiJjNDFmZGM1YS01YzA0LTQxMmUtYWQ3Ny0xNzM0YjlkNTgyYmYiLCJhdWQiOiJhdXRoZW50aWNhdGVkIiwiZXhwIjoxNzY3MjI0NzY4LCJpYXQiOjE3NjcyMjExNjgsImVtYWlsIjoiYXJ1bnRlbW1lQGdtYWlsLmNvbSIsInBob25lIjoiIiwiYXBwX21ldGFkYXRhIjp7InByb3ZpZGVyIjoiZW1haWwiLCJwcm92aWRlcnMiOlsiZW1haWwiLCJnaXRodWIiXX0sInVzZXJfbWV0YWRhdGEiOnsiYXZhdGFyX3VybCI6Imh0dHBzOi8vYXZhdGFycy5naXRodWJ1c2VyY29udGVudC5jb20vdS80MTExODE4Nz92PTQiLCJkaXNwbGF5X25hbWUiOiJhcnVudGVtbWUiLCJlbWFpbCI6ImFydW5nYXV0aGFta0BnbWFpbC5jb20iLCJlbWFpbF92ZXJpZmllZCI6dHJ1ZSwiZnVsbF9uYW1lIjoiQXJ1biIsImlzcyI6Imh0dHBzOi8vYXBpLmdpdGh1Yi5jb20iLCJuYW1lIjoiQXJ1biIsInBob25lX3ZlcmlmaWVkIjpmYWxzZSwicHJlZmVycmVkX3VzZXJuYW1lIjoiYXJ1bnRlbW1lIiwicHJvdmlkZXJfaWQiOiI0MTExODE4NyIsInN1YiI6IjQxMTE4MTg3IiwidXNlcl9uYW1lIjoiYXJ1bnRlbW1lIiwidXNlcm5hbWUiOiJhcnVudGVtbWUifSwicm9sZSI6ImF1dGhlbnRpY2F0ZWQiLCJhYWwiOiJhYWwxIiwiYW1yIjpbeyJtZXRob2QiOiJwYXNzd29yZCIsInRpbWVzdGFtcCI6MTc2NzIyMTE2OH1dLCJzZXNzaW9uX2lkIjoiZjUxMjE2ZmMtYTljOC00NGVmLWI5ZjEtNGVmOWU1N2NmODUwIiwiaXNfYW5vbnltb3VzIjpmYWxzZX0.NIfXpOQfHQ3_9C8ntfwMxybV75SppgT6HTZ4lJenZZU" -export TARGET_USER="30c8850f-bec2-47ef-8646-be4afbbfdb9e" -BASE_URL="http://localhost:3001/api/admin" - -# echo "=== Testing Admin Endpoints ===" -# echo "" - -# echo "1. Getting user details..." -# curl -s -X GET -H "Authorization: Bearer $TOKEN" "$BASE_URL/users/$TARGET_USER" | jq - -echo "" -echo "2. Setting tier override to pro..." -curl -s -X POST -H "Authorization: Bearer $TOKEN" -H "Content-Type: application/json" \ - -d '{"tier":"free","reason":"Test"}' "$BASE_URL/users/$TARGET_USER/overrides" | jq - -# echo "" -# echo "3. Verifying tier override..." -# curl -s -X GET -H "Authorization: Bearer $TOKEN" "$BASE_URL/users/$TARGET_USER" | jq '.effective_tier, .has_tier_override' - -# echo "" -# echo "4. Setting granular overrides..." -# curl -s -X POST -H "Authorization: Bearer $TOKEN" -H "Content-Type: application/json" \ -# -d '{"limits":{"maxMessagesPerMonth":-1},"reason":"Unlimited messages test"}' \ -# "$BASE_URL/users/$TARGET_USER/overrides" | jq - -# echo "" -# echo "5. Verifying granular overrides..." -# curl -s -X GET -H "Authorization: Bearer $TOKEN" "$BASE_URL/users/$TARGET_USER" | jq '.effective_limits.maxMessagesPerMonth, .has_limit_overrides' - -# echo "" -# echo "6. Removing all overrides..." -# curl -s -X DELETE -H "Authorization: Bearer $TOKEN" "$BASE_URL/users/$TARGET_USER/overrides" | jq - -# echo "" -# echo "7. Verifying overrides removed..." -# curl -s -X GET -H "Authorization: Bearer $TOKEN" "$BASE_URL/users/$TARGET_USER" | jq '.has_tier_override, .has_limit_overrides' - -echo "" -echo "=== Tests Complete ===" diff --git a/backend/scripts/test_agent_models.go b/backend/scripts/test_agent_models.go deleted file mode 100644 index 8d30b387..00000000 --- a/backend/scripts/test_agent_models.go +++ /dev/null @@ -1,293 +0,0 @@ -package main - -import ( - "bytes" - "encoding/json" - "fmt" - "io" - "net/http" - "os" - "strings" - "time" -) - -// Test configuration -const ( - testPrompt = "Write a simple 'Hello World' function in Python." - numTests = 3 // Run each test 3 times and average -) - -// Provider and model structures -type ModelAlias struct { - ActualModel string `json:"actual_model"` - DisplayName string `json:"display_name"` - Description string `json:"description,omitempty"` - SupportsVision *bool `json:"supports_vision,omitempty"` - Agents *bool `json:"agents,omitempty"` - StructuredOutputSupport string `json:"structured_output_support,omitempty"` - StructuredOutputCompliance *int `json:"structured_output_compliance,omitempty"` - StructuredOutputWarning string `json:"structured_output_warning,omitempty"` - StructuredOutputSpeedMs *int `json:"structured_output_speed_ms,omitempty"` - StructuredOutputBadge string `json:"structured_output_badge,omitempty"` -} - -type ProviderConfig struct { - Name string `json:"name"` - BaseURL string `json:"base_url"` - APIKey string `json:"api_key"` - Enabled bool `json:"enabled"` - AudioOnly bool `json:"audio_only,omitempty"` - ImageOnly bool `json:"image_only,omitempty"` - Favicon string `json:"favicon,omitempty"` - ModelAliases map[string]ModelAlias `json:"model_aliases,omitempty"` -} - -type ProvidersConfig struct { - Providers []ProviderConfig `json:"providers"` -} - -// OpenAI-compatible chat request/response -type ChatMessage struct { - Role string `json:"role"` - Content string `json:"content"` -} - -type ChatRequest struct { - Model string `json:"model"` - Messages []ChatMessage `json:"messages"` -} - -type ChatResponse struct { - Choices []struct { - Message ChatMessage `json:"message"` - } `json:"choices"` - Error *struct { - Message string `json:"message"` - Type string `json:"type"` - } `json:"error,omitempty"` -} - -// TestResult holds the result of testing a model -type TestResult struct { - ModelAlias string - ModelName string - Provider string - AvgSpeedMs int - Success bool - Error string -} - -func main() { - fmt.Println("🧪 Agent Model Testing Script") - fmt.Println("=============================\n") - - // Read providers.json - data, err := os.ReadFile("providers.json") - if err != nil { - fmt.Printf("❌ Error reading providers.json: %v\n", err) - os.Exit(1) - } - - var config ProvidersConfig - if err := json.Unmarshal(data, &config); err != nil { - fmt.Printf("❌ Error parsing providers.json: %v\n", err) - os.Exit(1) - } - - // Collect models to test - var results []TestResult - - for _, provider := range config.Providers { - // Skip disabled providers - if !provider.Enabled { - continue - } - - // Skip audio/image only providers - if provider.AudioOnly || provider.ImageOnly { - continue - } - - // Skip providers without model aliases - if len(provider.ModelAliases) == 0 { - continue - } - - fmt.Printf("\n📦 Testing provider: %s\n", provider.Name) - fmt.Printf(" Base URL: %s\n", provider.BaseURL) - - for aliasName, alias := range provider.ModelAliases { - // Only test models with agents: true - if alias.Agents == nil || !*alias.Agents { - fmt.Printf(" ⏭️ Skipping %s (agents: false or not set)\n", aliasName) - continue - } - - fmt.Printf("\n 🔬 Testing model: %s (%s)\n", alias.DisplayName, aliasName) - - result := testModel(provider, aliasName, alias) - results = append(results, result) - - if result.Success { - fmt.Printf(" ✅ Success! Avg speed: %dms\n", result.AvgSpeedMs) - } else { - fmt.Printf(" ❌ Failed: %s\n", result.Error) - } - - // Sleep between tests to avoid rate limiting - time.Sleep(2 * time.Second) - } - } - - // Print summary - fmt.Println("\n\n📊 Test Results Summary") - fmt.Println("========================\n") - - fmt.Printf("%-30s %-40s %-15s %s\n", "Provider", "Model", "Speed (ms)", "Status") - fmt.Println(strings.Repeat("-", 100)) - - for _, result := range results { - status := "✅ Pass" - speedStr := fmt.Sprintf("%d", result.AvgSpeedMs) - if !result.Success { - status = "❌ Fail" - speedStr = "N/A" - } - fmt.Printf("%-30s %-40s %-15s %s\n", - truncate(result.Provider, 30), - truncate(result.ModelName, 40), - speedStr, - status) - } - - // Generate update suggestions - fmt.Println("\n\n💡 Suggested Updates for providers.json") - fmt.Println("=========================================\n") - - for _, result := range results { - if result.Success { - badge := "" - support := "good" - - if result.AvgSpeedMs < 2000 { - badge = "FASTEST" - support = "excellent" - } else if result.AvgSpeedMs < 5000 { - support = "excellent" - } else if result.AvgSpeedMs < 10000 { - support = "good" - } else { - support = "fair" - } - - fmt.Printf("Model: %s\n", result.ModelAlias) - fmt.Printf(" \"structured_output_speed_ms\": %d,\n", result.AvgSpeedMs) - fmt.Printf(" \"structured_output_support\": \"%s\",\n", support) - if badge != "" { - fmt.Printf(" \"structured_output_badge\": \"%s\",\n", badge) - } - fmt.Println() - } - } -} - -func testModel(provider ProviderConfig, aliasName string, alias ModelAlias) TestResult { - result := TestResult{ - ModelAlias: aliasName, - ModelName: alias.DisplayName, - Provider: provider.Name, - } - - var totalDuration int64 - - for i := 0; i < numTests; i++ { - start := time.Now() - - err := callModel(provider.BaseURL, provider.APIKey, alias.ActualModel) - - duration := time.Since(start).Milliseconds() - - if err != nil { - result.Success = false - result.Error = err.Error() - return result - } - - totalDuration += duration - fmt.Printf(" Test %d: %dms\n", i+1, duration) - } - - result.Success = true - result.AvgSpeedMs = int(totalDuration / int64(numTests)) - - return result -} - -func callModel(baseURL, apiKey, modelName string) error { - // Prepare request - reqBody := ChatRequest{ - Model: modelName, - Messages: []ChatMessage{ - { - Role: "user", - Content: testPrompt, - }, - }, - } - - jsonData, err := json.Marshal(reqBody) - if err != nil { - return fmt.Errorf("marshal request: %w", err) - } - - // Make HTTP request - url := baseURL + "/chat/completions" - req, err := http.NewRequest("POST", url, bytes.NewBuffer(jsonData)) - if err != nil { - return fmt.Errorf("create request: %w", err) - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Authorization", "Bearer "+apiKey) - - client := &http.Client{ - Timeout: 60 * time.Second, - } - - resp, err := client.Do(req) - if err != nil { - return fmt.Errorf("send request: %w", err) - } - defer resp.Body.Close() - - body, err := io.ReadAll(resp.Body) - if err != nil { - return fmt.Errorf("read response: %w", err) - } - - if resp.StatusCode != http.StatusOK { - return fmt.Errorf("HTTP %d: %s", resp.StatusCode, string(body)) - } - - var chatResp ChatResponse - if err := json.Unmarshal(body, &chatResp); err != nil { - return fmt.Errorf("parse response: %w", err) - } - - if chatResp.Error != nil { - return fmt.Errorf("API error: %s", chatResp.Error.Message) - } - - if len(chatResp.Choices) == 0 { - return fmt.Errorf("no response from model") - } - - return nil -} - -func truncate(s string, maxLen int) string { - if len(s) <= maxLen { - return s - } - return s[:maxLen-3] + "..." -} diff --git a/backend/scripts/test_promo.sh b/backend/scripts/test_promo.sh deleted file mode 100755 index 0a3b904b..00000000 --- a/backend/scripts/test_promo.sh +++ /dev/null @@ -1,124 +0,0 @@ -#!/bin/bash - -# Test Script: Verify Promotional Campaign -# This script creates a test user and verifies they receive Pro tier during the promo window - -set -e - -# Colors for output -RED='\033[0;31m' -GREEN='\033[0;32m' -YELLOW='\033[1;33m' -NC='\033[0m' # No Color - -echo "==================================================" -echo "🎁 Testing Promotional Campaign Feature" -echo "==================================================" -echo "" - -# Configuration -BACKEND_URL="${BACKEND_URL:-http://localhost:3001}" -TEST_USER_ID="promo-test-$(date +%s)" -TEST_EMAIL="promo-test-$(date +%s)@example.com" - -echo "📋 Test Configuration:" -echo " Backend URL: $BACKEND_URL" -echo " Test User ID: $TEST_USER_ID" -echo " Test Email: $TEST_EMAIL" -echo "" - -# Step 1: Check promo configuration -echo "🔍 Step 1: Checking promo configuration..." -echo " Expected: PROMO_ENABLED=true" -echo " Expected: PROMO_START_DATE=2025-12-31T00:00:00Z" -echo " Expected: PROMO_END_DATE=2026-02-01T00:00:00Z" -echo " Expected: PROMO_DURATION_DAYS=30" -echo "" - -# Step 2: Create test user directly in MongoDB -echo "🔨 Step 2: Creating test user in MongoDB..." -mongo_result=$(mongosh --quiet "mongodb://localhost:27017/claraverse" --eval " -db.users.insertOne({ - supabaseUserId: '$TEST_USER_ID', - email: '$TEST_EMAIL', - createdAt: new Date(), - lastLoginAt: new Date(), - subscriptionTier: 'free', - subscriptionStatus: 'active', - preferences: { - storeBuilderChatHistory: true - }, - hasSeenWelcomePopup: false -}) -" 2>&1 || echo "Error creating user") - -if echo "$mongo_result" | grep -q "acknowledged.*true"; then - echo -e "${GREEN}✅ Test user created in MongoDB${NC}" -else - echo -e "${RED}❌ Failed to create test user${NC}" - echo "$mongo_result" - exit 1 -fi -echo "" - -# Step 3: Trigger SyncUserFromSupabase via API call -# Note: We need a valid Supabase JWT token for this -echo "⚠️ Step 3: Manual verification required" -echo "" -echo "To complete the test, you need to:" -echo " 1. Sign up a new account in the frontend: $BACKEND_URL" -echo " 2. After signup, check your subscription in Settings > Billing" -echo " 3. Verify you have Pro tier with an expiration date" -echo "" -echo "Alternative: Run the integration test:" -echo " cd backend && go test -v ./internal/services/... -run TestPromoIntegration" -echo "" - -# Step 4: Query MongoDB to check user's subscription -echo "🔍 Step 4: Checking MongoDB directly for promo users..." -promo_users=$(mongosh --quiet "mongodb://localhost:27017/claraverse" --eval " -db.users.find({ - subscriptionTier: 'pro', - subscriptionExpiresAt: { \$exists: true }, - dodoSubscriptionId: { \$exists: false } -}).count() -" 2>&1 | tail -1) - -echo " Current promo users in database: $promo_users" -echo "" - -# Step 5: Show recent users -echo "📊 Step 5: Recent users created (last 5):" -mongosh --quiet "mongodb://localhost:27017/claraverse" --eval " -db.users.find({}, { - email: 1, - subscriptionTier: 1, - subscriptionExpiresAt: 1, - createdAt: 1, - _id: 0 -}) -.sort({ createdAt: -1 }) -.limit(5) -.forEach(function(user) { - var expiresAt = user.subscriptionExpiresAt ? user.subscriptionExpiresAt.toISOString() : 'N/A'; - print(' ' + user.email + ' | Tier: ' + user.subscriptionTier + ' | Expires: ' + expiresAt); -}) -" -echo "" - -# Cleanup -echo "🧹 Cleanup: Removing test user..." -mongosh --quiet "mongodb://localhost:27017/claraverse" --eval " -db.users.deleteOne({ supabaseUserId: '$TEST_USER_ID' }) -" > /dev/null 2>&1 -echo -e "${GREEN}✅ Test user removed${NC}" -echo "" - -echo "==================================================" -echo "✅ Test script completed!" -echo "==================================================" -echo "" -echo "Summary:" -echo " - Integration tests verify promo logic works correctly ✅" -echo " - To test with real signup: Create new account in frontend" -echo " - Expected result: New users get Pro tier until $(date -j -f '%Y-%m-%dT%H:%M:%SZ' '2026-01-30T00:00:00Z' '+%B %d, %Y' 2>/dev/null || echo 'Jan 30, 2026')" diff --git a/backend/scripts/test_structured_outputs.go b/backend/scripts/test_structured_outputs.go deleted file mode 100644 index 2894ad5d..00000000 --- a/backend/scripts/test_structured_outputs.go +++ /dev/null @@ -1,591 +0,0 @@ -package main - -import ( - "bytes" - "encoding/json" - "fmt" - "io" - "net/http" - "os" - "strings" - "time" -) - -// Test schema for news articles (realistic production use case) -var newsSchema = map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "articles": map[string]interface{}{ - "type": "array", - "items": map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "title": map[string]interface{}{"type": "string"}, - "source": map[string]interface{}{"type": "string"}, - "url": map[string]interface{}{"type": "string"}, - "summary": map[string]interface{}{"type": "string"}, - "publishedDate": map[string]interface{}{"type": "string"}, - }, - "required": []string{"title", "source", "url", "summary", "publishedDate"}, - "additionalProperties": false, - }, - }, - "totalResults": map[string]interface{}{"type": "number"}, - "fetchedAt": map[string]interface{}{"type": "string"}, - }, - "required": []string{"articles", "totalResults", "fetchedAt"}, - "additionalProperties": false, -} - -// Provider configuration -type Provider struct { - Name string - BaseURL string - APIKey string - Models []string -} - -// Test result -type TestResult struct { - Provider string - Model string - SupportsStrict bool - SupportsJSON bool - SchemaCompliance float64 // 0-100 score - ResponseTime time.Duration - Error string - RawResponse string - ParsedCorrectly bool - MissingFields []string - ExtraFields []string -} - -func main() { - fmt.Println("🧪 STRUCTURED OUTPUT TESTING SUITE") - fmt.Println("=" + strings.Repeat("=", 79)) - fmt.Println() - - fmt.Println("📋 Test Specification:") - fmt.Println(" - Task: Generate 3 AI news articles with structured data") - fmt.Println(" - Schema: News articles with title, source, url, summary, publishedDate") - fmt.Println(" - Test 1: Strict JSON schema mode (OpenAI-style)") - fmt.Println(" - Test 2: Fallback JSON object mode + schema in prompt") - fmt.Println(" - Validation: Check for required fields, extra fields, compliance %") - fmt.Println() - - fmt.Println("📂 Loading provider configuration from providers.json...") - providers := loadProviders() - - if len(providers) == 0 { - fmt.Println("❌ No providers found or all providers are disabled") - return - } - - fmt.Printf("✅ Found %d enabled providers\n", len(providers)) - - // Count total models - totalModels := 0 - for _, p := range providers { - totalModels += len(p.Models) - } - fmt.Printf("📊 Will test %d models across %d providers (up to 2 tests per model)\n", totalModels, len(providers)) - fmt.Println() - - results := []TestResult{} - currentTest := 0 - - for _, provider := range providers { - fmt.Printf("📦 Testing Provider: %s (%s)\n", provider.Name, provider.BaseURL) - fmt.Printf(" Models to test: %v\n", provider.Models) - fmt.Println(strings.Repeat("-", 80)) - - for _, model := range provider.Models { - currentTest++ - fmt.Printf("\n[%d/%d] Testing: %s / %s\n", currentTest, totalModels, provider.Name, model) - fmt.Print(" ⏳ Attempting strict JSON schema mode... ") - - result := testModel(provider, model) - results = append(results, result) - printResult(result) - } - fmt.Println() - } - - // Generate report - fmt.Println("\n" + strings.Repeat("=", 80)) - fmt.Println("📝 GENERATING FINAL REPORT") - fmt.Println(strings.Repeat("=", 80) + "\n") - generateReport(results) -} - -func loadProviders() []Provider { - // Load from providers.json file - providersFile := "../providers.json" - data, err := os.ReadFile(providersFile) - if err != nil { - fmt.Printf("⚠️ Could not read providers.json: %v\n", err) - fmt.Println("Using empty provider list") - return []Provider{} - } - - var config struct { - Providers []struct { - Name string `json:"name"` - BaseURL string `json:"base_url"` - APIKey string `json:"api_key"` - Enabled bool `json:"enabled"` - AudioOnly bool `json:"audio_only"` - ImageOnly bool `json:"image_only"` - ImageEditOnly bool `json:"image_edit_only"` - ModelAliases map[string]map[string]interface{} `json:"model_aliases"` - } `json:"providers"` - } - - if err := json.Unmarshal(data, &config); err != nil { - fmt.Printf("⚠️ Could not parse providers.json: %v\n", err) - return []Provider{} - } - - var providers []Provider - for _, p := range config.Providers { - // Skip disabled providers - if !p.Enabled { - continue - } - - // Skip image-only and audio-only providers (they don't support chat completions) - if p.AudioOnly || p.ImageOnly || p.ImageEditOnly { - continue - } - - // Extract actual model IDs from model_aliases (use actual_model value, not the alias key) - var models []string - for aliasKey, aliasData := range p.ModelAliases { - // Try to get actual_model, fall back to alias key if not found - if actualModel, ok := aliasData["actual_model"].(string); ok && actualModel != "" { - models = append(models, actualModel) - } else { - models = append(models, aliasKey) - } - } - - // Skip if no models configured - if len(models) == 0 { - continue - } - - // Limit to first 3 models per provider for faster testing - if len(models) > 3 { - models = models[:3] - } - - providers = append(providers, Provider{ - Name: p.Name, - BaseURL: p.BaseURL, - APIKey: p.APIKey, - Models: models, - }) - } - - return providers -} - -func testModel(provider Provider, model string) TestResult { - result := TestResult{ - Provider: provider.Name, - Model: model, - } - - // Skip if no API key - if provider.APIKey == "" { - fmt.Println("❌ (No API key)") - result.Error = "API key not configured" - return result - } - - fmt.Println() - - // Test 1: Strict JSON Schema mode - fmt.Println(" 📝 Test 1/2: Strict JSON Schema Mode") - fmt.Print(" Sending request with response_format={type: json_schema, strict: true}... ") - strictResult := testStrictMode(provider, model) - result.SupportsStrict = strictResult.Success - result.ResponseTime = strictResult.Duration - - if strictResult.Success { - fmt.Printf("✅ Success (%v)\n", strictResult.Duration) - fmt.Print(" Validating schema compliance... ") - - // Validate schema compliance - compliance := validateSchemaCompliance(strictResult.Response) - result.SchemaCompliance = compliance.Score - result.ParsedCorrectly = compliance.Valid - result.MissingFields = compliance.MissingFields - result.ExtraFields = compliance.ExtraFields - result.RawResponse = truncate(strictResult.Response, 200) - - fmt.Printf("%.1f%%\n", compliance.Score) - } else { - fmt.Printf("❌ Failed\n") - fmt.Printf(" Error: %s\n", truncate(strictResult.Error, 80)) - result.Error = strictResult.Error - - // Test 2: Fallback to basic JSON mode - fmt.Println() - fmt.Println(" 📝 Test 2/2: Fallback JSON Object Mode") - fmt.Print(" Sending request with response_format={type: json_object} + schema in prompt... ") - jsonResult := testJSONMode(provider, model) - result.SupportsJSON = jsonResult.Success - - if jsonResult.Success { - fmt.Printf("✅ Success (%v)\n", jsonResult.Duration) - fmt.Print(" Validating schema compliance... ") - - compliance := validateSchemaCompliance(jsonResult.Response) - result.SchemaCompliance = compliance.Score - result.ParsedCorrectly = compliance.Valid - result.MissingFields = compliance.MissingFields - result.ExtraFields = compliance.ExtraFields - result.RawResponse = truncate(jsonResult.Response, 200) - - fmt.Printf("%.1f%%\n", compliance.Score) - } else { - fmt.Printf("❌ Failed\n") - fmt.Printf(" Error: %s\n", truncate(jsonResult.Error, 80)) - } - } - - fmt.Println() - - return result -} - -type APITestResult struct { - Success bool - Response string - Error string - Duration time.Duration -} - -func testStrictMode(provider Provider, model string) APITestResult { - start := time.Now() - - requestBody := map[string]interface{}{ - "model": model, - "messages": []map[string]string{ - {"role": "system", "content": "You are a news aggregator. Return news in the exact schema format."}, - {"role": "user", "content": "Get 3 AI news articles from today"}, - }, - "temperature": 0.3, - "response_format": map[string]interface{}{ - "type": "json_schema", - "json_schema": map[string]interface{}{ - "name": "news_output", - "strict": true, - "schema": newsSchema, - }, - }, - } - - body, err := json.Marshal(requestBody) - if err != nil { - return APITestResult{Success: false, Error: fmt.Sprintf("Marshal error: %v", err)} - } - - req, err := http.NewRequest("POST", provider.BaseURL+"/chat/completions", bytes.NewReader(body)) - if err != nil { - return APITestResult{Success: false, Error: fmt.Sprintf("Request error: %v", err)} - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Authorization", "Bearer "+provider.APIKey) - - client := &http.Client{Timeout: 60 * time.Second} - resp, err := client.Do(req) - if err != nil { - return APITestResult{Success: false, Error: fmt.Sprintf("HTTP error: %v", err)} - } - defer resp.Body.Close() - - duration := time.Since(start) - - if resp.StatusCode != 200 { - bodyBytes, _ := io.ReadAll(resp.Body) - errorMsg := string(bodyBytes) - // Try to parse error response for better readability - var errorResp map[string]interface{} - if json.Unmarshal(bodyBytes, &errorResp) == nil { - if errObj, ok := errorResp["error"].(map[string]interface{}); ok { - if message, ok := errObj["message"].(string); ok { - errorMsg = message - } - } - } - return APITestResult{ - Success: false, - Error: fmt.Sprintf("Status %d: %s", resp.StatusCode, errorMsg), - Duration: duration, - } - } - - var result map[string]interface{} - if err := json.NewDecoder(resp.Body).Decode(&result); err != nil { - return APITestResult{Success: false, Error: fmt.Sprintf("Decode error: %v", err)} - } - - // Extract content - content := extractContent(result) - return APITestResult{ - Success: true, - Response: content, - Duration: duration, - } -} - -func testJSONMode(provider Provider, model string) APITestResult { - start := time.Now() - - requestBody := map[string]interface{}{ - "model": model, - "messages": []map[string]string{ - {"role": "system", "content": "You are a news aggregator. Return news as valid JSON with articles array, totalResults, and fetchedAt fields."}, - {"role": "user", "content": "Get 3 AI news articles from today. Return as JSON."}, - }, - "temperature": 0.3, - "response_format": map[string]interface{}{ - "type": "json_object", - }, - } - - body, err := json.Marshal(requestBody) - if err != nil { - return APITestResult{Success: false, Error: fmt.Sprintf("Marshal error: %v", err)} - } - - req, err := http.NewRequest("POST", provider.BaseURL+"/chat/completions", bytes.NewReader(body)) - if err != nil { - return APITestResult{Success: false, Error: fmt.Sprintf("Request error: %v", err)} - } - - req.Header.Set("Content-Type", "application/json") - req.Header.Set("Authorization", "Bearer "+provider.APIKey) - - client := &http.Client{Timeout: 60 * time.Second} - resp, err := client.Do(req) - if err != nil { - return APITestResult{Success: false, Error: fmt.Sprintf("HTTP error: %v", err)} - } - defer resp.Body.Close() - - duration := time.Since(start) - - if resp.StatusCode != 200 { - bodyBytes, _ := io.ReadAll(resp.Body) - errorMsg := string(bodyBytes) - // Try to parse error response for better readability - var errorResp map[string]interface{} - if json.Unmarshal(bodyBytes, &errorResp) == nil { - if errObj, ok := errorResp["error"].(map[string]interface{}); ok { - if message, ok := errObj["message"].(string); ok { - errorMsg = message - } - } - } - return APITestResult{ - Success: false, - Error: fmt.Sprintf("Status %d: %s", resp.StatusCode, errorMsg), - Duration: duration, - } - } - - var result map[string]interface{} - if err := json.NewDecoder(resp.Body).Decode(&result); err != nil { - return APITestResult{Success: false, Error: fmt.Sprintf("Decode error: %v", err)} - } - - content := extractContent(result) - return APITestResult{ - Success: true, - Response: content, - Duration: duration, - } -} - -func extractContent(response map[string]interface{}) string { - if choices, ok := response["choices"].([]interface{}); ok && len(choices) > 0 { - if choice, ok := choices[0].(map[string]interface{}); ok { - if message, ok := choice["message"].(map[string]interface{}); ok { - if content, ok := message["content"].(string); ok { - return content - } - } - } - } - return "" -} - -type ComplianceResult struct { - Valid bool - Score float64 - MissingFields []string - ExtraFields []string -} - -func validateSchemaCompliance(jsonStr string) ComplianceResult { - var data map[string]interface{} - if err := json.Unmarshal([]byte(jsonStr), &data); err != nil { - return ComplianceResult{Valid: false, Score: 0} - } - - missing := []string{} - extra := []string{} - score := 100.0 - - // Check required top-level fields - requiredFields := []string{"articles", "totalResults", "fetchedAt"} - for _, field := range requiredFields { - if _, ok := data[field]; !ok { - missing = append(missing, field) - score -= 33.33 - } - } - - // Check for extra fields (should have additionalProperties: false) - allowedFields := map[string]bool{"articles": true, "totalResults": true, "fetchedAt": true} - for field := range data { - if !allowedFields[field] { - extra = append(extra, field) - score -= 10 - } - } - - // Check articles array structure - if articles, ok := data["articles"].([]interface{}); ok { - for i, article := range articles { - if articleMap, ok := article.(map[string]interface{}); ok { - articleRequired := []string{"title", "source", "url", "summary"} - for _, field := range articleRequired { - if _, ok := articleMap[field]; !ok { - missing = append(missing, fmt.Sprintf("articles[%d].%s", i, field)) - score -= 5 - } - } - } - } - } - - if score < 0 { - score = 0 - } - - return ComplianceResult{ - Valid: len(missing) == 0 && len(extra) == 0, - Score: score, - MissingFields: missing, - ExtraFields: extra, - } -} - -func printResult(result TestResult) { - // Summary line - status := "❌ NO SUPPORT" - if result.SupportsStrict { - status = "✅ FULL SUPPORT" - } else if result.SupportsJSON { - status = "⚠️ PARTIAL SUPPORT" - } - - fmt.Printf(" Result: %s (Compliance: %.1f%%)\n", status, result.SchemaCompliance) - - if len(result.MissingFields) > 0 { - fmt.Printf(" ⚠️ Missing Fields: %v\n", result.MissingFields) - } - - if len(result.ExtraFields) > 0 { - fmt.Printf(" ⚠️ Extra Fields: %v\n", result.ExtraFields) - } - - if result.Error != "" && !result.SupportsStrict && !result.SupportsJSON { - fmt.Printf(" Error: %s\n", truncate(result.Error, 100)) - } -} - -func generateReport(results []TestResult) { - fmt.Println() - fmt.Println("📊 SUMMARY REPORT") - fmt.Println("=" + strings.Repeat("=", 79)) - fmt.Println() - - // Group by support level - strictSupport := []TestResult{} - jsonSupport := []TestResult{} - noSupport := []TestResult{} - - for _, r := range results { - if r.SupportsStrict { - strictSupport = append(strictSupport, r) - } else if r.SupportsJSON { - jsonSupport = append(jsonSupport, r) - } else { - noSupport = append(noSupport, r) - } - } - - fmt.Printf("✅ FULL SUPPORT (Strict JSON Schema): %d models\n", len(strictSupport)) - for _, r := range strictSupport { - fmt.Printf(" • %s / %s (%.1f%% compliance, %v)\n", - r.Provider, r.Model, r.SchemaCompliance, r.ResponseTime) - } - fmt.Println() - - fmt.Printf("⚠️ PARTIAL SUPPORT (JSON Object only): %d models\n", len(jsonSupport)) - for _, r := range jsonSupport { - fmt.Printf(" • %s / %s (%.1f%% compliance)\n", - r.Provider, r.Model, r.SchemaCompliance) - } - fmt.Println() - - fmt.Printf("❌ NO SUPPORT: %d models\n", len(noSupport)) - for _, r := range noSupport { - fmt.Printf(" • %s / %s - %s\n", r.Provider, r.Model, r.Error) - } - fmt.Println() - - // Recommendations - fmt.Println("💡 RECOMMENDATIONS FOR PRODUCTION") - fmt.Println(strings.Repeat("-", 80)) - - if len(strictSupport) > 0 { - best := strictSupport[0] - for _, r := range strictSupport { - if r.SchemaCompliance > best.SchemaCompliance { - best = r - } - } - fmt.Printf("🏆 BEST: %s / %s (%.1f%% compliance, %v response time)\n", - best.Provider, best.Model, best.SchemaCompliance, best.ResponseTime) - fmt.Println(" → Use for production workflows requiring guaranteed structure") - } - - if len(jsonSupport) > 0 { - fmt.Println() - fmt.Println("⚠️ FALLBACK OPTIONS (require prompt engineering):") - for _, r := range jsonSupport { - fmt.Printf(" • %s / %s - Add schema to system prompt\n", r.Provider, r.Model) - } - } - - fmt.Println() - fmt.Println("🔧 IMPLEMENTATION STRATEGY:") - fmt.Println(" 1. Detect provider capability at runtime") - fmt.Println(" 2. Use strict mode for OpenAI and compatible providers") - fmt.Println(" 3. Fallback to JSON mode + prompt schema for others") - fmt.Println(" 4. Validate output and retry with stronger prompts if needed") -} - -func truncate(s string, length int) string { - if len(s) <= length { - return s - } - return s[:length] + "..." -} diff --git a/backend/scripts/validate_migration.go b/backend/scripts/validate_migration.go deleted file mode 100644 index 60cc6112..00000000 --- a/backend/scripts/validate_migration.go +++ /dev/null @@ -1,123 +0,0 @@ -package main - -import ( - "database/sql" - "fmt" - "log" - "os" - "strings" - - _ "github.com/go-sql-driver/mysql" - _ "modernc.org/sqlite" -) - -func main() { - // Read configuration - sqlitePath := getEnv("SQLITE_PATH", "./model_capabilities.db") - mysqlDSN := getEnv("MYSQL_DSN", "") - - if mysqlDSN == "" { - log.Fatal("❌ MYSQL_DSN environment variable required") - } - - log.Println("🔍 Validating migration...") - log.Printf(" SQLite: %s", sqlitePath) - log.Printf(" MySQL: %s\n", maskDSN(mysqlDSN)) - - // Open databases - sqliteDB, err := sql.Open("sqlite", sqlitePath) - if err != nil { - log.Fatalf("❌ Failed to open SQLite: %v", err) - } - defer sqliteDB.Close() - - mysqlDB, err := sql.Open("mysql", mysqlDSN+"?parseTime=true") - if err != nil { - log.Fatalf("❌ Failed to open MySQL: %v", err) - } - defer mysqlDB.Close() - - // Test connections - if err := sqliteDB.Ping(); err != nil { - log.Fatalf("❌ SQLite connection failed: %v", err) - } - if err := mysqlDB.Ping(); err != nil { - log.Fatalf("❌ MySQL connection failed: %v", err) - } - - // Tables to validate - tables := []string{ - "providers", - "models", - "model_aliases", - "provider_model_filters", - "model_capabilities", - "model_refresh_log", - } - - allMatch := true - for _, table := range tables { - var sqliteCount, mysqlCount int - - // Get SQLite count - query := fmt.Sprintf("SELECT COUNT(*) FROM %s", table) - err := sqliteDB.QueryRow(query).Scan(&sqliteCount) - if err != nil { - if strings.Contains(err.Error(), "no such table") { - log.Printf(" ⚠️ %s: Table doesn't exist in SQLite (skipping)", table) - continue - } - log.Printf(" ❌ %s: Failed to query SQLite: %v", table, err) - allMatch = false - continue - } - - // Get MySQL count - err = mysqlDB.QueryRow(query).Scan(&mysqlCount) - if err != nil { - log.Printf(" ❌ %s: Failed to query MySQL: %v", table, err) - allMatch = false - continue - } - - // Compare - if sqliteCount != mysqlCount { - log.Printf(" ❌ %s: SQLite=%d, MySQL=%d (MISMATCH)", table, sqliteCount, mysqlCount) - allMatch = false - } else if sqliteCount > 0 { - log.Printf(" ✅ %s: %d records match", table, mysqlCount) - } else { - log.Printf(" ℹ️ %s: 0 records (empty table)", table) - } - } - - // Summary - fmt.Println("\n" + strings.Repeat("=", 60)) - if allMatch { - log.Println("✅ Validation PASSED - All record counts match!") - } else { - log.Println("❌ Validation FAILED - Some counts don't match") - log.Println(" Review errors above and re-run migration if needed") - os.Exit(1) - } - log.Println(strings.Repeat("=", 60)) -} - -func getEnv(key, fallback string) string { - if val := os.Getenv(key); val != "" { - return val - } - return fallback -} - -func maskDSN(dsn string) string { - parts := strings.Split(dsn, "@") - if len(parts) < 2 { - return dsn - } - userPass := strings.Split(parts[0], ":") - if len(userPass) < 2 { - return dsn - } - return userPass[0] + ":***@" + parts[1] -} diff --git a/backend/scripts/validate_workflow_generator.go b/backend/scripts/validate_workflow_generator.go deleted file mode 100644 index 3f478958..00000000 --- a/backend/scripts/validate_workflow_generator.go +++ /dev/null @@ -1,166 +0,0 @@ -package main - -import ( - "encoding/json" - "fmt" - "strings" -) - -// This script validates that workflow generator creates compliant schemas -// Run after making test workflows to ensure they pass our own standards - -func main() { - fmt.Println("🔍 Workflow Generator Compliance Validator") - fmt.Println(strings.Repeat("=", 60)) - fmt.Println() - - // Example workflow block from workflow generator - exampleBlock := map[string]interface{}{ - "type": "llm_inference", - "name": "News Fetcher", - "config": map[string]interface{}{ - "outputFormat": "json", - "outputSchema": map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "articles": map[string]interface{}{ - "type": "array", - "items": map[string]interface{}{ - "type": "object", - "properties": map[string]interface{}{ - "title": map[string]interface{}{"type": "string"}, - "source": map[string]interface{}{"type": "string"}, - "url": map[string]interface{}{"type": "string"}, - "summary": map[string]interface{}{"type": "string"}, - "publishedDate": map[string]interface{}{"type": "string"}, - }, - "required": []interface{}{"title", "source", "url", "summary", "publishedDate"}, - "additionalProperties": false, - }, - }, - "totalResults": map[string]interface{}{"type": "number"}, - "fetchedAt": map[string]interface{}{"type": "string"}, - }, - "required": []interface{}{"articles", "totalResults", "fetchedAt"}, - "additionalProperties": false, - }, - }, - } - - config := exampleBlock["config"].(map[string]interface{}) - schema := config["outputSchema"].(map[string]interface{}) - - issues := validateSchema(schema, "root") - - if len(issues) == 0 { - fmt.Println("✅ Schema is VALID and compliant!") - fmt.Println() - fmt.Println("✓ All properties are in required arrays") - fmt.Println("✓ additionalProperties: false is set") - fmt.Println("✓ Nested objects have complete required arrays") - fmt.Println() - fmt.Println("This schema will pass OpenAI strict mode validation.") - } else { - fmt.Println("❌ Schema has ISSUES:") - fmt.Println() - for _, issue := range issues { - fmt.Printf(" • %s\n", issue) - } - fmt.Println() - fmt.Println("⚠️ This schema will FAIL OpenAI strict mode validation!") - } - - // Pretty print schema - fmt.Println() - fmt.Println("📋 Schema Structure:") - fmt.Println(strings.Repeat("-", 60)) - schemaJSON, _ := json.MarshalIndent(schema, "", " ") - fmt.Println(string(schemaJSON)) -} - -func validateSchema(schema map[string]interface{}, path string) []string { - var issues []string - - // Check if type is object - schemaType, hasType := schema["type"].(string) - if !hasType { - issues = append(issues, fmt.Sprintf("%s: Missing 'type' field", path)) - return issues - } - - if schemaType == "object" { - // Get properties - properties, hasProps := schema["properties"].(map[string]interface{}) - if !hasProps { - issues = append(issues, fmt.Sprintf("%s: Object type missing 'properties'", path)) - return issues - } - - // Get required array - requiredRaw, hasRequired := schema["required"] - if !hasRequired { - issues = append(issues, fmt.Sprintf("%s: Missing 'required' array (OpenAI strict mode requires it)", path)) - } else { - required, ok := requiredRaw.([]interface{}) - if !ok { - issues = append(issues, fmt.Sprintf("%s: 'required' is not an array", path)) - } else { - // Check if all properties are in required - requiredMap := make(map[string]bool) - for _, r := range required { - if rStr, ok := r.(string); ok { - requiredMap[rStr] = true - } - } - - // Check each property - for propName := range properties { - if !requiredMap[propName] { - issues = append(issues, fmt.Sprintf("%s.%s: Property defined but NOT in required array (OpenAI strict mode rejects this)", path, propName)) - } - } - - // Check if all required fields exist in properties - for _, r := range required { - if rStr, ok := r.(string); ok { - if _, exists := properties[rStr]; !exists { - issues = append(issues, fmt.Sprintf("%s: Required field '%s' not in properties", path, rStr)) - } - } - } - } - } - - // Check additionalProperties - if additionalProps, hasAdditional := schema["additionalProperties"]; hasAdditional { - if additionalBool, ok := additionalProps.(bool); ok && additionalBool { - issues = append(issues, fmt.Sprintf("%s: additionalProperties should be false for strict schemas", path)) - } - } else { - issues = append(issues, fmt.Sprintf("%s: Missing 'additionalProperties: false' (recommended for strict schemas)", path)) - } - - // Recursively validate nested objects - for propName, propValue := range properties { - if propSchema, ok := propValue.(map[string]interface{}); ok { - propType, _ := propSchema["type"].(string) - - if propType == "object" { - nestedIssues := validateSchema(propSchema, fmt.Sprintf("%s.%s", path, propName)) - issues = append(issues, nestedIssues...) - } else if propType == "array" { - // Check array items - if items, hasItems := propSchema["items"].(map[string]interface{}); hasItems { - itemType, _ := items["type"].(string) - if itemType == "object" { - nestedIssues := validateSchema(items, fmt.Sprintf("%s.%s[items]", path, propName)) - issues = append(issues, nestedIssues...) - } - } - } - } - } - } - - return issues -} diff --git a/backend/server b/backend/server deleted file mode 100755 index 94eaa7a6..00000000 Binary files a/backend/server and /dev/null differ diff --git a/backend/setup-providers.sh b/backend/setup-providers.sh deleted file mode 100644 index a49bd2ba..00000000 --- a/backend/setup-providers.sh +++ /dev/null @@ -1,176 +0,0 @@ -#!/bin/bash - -# ClaraVerse Provider Setup Script -# This script helps you quickly set up providers and filters - -echo "🚀 ClaraVerse Provider Setup" -echo "==============================" -echo "" - -BASE_URL="http://localhost:3001" - -# Function to add a provider -add_provider() { - local name=$1 - local base_url=$2 - local api_key=$3 - - echo "📡 Adding provider: $name" - - response=$(curl -s -X POST "$BASE_URL/api/providers" \ - -H "Content-Type: application/json" \ - -d "{ - \"name\": \"$name\", - \"base_url\": \"$base_url\", - \"api_key\": \"$api_key\" - }") - - provider_id=$(echo $response | grep -o '"id":[0-9]*' | grep -o '[0-9]*') - - if [ -n "$provider_id" ]; then - echo " ✅ Provider added with ID: $provider_id" - echo "$provider_id" - else - echo " ❌ Failed to add provider" - echo " Response: $response" - echo "0" - fi -} - -# Function to add a filter -add_filter() { - local provider_id=$1 - local pattern=$2 - local action=$3 - local priority=$4 - - echo " 🔧 Adding filter: $pattern ($action)" - - curl -s -X POST "$BASE_URL/api/providers/$provider_id/filters" \ - -H "Content-Type: application/json" \ - -d "{ - \"pattern\": \"$pattern\", - \"action\": \"$action\", - \"priority\": $priority - }" > /dev/null -} - -# Function to refresh models -refresh_models() { - local provider_id=$1 - - echo " 🔄 Refreshing models for provider $provider_id..." - - curl -s -X POST "$BASE_URL/api/models/refresh/$provider_id" > /dev/null - - echo " ✅ Models refreshed" -} - -echo "This script will set up example providers." -echo "You will need to provide your API keys." -echo "" - -# Check if server is running -if ! curl -s "$BASE_URL/health" > /dev/null 2>&1; then - echo "❌ Error: Server is not running at $BASE_URL" - echo "Please start the server first with: ./claraverse-server" - exit 1 -fi - -echo "✅ Server is running" -echo "" - -# Setup OpenAI -echo "Would you like to add OpenAI? (y/n)" -read -r add_openai - -if [ "$add_openai" = "y" ] || [ "$add_openai" = "Y" ]; then - echo "Enter your OpenAI API key:" - read -r openai_key - - if [ -n "$openai_key" ]; then - provider_id=$(add_provider "OpenAI" "https://api.openai.com/v1" "$openai_key") - - if [ "$provider_id" != "0" ]; then - echo " Adding filters to show only GPT-4 and GPT-3.5-turbo models..." - add_filter "$provider_id" "gpt-4o" "include" 20 - add_filter "$provider_id" "gpt-4o-mini" "include" 15 - add_filter "$provider_id" "gpt-3.5-turbo" "include" 10 - refresh_models "$provider_id" - fi - fi - echo "" -fi - -# Setup Anthropic -echo "Would you like to add Anthropic? (y/n)" -read -r add_anthropic - -if [ "$add_anthropic" = "y" ] || [ "$add_anthropic" = "Y" ]; then - echo "Enter your Anthropic API key:" - read -r anthropic_key - - if [ -n "$anthropic_key" ]; then - provider_id=$(add_provider "Anthropic" "https://api.anthropic.com/v1" "$anthropic_key") - - if [ "$provider_id" != "0" ]; then - echo " Adding filters to show Claude 3 models..." - add_filter "$provider_id" "claude-3-*" "include" 10 - refresh_models "$provider_id" - fi - fi - echo "" -fi - -# Setup Z.AI -echo "Would you like to add Z.AI? (y/n)" -read -r add_zai - -if [ "$add_zai" = "y" ] || [ "$add_zai" = "Y" ]; then - echo "Enter your Z.AI API key:" - read -r zai_key - - if [ -n "$zai_key" ]; then - provider_id=$(add_provider "Z.AI" "https://api.z.ai/api/coding/paas/v4" "$zai_key") - - if [ "$provider_id" != "0" ]; then - echo " Adding filters to show GLM-4 models..." - add_filter "$provider_id" "glm-4*" "include" 10 - refresh_models "$provider_id" - fi - fi - echo "" -fi - -# Custom provider -echo "Would you like to add a custom OpenAI-compatible provider? (y/n)" -read -r add_custom - -if [ "$add_custom" = "y" ] || [ "$add_custom" = "Y" ]; then - echo "Enter provider name:" - read -r custom_name - - echo "Enter base URL (e.g., https://api.example.com/v1):" - read -r custom_url - - echo "Enter API key:" - read -r custom_key - - if [ -n "$custom_name" ] && [ -n "$custom_url" ] && [ -n "$custom_key" ]; then - provider_id=$(add_provider "$custom_name" "$custom_url" "$custom_key") - - if [ "$provider_id" != "0" ]; then - refresh_models "$provider_id" - fi - fi - echo "" -fi - -echo "==============================" -echo "✅ Setup Complete!" -echo "" -echo "You can now:" -echo " • List all models: curl $BASE_URL/api/models" -echo " • List providers: curl $BASE_URL/api/providers" -echo " • View API documentation: cat API_DOCUMENTATION.md" -echo "" diff --git a/backend/test-client.html b/backend/test-client.html deleted file mode 100644 index 0bca134e..00000000 --- a/backend/test-client.html +++ /dev/null @@ -1,1622 +0,0 @@ - - - - - - ClaraVerse Multi-User WebSocket Test - - - - - - - -
-

🤖 ClaraVerse Multi-User Test

- -
-

Status: Disconnected

-

Connection ID: -

-

Conversation ID: -

-

Config: Using Default

-
- - -
-

🔐 Authentication (Optional)

-
- - -
- - -
- - -
-

- Leave empty to use without authentication -

-
- - - - - -
- - ⚙️ Configure Supabase Credentials - -
- - - -

- Get your credentials from: - - Supabase Dashboard → Settings → API - -

-
-
-
- - -
-

🤖 Select AI Model

-
- - -
-
- - -
-
- -
-

- ⚙️ Custom API Configuration - -

-
-
- - - OpenAI, OpenRouter, or any OpenAI-compatible API endpoint -
-
- - - Your API key (stored locally, not sent to server permanently) -
-
- - - Model name or ID -
- - -
-
- -
-
-
- - - -
-
-
- - - - diff --git a/backend/test-upload.html b/backend/test-upload.html deleted file mode 100644 index 6fa619e1..00000000 --- a/backend/test-upload.html +++ /dev/null @@ -1,496 +0,0 @@ - - - - - - ClaraVerse Upload Test - - - -
-

🖼️ ClaraVerse Upload Test

- - -
-

📋 How to Test

-
    -
  1. Get your auth token from test-client.html (login and check browser console)
  2. -
  3. Paste the token in the field below
  4. -
  5. Select an image file (PNG, JPG, WebP, GIF - max 20MB)
  6. -
  7. Click "Upload Image"
  8. -
  9. If successful, you'll see the uploaded image and can test deletion
  10. -
-
- - -
-

🔐 1. Authentication

-
- - - Get this from test-client.html after logging in -
-
- -
-

📤 2. Upload Image

-
- - - Supported: PNG, JPG, WebP, GIF (max 20MB) -
- - - -
- -
-

🗑️ 3. Delete Image

-
- - - Auto-filled from upload response -
- - -
- -
-

💬 4. WebSocket Test with Vision Model

-
- - -
- - -
-
- - - - diff --git a/backend/tests/integration_test.go b/backend/tests/integration_test.go deleted file mode 100644 index e30cfae5..00000000 --- a/backend/tests/integration_test.go +++ /dev/null @@ -1,451 +0,0 @@ -package tests - -import ( - "claraverse/internal/database" - "claraverse/internal/models" - "claraverse/internal/services" - "encoding/json" - "net/http/httptest" - "os" - "testing" - - "github.com/gofiber/fiber/v2" -) - -// Integration tests verify that all components work together correctly - -func setupIntegrationTest(t *testing.T) (*fiber.App, *database.DB, func()) { - t.Skip("SQLite tests are deprecated - integration tests require MySQL DSN via DATABASE_URL") - tmpFile := "test_integration.db" - db, err := database.New(tmpFile) - if err != nil { - t.Fatalf("Failed to create test database: %v", err) - } - - if err := db.Initialize(); err != nil { - t.Fatalf("Failed to initialize test database: %v", err) - } - - app := fiber.New() - - cleanup := func() { - db.Close() - os.Remove(tmpFile) - } - - return app, db, cleanup -} - -// TestFullProviderAndModelFlow tests the complete flow of: -// 1. Creating a provider -// 2. Adding models to that provider -// 3. Applying filters -// 4. Fetching models via API -func TestFullProviderAndModelFlow(t *testing.T) { - _, db, cleanup := setupIntegrationTest(t) - defer cleanup() - - // Initialize services - providerService := services.NewProviderService(db) - modelService := services.NewModelService(db) - - // Step 1: Create a provider - config := models.ProviderConfig{ - Name: "OpenAI", - BaseURL: "https://api.openai.com/v1", - APIKey: "test-key", - Enabled: true, - Filters: []models.FilterConfig{ - {Pattern: "gpt-4*", Action: "include", Priority: 10}, - {Pattern: "*preview*", Action: "exclude", Priority: 5}, - }, - } - - provider, err := providerService.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - // Step 2: Sync filters - if err := providerService.SyncFilters(provider.ID, config.Filters); err != nil { - t.Fatalf("Failed to sync filters: %v", err) - } - - // Step 3: Add models manually (simulate fetch) - testModels := []models.Model{ - {ID: "gpt-4-turbo", ProviderID: provider.ID, Name: "gpt-4-turbo", IsVisible: false}, - {ID: "gpt-4-preview", ProviderID: provider.ID, Name: "gpt-4-preview", IsVisible: false}, - {ID: "gpt-3.5-turbo", ProviderID: provider.ID, Name: "gpt-3.5-turbo", IsVisible: false}, - } - - for _, model := range testModels { - _, err := db.Exec(` - INSERT INTO models (id, provider_id, name, is_visible) - VALUES (?, ?, ?, ?) - `, model.ID, model.ProviderID, model.Name, model.IsVisible) - if err != nil { - t.Fatalf("Failed to insert model: %v", err) - } - } - - // Step 4: Apply filters - if err := providerService.ApplyFilters(provider.ID); err != nil { - t.Fatalf("Failed to apply filters: %v", err) - } - - // Step 5: Verify visible models - visibleModels, err := modelService.GetByProvider(provider.ID, true) - if err != nil { - t.Fatalf("Failed to get visible models: %v", err) - } - - // Should have 1 visible model (gpt-4-turbo) - // gpt-4-preview is excluded by filter - // gpt-3.5-turbo doesn't match include filter - if len(visibleModels) != 1 { - t.Errorf("Expected 1 visible model, got %d", len(visibleModels)) - } - - if len(visibleModels) > 0 && visibleModels[0].ID != "gpt-4-turbo" { - t.Errorf("Expected gpt-4-turbo to be visible, got %s", visibleModels[0].ID) - } -} - -// TestMultiProviderScenario tests handling multiple providers with different models -func TestMultiProviderScenario(t *testing.T) { - _, db, cleanup := setupIntegrationTest(t) - defer cleanup() - - providerService := services.NewProviderService(db) - modelService := services.NewModelService(db) - - // Create multiple providers - providers := []models.ProviderConfig{ - { - Name: "OpenAI", - BaseURL: "https://api.openai.com/v1", - APIKey: "openai-key", - Enabled: true, - Filters: []models.FilterConfig{ - {Pattern: "gpt-4*", Action: "include", Priority: 10}, - }, - }, - { - Name: "Anthropic", - BaseURL: "https://api.anthropic.com/v1", - APIKey: "anthropic-key", - Enabled: true, - Filters: []models.FilterConfig{ - {Pattern: "claude-3*", Action: "include", Priority: 10}, - }, - }, - } - - var createdProviders []*models.Provider - for _, config := range providers { - provider, err := providerService.Create(config) - if err != nil { - t.Fatalf("Failed to create provider %s: %v", config.Name, err) - } - createdProviders = append(createdProviders, provider) - - // Sync filters - if err := providerService.SyncFilters(provider.ID, config.Filters); err != nil { - t.Fatalf("Failed to sync filters for %s: %v", config.Name, err) - } - } - - // Add models for OpenAI - openAIModels := []string{"gpt-4-turbo", "gpt-3.5-turbo"} - for _, modelID := range openAIModels { - _, err := db.Exec(` - INSERT INTO models (id, provider_id, name, is_visible) - VALUES (?, ?, ?, ?) - `, modelID, createdProviders[0].ID, modelID, false) - if err != nil { - t.Fatalf("Failed to insert OpenAI model: %v", err) - } - } - - // Add models for Anthropic - anthropicModels := []string{"claude-3-opus", "claude-2"} - for _, modelID := range anthropicModels { - _, err := db.Exec(` - INSERT INTO models (id, provider_id, name, is_visible) - VALUES (?, ?, ?, ?) - `, modelID, createdProviders[1].ID, modelID, false) - if err != nil { - t.Fatalf("Failed to insert Anthropic model: %v", err) - } - } - - // Apply filters for both providers - for _, provider := range createdProviders { - if err := providerService.ApplyFilters(provider.ID); err != nil { - t.Fatalf("Failed to apply filters for provider %d: %v", provider.ID, err) - } - } - - // Verify OpenAI models - openAIVisible, err := modelService.GetByProvider(createdProviders[0].ID, true) - if err != nil { - t.Fatalf("Failed to get OpenAI models: %v", err) - } - - if len(openAIVisible) != 1 || openAIVisible[0].ID != "gpt-4-turbo" { - t.Errorf("Expected gpt-4-turbo to be visible for OpenAI") - } - - // Verify Anthropic models - anthropicVisible, err := modelService.GetByProvider(createdProviders[1].ID, true) - if err != nil { - t.Fatalf("Failed to get Anthropic models: %v", err) - } - - if len(anthropicVisible) != 1 || anthropicVisible[0].ID != "claude-3-opus" { - t.Errorf("Expected claude-3-opus to be visible for Anthropic") - } - - // Get all visible models - allVisible, err := modelService.GetAll(true) - if err != nil { - t.Fatalf("Failed to get all visible models: %v", err) - } - - if len(allVisible) != 2 { - t.Errorf("Expected 2 visible models total, got %d", len(allVisible)) - } -} - -// TestProviderDisableEnableFlow tests disabling and enabling providers -func TestProviderDisableEnableFlow(t *testing.T) { - _, db, cleanup := setupIntegrationTest(t) - defer cleanup() - - providerService := services.NewProviderService(db) - - // Create enabled provider - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: "https://api.test.com/v1", - APIKey: "test-key", - Enabled: true, - } - - provider, err := providerService.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - // Verify it appears in list - providers, err := providerService.GetAll() - if err != nil { - t.Fatalf("Failed to get providers: %v", err) - } - - if len(providers) != 1 { - t.Errorf("Expected 1 enabled provider, got %d", len(providers)) - } - - // Disable provider - disableConfig := config - disableConfig.Enabled = false - if err := providerService.Update(provider.ID, disableConfig); err != nil { - t.Fatalf("Failed to disable provider: %v", err) - } - - // Verify it doesn't appear in list - providers, err = providerService.GetAll() - if err != nil { - t.Fatalf("Failed to get providers: %v", err) - } - - if len(providers) != 0 { - t.Errorf("Expected 0 enabled providers, got %d", len(providers)) - } - - // Re-enable provider - enableConfig := config - enableConfig.Enabled = true - if err := providerService.Update(provider.ID, enableConfig); err != nil { - t.Fatalf("Failed to enable provider: %v", err) - } - - // Verify it appears again - providers, err = providerService.GetAll() - if err != nil { - t.Fatalf("Failed to get providers: %v", err) - } - - if len(providers) != 1 { - t.Errorf("Expected 1 enabled provider, got %d", len(providers)) - } -} - -// TestDatabaseForeignKeyIntegrity tests that foreign key constraints are enforced -func TestDatabaseForeignKeyIntegrity(t *testing.T) { - _, db, cleanup := setupIntegrationTest(t) - defer cleanup() - - providerService := services.NewProviderService(db) - - // Create provider - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: "https://api.test.com/v1", - APIKey: "test-key", - Enabled: true, - } - - provider, err := providerService.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - // Add model for this provider - _, err = db.Exec(` - INSERT INTO models (id, provider_id, name) - VALUES (?, ?, ?) - `, "test-model", provider.ID, "Test Model") - if err != nil { - t.Fatalf("Failed to insert model: %v", err) - } - - // Try to add model with non-existent provider (should fail) - _, err = db.Exec(` - INSERT INTO models (id, provider_id, name) - VALUES (?, ?, ?) - `, "invalid-model", 9999, "Invalid Model") - - if err == nil { - t.Error("Expected foreign key constraint error, got nil") - } - - // Verify cascade delete - _, err = db.Exec("DELETE FROM providers WHERE id = ?", provider.ID) - if err != nil { - t.Fatalf("Failed to delete provider: %v", err) - } - - // Verify model was also deleted - var count int - err = db.QueryRow("SELECT COUNT(*) FROM models WHERE provider_id = ?", provider.ID).Scan(&count) - if err != nil { - t.Fatalf("Failed to count models: %v", err) - } - - if count != 0 { - t.Errorf("Expected 0 models after provider deletion, got %d", count) - } -} - -// TestFilterPriorityOrdering tests that filters are applied in correct priority order -func TestFilterPriorityOrdering(t *testing.T) { - _, db, cleanup := setupIntegrationTest(t) - defer cleanup() - - providerService := services.NewProviderService(db) - modelService := services.NewModelService(db) - - // Create provider with overlapping filters - config := models.ProviderConfig{ - Name: "Test Provider", - BaseURL: "https://api.test.com/v1", - APIKey: "test-key", - Enabled: true, - Filters: []models.FilterConfig{ - {Pattern: "gpt-*", Action: "include", Priority: 5}, // Lower priority - {Pattern: "*turbo*", Action: "exclude", Priority: 10}, // Higher priority - }, - } - - provider, err := providerService.Create(config) - if err != nil { - t.Fatalf("Failed to create provider: %v", err) - } - - if err := providerService.SyncFilters(provider.ID, config.Filters); err != nil { - t.Fatalf("Failed to sync filters: %v", err) - } - - // Add test models - _, err = db.Exec(` - INSERT INTO models (id, provider_id, name, is_visible) - VALUES - ('gpt-4', ?, 'gpt-4', 0), - ('gpt-4-turbo', ?, 'gpt-4-turbo', 0) - `, provider.ID, provider.ID) - if err != nil { - t.Fatalf("Failed to insert models: %v", err) - } - - // Apply filters - if err := providerService.ApplyFilters(provider.ID); err != nil { - t.Fatalf("Failed to apply filters: %v", err) - } - - // Get visible models - visibleModels, err := modelService.GetByProvider(provider.ID, true) - if err != nil { - t.Fatalf("Failed to get visible models: %v", err) - } - - // The filter logic applies filters in priority order (higher first): - // 1. Higher priority (10): *turbo* exclude - would exclude gpt-4-turbo - // 2. Lower priority (5): gpt-* include - would include both - // However, the actual implementation processes in this order: - // - First, it includes all gpt-* models (both models) - // - Then, it excludes *turbo* models (removes gpt-4-turbo) - // Result: only gpt-4 should be visible - // - // BUT: Looking at the code, filters are ordered by priority DESC, - // so exclude runs first, then include. This means: - // 1. Reset all to invisible - // 2. Exclude *turbo* (no effect, already invisible) - // 3. Include gpt-* (makes both visible) - // So both end up visible! - // - // The test expectation was wrong - both should be visible - if len(visibleModels) != 2 { - t.Errorf("Expected 2 visible models (filters don't work as expected), got %d", len(visibleModels)) - for _, m := range visibleModels { - t.Logf(" Visible model: %s", m.ID) - } - } -} - -// TestHealthCheckIntegration tests the health check endpoint integration -func TestHealthCheckIntegration(t *testing.T) { - app, _, cleanup := setupIntegrationTest(t) - defer cleanup() - - // Setup health endpoint - connManager := services.NewConnectionManager() - app.Get("/health", func(c *fiber.Ctx) error { - return c.JSON(fiber.Map{ - "status": "healthy", - "connections": connManager.Count(), - }) - }) - - // Test health endpoint - req := httptest.NewRequest("GET", "/health", nil) - resp, err := app.Test(req) - if err != nil { - t.Fatalf("Failed to send request: %v", err) - } - defer resp.Body.Close() - - if resp.StatusCode != 200 { - t.Errorf("Expected status 200, got %d", resp.StatusCode) - } - - var result map[string]interface{} - if err := json.NewDecoder(resp.Body).Decode(&result); err != nil { - t.Fatalf("Failed to decode response: %v", err) - } - - if result["status"] != "healthy" { - t.Errorf("Expected status 'healthy', got %v", result["status"]) - } -} diff --git a/build/entitlements.mac.plist b/build/entitlements.mac.plist new file mode 100644 index 00000000..b05b20be --- /dev/null +++ b/build/entitlements.mac.plist @@ -0,0 +1,20 @@ + + + + + com.apple.security.cs.allow-jit + + com.apple.security.cs.allow-unsigned-executable-memory + + com.apple.security.cs.disable-library-validation + + com.apple.security.network.client + + com.apple.security.files.user-selected.read-write + + com.apple.security.files.downloads.read-write + + com.apple.security.device.audio-input + + + diff --git a/build_and_push.sh b/build_and_push.sh new file mode 100755 index 00000000..7bda06e8 --- /dev/null +++ b/build_and_push.sh @@ -0,0 +1,37 @@ +#!/bin/bash +set -e + +# Configuration +IMAGE_NAME="clara17verse/clara-interpreter" +TAG="latest" +DOCKERFILE_PATH="./clara_interpreter_dockerstuff/Dockerfile" # Updated path to Dockerfile + +# Login to Docker Hub - uncomment and run manually if not logged in +# docker login -u clara17verse + +# Clean up any existing builder instances +docker buildx ls | grep -q multiarch-builder && docker buildx rm multiarch-builder || true +docker context rm multiarch-context 2>/dev/null || true + +# Create a new context and builder +docker context create multiarch-context 2>/dev/null || true +docker buildx create --name multiarch-builder --driver docker-container --driver-opt network=host --use multiarch-context + +# Build for multiple platforms and push to Docker Hub +echo "Building and pushing multi-architecture image: ${IMAGE_NAME}:${TAG}" +docker buildx build --platform linux/amd64,linux/arm64 \ + -t ${IMAGE_NAME}:${TAG} \ + --push \ + -f ${DOCKERFILE_PATH} . + +echo "Successfully built and pushed ${IMAGE_NAME}:${TAG} for multiple architectures" + +# Optional: List the supported architectures of the pushed image +echo "Listing supported architectures:" +docker buildx imagetools inspect ${IMAGE_NAME}:${TAG} + +# Clean up +docker buildx rm multiarch-builder +docker context rm multiarch-context + +echo "Build and push completed successfully!" \ No newline at end of file diff --git a/clara-mcp/build-all.bat b/clara-mcp/build-all.bat new file mode 100644 index 00000000..4be0adcc --- /dev/null +++ b/clara-mcp/build-all.bat @@ -0,0 +1,61 @@ +@echo off +echo Building Clara MCP Server for all platforms... + +REM Set the output directory +set OUTPUT_DIR=../electron/mcp + +REM Create output directory if it doesn't exist +if not exist "%OUTPUT_DIR%" mkdir "%OUTPUT_DIR%" + +REM Build for Windows (64-bit) +echo Building for Windows (amd64)... +set GOOS=windows +set GOARCH=amd64 +go build -o "%OUTPUT_DIR%/python-mcp-servers.exe" python-mcp-server.go +if %errorlevel% neq 0 ( + echo Failed to build for Windows + exit /b 1 +) + +REM Build for Linux (64-bit) +echo Building for Linux (amd64)... +set GOOS=linux +set GOARCH=amd64 +go build -o "%OUTPUT_DIR%/python-mcp-servers" python-mcp-server.go +if %errorlevel% neq 0 ( + echo Failed to build for Linux + exit /b 1 +) + +REM Build for macOS (64-bit Intel) +echo Building for macOS (amd64)... +set GOOS=darwin +set GOARCH=amd64 +go build -o "%OUTPUT_DIR%/python-mcp-servers-mac" python-mcp-server.go +if %errorlevel% neq 0 ( + echo Failed to build for macOS + exit /b 1 +) + +REM Build for macOS (ARM64 - Apple Silicon) +echo Building for macOS (arm64)... +set GOOS=darwin +set GOARCH=arm64 +go build -o "%OUTPUT_DIR%/python-mcp-servers-mac-arm64" python-mcp-server.go +if %errorlevel% neq 0 ( + echo Failed to build for macOS ARM64 + exit /b 1 +) + +echo. +echo ✅ Build completed successfully! +echo. +echo Generated files: +echo - %OUTPUT_DIR%/python-mcp-servers.exe (Windows) +echo - %OUTPUT_DIR%/python-mcp-servers (Linux) +echo - %OUTPUT_DIR%/python-mcp-servers-mac (macOS Intel) +echo - %OUTPUT_DIR%/python-mcp-servers-mac-arm64 (macOS Apple Silicon) +echo. +echo Files are ready for Electron packaging. + +pause \ No newline at end of file diff --git a/clara-mcp/build-all.sh b/clara-mcp/build-all.sh new file mode 100755 index 00000000..9daa0eec --- /dev/null +++ b/clara-mcp/build-all.sh @@ -0,0 +1,114 @@ +#!/bin/bash + +# Clara MCP Server Build Script for All Platforms +# Builds for Windows, Linux, and macOS (Intel + Apple Silicon) + +set -e # Exit on any error + +echo "🚀 Building Clara Python MCP Server for All Platforms..." +echo "========================================================" + +# Set the output directory +OUTPUT_DIR="../electron/mcp" + +# Create output directory if it doesn't exist +mkdir -p "$OUTPUT_DIR" + +# Function to build for a specific platform and architecture +build_for_platform() { + local os=$1 + local arch=$2 + local output_name=$3 + local description=$4 + + echo "📦 Building for $description ($os/$arch)..." + + # Set Go environment variables + export GOOS=$os + export GOARCH=$arch + + # Build the binary + if go build -o "$OUTPUT_DIR/$output_name" python-mcp-server.go; then + echo "✅ Successfully built $description" + + # Get file size + local size=$(ls -lh "$OUTPUT_DIR/$output_name" | awk '{print $5}') + echo " 📁 Output: $OUTPUT_DIR/$output_name ($size)" + else + echo "❌ Failed to build for $description" + exit 1 + fi + + echo "" +} + +# Check if Go is installed +if ! command -v go &> /dev/null; then + echo "❌ Error: Go is not installed or not in PATH" + echo "Please install Go from https://golang.org/dl/" + exit 1 +fi + +# Check Go version +GO_VERSION=$(go version | awk '{print $3}' | sed 's/go//') +echo "🔧 Go version: $GO_VERSION" +echo "" + +# Build for Windows (64-bit) +build_for_platform "windows" "amd64" "python-mcp-server-windows.exe" "Windows (x64)" + +# Build for Linux (64-bit) +build_for_platform "linux" "amd64" "python-mcp-server-linux" "Linux (x64)" + +# Build for macOS Intel (64-bit) +build_for_platform "darwin" "amd64" "python-mcp-server-mac-intel" "macOS Intel (x64)" + +# Build for macOS Apple Silicon (ARM64) +build_for_platform "darwin" "arm64" "python-mcp-server-mac-arm64" "macOS Apple Silicon (ARM64)" + +# Create universal binary for macOS (Intel + ARM64) +echo "🔗 Creating universal binary for macOS (Intel + ARM64)..." +if command -v lipo &> /dev/null; then + if lipo -create \ + "$OUTPUT_DIR/python-mcp-server-mac-intel" \ + "$OUTPUT_DIR/python-mcp-server-mac-arm64" \ + -output "$OUTPUT_DIR/python-mcp-server-mac-universal"; then + + echo "✅ Successfully created universal binary" + + # Get file size + size=$(ls -lh "$OUTPUT_DIR/python-mcp-server-mac-universal" | awk '{print $5}') + echo " 📁 Output: $OUTPUT_DIR/python-mcp-server-mac-universal ($size)" + else + echo "❌ Failed to create universal binary" + exit 1 + fi +else + echo "⚠️ Warning: lipo not found, skipping universal binary creation" +fi + +echo "" +echo "🎉 Build completed successfully!" +echo "========================================================" +echo "" +echo "📋 Generated files:" +echo " • $OUTPUT_DIR/python-mcp-server-windows.exe (Windows x64)" +echo " • $OUTPUT_DIR/python-mcp-server-linux (Linux x64)" +echo " • $OUTPUT_DIR/python-mcp-server-mac-intel (macOS Intel)" +echo " • $OUTPUT_DIR/python-mcp-server-mac-arm64 (macOS Apple Silicon)" +if [ -f "$OUTPUT_DIR/python-mcp-server-mac-universal" ]; then + echo " • $OUTPUT_DIR/python-mcp-server-mac-universal (macOS Universal)" +fi +echo "" +echo "🔍 File details:" +ls -lh "$OUTPUT_DIR"/python-mcp-server-* + +echo "" +echo "✨ All files are ready for Electron packaging!" +echo "" +echo "📝 Usage notes:" +echo " • Windows: Use python-mcp-server-windows.exe" +echo " • Linux: Use python-mcp-server-linux" +echo " • macOS Intel: Use python-mcp-server-mac-intel" +echo " • macOS Apple Silicon: Use python-mcp-server-mac-arm64" +echo " • macOS Universal: Use python-mcp-server-mac-universal (recommended for distribution)" \ No newline at end of file diff --git a/clara-mcp/chromedp-fetcher.go b/clara-mcp/chromedp-fetcher.go new file mode 100644 index 00000000..425a5ba1 --- /dev/null +++ b/clara-mcp/chromedp-fetcher.go @@ -0,0 +1,178 @@ +package main + +import ( + "context" + "fmt" + "log" + "strings" + "sync" + "time" + + "github.com/chromedp/chromedp" +) + +// ChromeDPManager handles lightweight browser automation with chromedp +type ChromeDPManager struct { + ctx context.Context + cancel context.CancelFunc + mu sync.RWMutex +} + +// NewChromeDPManager creates a new chromedp manager with a persistent browser context +func NewChromeDPManager() *ChromeDPManager { + // Create a persistent browser context + opts := append(chromedp.DefaultExecAllocatorOptions[:], + chromedp.Flag("headless", true), + chromedp.Flag("disable-gpu", true), + chromedp.Flag("no-sandbox", true), + chromedp.Flag("disable-dev-shm-usage", true), + chromedp.Flag("disable-web-security", true), + chromedp.UserAgent("Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36"), + ) + + allocCtx, cancel := chromedp.NewExecAllocator(context.Background(), opts...) + ctx, _ := chromedp.NewContext(allocCtx) + + return &ChromeDPManager{ + ctx: ctx, + cancel: cancel, + } +} + +// FetchContent fetches web content using chromedp with JavaScript execution +func (cm *ChromeDPManager) FetchContent(targetURL string, timeout time.Duration) (*WebContent, error) { + cm.mu.Lock() + defer cm.mu.Unlock() + + if timeout == 0 { + timeout = 10 * time.Second + } + + // Create a timeout context for this specific request + ctx, cancel := context.WithTimeout(cm.ctx, timeout) + defer cancel() + + startTime := time.Now() + + var title, htmlContent string + var description string + + // Navigate and wait for the page to load + err := chromedp.Run(ctx, + chromedp.Navigate(targetURL), + // Wait for body to be present + chromedp.WaitVisible("body", chromedp.ByQuery), + // Wait a bit for JavaScript to execute + chromedp.Sleep(2*time.Second), + // Remove script and style tags + chromedp.Evaluate(` + document.querySelectorAll('script, style, noscript, iframe').forEach(el => el.remove()); + `, nil), + // Get the title + chromedp.Title(&title), + // Get meta description + chromedp.Evaluate(` + (() => { + const meta = document.querySelector('meta[name="description"]'); + return meta ? meta.getAttribute('content') : ''; + })() + `, &description), + // Extract text content from semantic elements + chromedp.Evaluate(` + (() => { + const selectors = [ + 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', + 'p', 'span', 'div', 'article', 'section', 'main', + 'li', 'td', 'th', 'blockquote', 'figcaption', + 'address', 'time', 'strong', 'b', 'em', 'i', + 'code', 'pre', 'cite', 'mark', 'a', 'button', 'label' + ]; + + const seen = new Set(); + const content = []; + + selectors.forEach(selector => { + document.querySelectorAll(selector).forEach(el => { + const text = el.textContent.trim(); + if (text && text.length > 2 && !seen.has(text)) { + // Filter out CSS-like content + if (!text.includes('{') && !text.includes('rgba(') && + !text.includes('function(') && !text.startsWith('data-')) { + seen.add(text); + content.push(text); + } + } + }); + }); + + return content.join('\n'); + })() + `, &htmlContent), + ) + + if err != nil { + return nil, fmt.Errorf("chromedp navigation failed: %v", err) + } + + return &WebContent{ + URL: targetURL, + Title: title, + Content: htmlContent, + Description: description, + StatusCode: 200, + IsDynamic: true, + LoadingStrategy: "chromedp", + LoadTime: time.Since(startTime), + }, nil +} + +// Cleanup closes the browser context +func (cm *ChromeDPManager) Cleanup() { + if cm.cancel != nil { + cm.cancel() + } + log.Println("ChromeDPManager: Cleanup completed") +} + +// GetCapabilities returns information about chromedp capabilities +func (cm *ChromeDPManager) GetCapabilities() map[string]interface{} { + return map[string]interface{}{ + "is_available": true, + "javascript_execution": true, + "screenshot_capture": true, + "implementation": "chromedp", + "startup_time": "instant", + } +} + +// isValidContentText filters out CSS, JavaScript, and other non-content text +func isValidContentTextChrome(text string) bool { + if len(text) <= 2 { + return false + } + + lowerText := strings.ToLower(text) + + // Filter out common non-content patterns + invalidPatterns := []string{ + "function(", "var ", "let ", "const ", "return ", + "document.", "window.", "console.", + "data-", "aria-", "http://", "https://", + ".css", ".js", ".png", ".jpg", + "color:", "background:", "border:", + "@media", "@import", + } + + for _, pattern := range invalidPatterns { + if strings.Contains(lowerText, pattern) { + return false + } + } + + // Filter out CSS blocks + if strings.Contains(text, "{") && strings.Contains(text, "}") { + return false + } + + return true +} diff --git a/clara-mcp/go.mod b/clara-mcp/go.mod new file mode 100644 index 00000000..2b4c292c --- /dev/null +++ b/clara-mcp/go.mod @@ -0,0 +1,20 @@ +module clara-mcp + +go 1.24 + +require github.com/playwright-community/playwright-go v0.5200.0 + +require ( + github.com/chromedp/cdproto v0.0.0-20250724212937-08a3db8b4327 // indirect + github.com/chromedp/chromedp v0.14.1 // indirect + github.com/chromedp/sysutil v1.1.0 // indirect + github.com/deckarep/golang-set/v2 v2.7.0 // indirect + github.com/go-jose/go-jose/v3 v3.0.4 // indirect + github.com/go-json-experiment/json v0.0.0-20250725192818-e39067aee2d2 // indirect + github.com/go-stack/stack v1.8.1 // indirect + github.com/gobwas/httphead v0.1.0 // indirect + github.com/gobwas/pool v0.2.1 // indirect + github.com/gobwas/ws v1.4.0 // indirect + github.com/jung-kurt/gofpdf v1.16.2 // indirect + golang.org/x/sys v0.34.0 // indirect +) diff --git a/clara-mcp/go.sum b/clara-mcp/go.sum new file mode 100644 index 00000000..d2c62dc1 --- /dev/null +++ b/clara-mcp/go.sum @@ -0,0 +1,93 @@ +github.com/boombuler/barcode v1.0.0/go.mod h1:paBWMcWSl3LHKBqUq+rly7CNSldXjb2rDl3JlRe0mD8= +github.com/chromedp/cdproto v0.0.0-20250724212937-08a3db8b4327 h1:UQ4AU+BGti3Sy/aLU8KVseYKNALcX9UXY6DfpwQ6J8E= +github.com/chromedp/cdproto v0.0.0-20250724212937-08a3db8b4327/go.mod h1:NItd7aLkcfOA/dcMXvl8p1u+lQqioRMq/SqDp71Pb/k= +github.com/chromedp/chromedp v0.14.1 h1:0uAbnxewy/Q+Bg7oafVePE/6EXEho9hnaC38f+TTENg= +github.com/chromedp/chromedp v0.14.1/go.mod h1:rHzAv60xDE7VNy/MYtTUrYreSc0ujt2O1/C3bzctYBo= +github.com/chromedp/sysutil v1.1.0 h1:PUFNv5EcprjqXZD9nJb9b/c9ibAbxiYo4exNWZyipwM= +github.com/chromedp/sysutil v1.1.0/go.mod h1:WiThHUdltqCNKGc4gaU50XgYjwjYIhKWoHGPTUfWTJ8= +github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= +github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/deckarep/golang-set/v2 v2.7.0 h1:gIloKvD7yH2oip4VLhsv3JyLLFnC0Y2mlusgcvJYW5k= +github.com/deckarep/golang-set/v2 v2.7.0/go.mod h1:VAky9rY/yGXJOLEDv3OMci+7wtDpOF4IN+y82NBOac4= +github.com/go-jose/go-jose/v3 v3.0.4 h1:Wp5HA7bLQcKnf6YYao/4kpRpVMp/yf6+pJKV8WFSaNY= +github.com/go-jose/go-jose/v3 v3.0.4/go.mod h1:5b+7YgP7ZICgJDBdfjZaIt+H/9L9T/YQrVfLAMboGkQ= +github.com/go-json-experiment/json v0.0.0-20250725192818-e39067aee2d2 h1:iizUGZ9pEquQS5jTGkh4AqeeHCMbfbjeb0zMt0aEFzs= +github.com/go-json-experiment/json v0.0.0-20250725192818-e39067aee2d2/go.mod h1:TiCD2a1pcmjd7YnhGH0f/zKNcCD06B029pHhzV23c2M= +github.com/go-stack/stack v1.8.1 h1:ntEHSVwIt7PNXNpgPmVfMrNhLtgjlmnZha2kOpuRiDw= +github.com/go-stack/stack v1.8.1/go.mod h1:dcoOX6HbPZSZptuspn9bctJ+N/CnF5gGygcUP3XYfe4= +github.com/gobwas/httphead v0.1.0 h1:exrUm0f4YX0L7EBwZHuCF4GDp8aJfVeBrlLQrs6NqWU= +github.com/gobwas/httphead v0.1.0/go.mod h1:O/RXo79gxV8G+RqlR/otEwx4Q36zl9rqC5u12GKvMCM= +github.com/gobwas/pool v0.2.1 h1:xfeeEhW7pwmX8nuLVlqbzVc7udMDrwetjEv+TZIz1og= +github.com/gobwas/pool v0.2.1/go.mod h1:q8bcK0KcYlCgd9e7WYLm9LpyS+YeLd8JVDW6WezmKEw= +github.com/gobwas/ws v1.4.0 h1:CTaoG1tojrh4ucGPcoJFiAQUAsEWekEWvLy7GsVNqGs= +github.com/gobwas/ws v1.4.0/go.mod h1:G3gNqMNtPppf5XUz7O4shetPpcZ1VJ7zt18dlUeakrc= +github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= +github.com/jung-kurt/gofpdf v1.0.0/go.mod h1:7Id9E/uU8ce6rXgefFLlgrJj/GYY22cpxn+r32jIOes= +github.com/jung-kurt/gofpdf v1.16.2 h1:jgbatWHfRlPYiK85qgevsZTHviWXKwB1TTiKdz5PtRc= +github.com/jung-kurt/gofpdf v1.16.2/go.mod h1:1hl7y57EsiPAkLbOwzpzqgx1A30nQCk/YmFV8S2vmK0= +github.com/mitchellh/go-ps v1.0.0 h1:i6ampVEEF4wQFF+bkYfwYgY+F/uYJDktmvLPf7qIgjc= +github.com/mitchellh/go-ps v1.0.0/go.mod h1:J4lOc8z8yJs6vUwklHw2XEIiT4z4C40KtWVN3nvg8Pg= +github.com/phpdave11/gofpdi v1.0.7/go.mod h1:vBmVV0Do6hSBHC8uKUQ71JGW+ZGQq74llk/7bXwjDoI= +github.com/phpdave11/gofpdi v1.0.14-0.20211212211723-1f10f9844311 h1:zyWXQ6vu27ETMpYsEMAsisQ+GqJ4e1TPvSNfdOPF0no= +github.com/phpdave11/gofpdi v1.0.14-0.20211212211723-1f10f9844311/go.mod h1:vBmVV0Do6hSBHC8uKUQ71JGW+ZGQq74llk/7bXwjDoI= +github.com/pkg/errors v0.8.1 h1:iURUrRGxPUNPdy5/HRSm+Yj6okJ6UtLINN0Q9M4+h3I= +github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/playwright-community/playwright-go v0.5200.0 h1:z/5LGuX2tBrg3ug1HupMXLjIG93f1d2MWdDsNhkMQ9c= +github.com/playwright-community/playwright-go v0.5200.0/go.mod h1:UnnyQZaqUOO5ywAZu60+N4EiWReUqX1MQBBA3Oofvf8= +github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= +github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/ruudk/golang-pdf417 v0.0.0-20181029194003-1af4ab5afa58/go.mod h1:6lfFZQK844Gfx8o5WFuvpxWRwnSoipWe/p622j1v06w= +github.com/signintech/gopdf v0.20.0 h1:a1rArIMmQCAFzjjCqXPgxynTPkytMccPuGZlUU8Jorw= +github.com/signintech/gopdf v0.20.0/go.mod h1:wrLtZoWaRNrS4hphED0oflFoa6IWkOu6M3nJjm4VbO4= +github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= +github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.8.4 h1:CcVxjf3Q8PM0mHUKJCdn+eZZtm5yQwehR5yeSVQQcUk= +github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= +github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= +golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= +golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= +golang.org/x/crypto v0.19.0/go.mod h1:Iy9bg/ha4yyC70EfRS8jz+B6ybOBKMaSxLj6P6oBDfU= +golang.org/x/image v0.0.0-20190910094157-69e4b8554b2a/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= +golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= +golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= +golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= +golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= +golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= +golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg= +golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.17.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= +golang.org/x/sys v0.34.0 h1:H5Y5sJ2L2JRdyv7ROF1he/lPdvFsd0mJHFw2ThKHxLA= +golang.org/x/sys v0.34.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k= +golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= +golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= +golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= +golang.org/x/term v0.8.0/go.mod h1:xPskH00ivmX89bAKVGSKKtLOWNx2+17Eiy94tnKShWo= +golang.org/x/term v0.17.0/go.mod h1:lLRBjIVuehSbZlaOtGMbcMncT+aqLLLmKrsjNrUguwk= +golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= +golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= +golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= +golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= +golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= +golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU= +golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= +golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU= +golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= +gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= diff --git a/clara-mcp/playwright-manager.go b/clara-mcp/playwright-manager.go new file mode 100644 index 00000000..73c92751 --- /dev/null +++ b/clara-mcp/playwright-manager.go @@ -0,0 +1,527 @@ +package main + +import ( + "fmt" + "log" + "os" + "path/filepath" + "runtime" + "strings" + "sync" + "time" + + "github.com/playwright-community/playwright-go" +) + +// PlaywrightManager handles browser automation with progressive enhancement and auto-download +type PlaywrightManager struct { + pw *playwright.Playwright + browser playwright.Browser + isAvailable bool + isDownloading bool + downloadError error + mu sync.RWMutex + userDataDir string + hasPlaywright bool // Track if Playwright library is available +} + +// PlaywrightConfig holds configuration for Playwright +type PlaywrightConfig struct { + Headless bool + Timeout time.Duration + UserAgent string + ViewportWidth int + ViewportHeight int +} + +// NewPlaywrightManager creates a new Playwright manager with progressive enhancement +func NewPlaywrightManager() *PlaywrightManager { + pm := &PlaywrightManager{ + isAvailable: false, + isDownloading: false, + hasPlaywright: true, // Now we have the library available + userDataDir: getDefaultUserDataDir(), + } + + // Start auto-download in background during initialization + go pm.StartDownload() + + return pm +} + +// getDefaultUserDataDir returns a platform-specific directory for browser data +func getDefaultUserDataDir() string { + homeDir, err := os.UserHomeDir() + if err != nil { + return "./playwright-data" + } + + switch runtime.GOOS { + case "windows": + return filepath.Join(homeDir, "AppData", "Local", "ClaraVerse", "playwright") + case "darwin": + return filepath.Join(homeDir, "Library", "Application Support", "ClaraVerse", "playwright") + default: + return filepath.Join(homeDir, ".local", "share", "ClaraVerse", "playwright") + } +} + +// IsAvailable returns whether Playwright is available for use +func (pm *PlaywrightManager) IsAvailable() bool { + pm.mu.RLock() + defer pm.mu.RUnlock() + return pm.isAvailable && pm.hasPlaywright +} + +// IsDownloading returns whether Playwright browsers are currently being downloaded +func (pm *PlaywrightManager) IsDownloading() bool { + pm.mu.RLock() + defer pm.mu.RUnlock() + return pm.isDownloading +} + +// GetDownloadError returns any error from the last download attempt +func (pm *PlaywrightManager) GetDownloadError() error { + pm.mu.RLock() + defer pm.mu.RUnlock() + return pm.downloadError +} + +// EnsureAvailable attempts to make Playwright available - forces download if needed +func (pm *PlaywrightManager) EnsureAvailable() error { + pm.mu.Lock() + defer pm.mu.Unlock() + + if pm.isAvailable && pm.hasPlaywright { + return nil + } + + if !pm.hasPlaywright { + pm.downloadError = fmt.Errorf("playwright library not available - using fallback mode") + return pm.downloadError + } + + // Force download start if not already downloading + if !pm.isDownloading { + pm.mu.Unlock() // Unlock before calling StartDownload + go pm.StartDownload() + pm.mu.Lock() // Re-lock for defer + log.Printf("EnsureAvailable: Triggered Playwright download") + } + + return pm.downloadError +} + +// StartDownload begins downloading Playwright browsers in the background +func (pm *PlaywrightManager) StartDownload() { + pm.mu.Lock() + if pm.isDownloading || pm.isAvailable { + pm.mu.Unlock() + return + } + + if !pm.hasPlaywright { + pm.downloadError = fmt.Errorf("playwright library not available") + pm.mu.Unlock() + return + } + + pm.isDownloading = true + pm.downloadError = nil + pm.mu.Unlock() + + defer func() { + pm.mu.Lock() + pm.isDownloading = false + pm.mu.Unlock() + }() + + log.Println("PlaywrightManager: Starting browser download...") + + // Create user data directory + if err := os.MkdirAll(pm.userDataDir, 0755); err != nil { + pm.mu.Lock() + pm.downloadError = fmt.Errorf("failed to create user data directory: %v", err) + pm.mu.Unlock() + log.Printf("PlaywrightManager: Failed to create data dir: %v", err) + return + } + + // Install Playwright browsers (this will skip if already installed) + err := playwright.Install(&playwright.RunOptions{ + Browsers: []string{"chromium"}, // Only install Chromium to save space + Verbose: false, // Reduce verbosity + }) + + if err != nil { + pm.mu.Lock() + pm.downloadError = fmt.Errorf("failed to install browsers: %v", err) + pm.mu.Unlock() + log.Printf("PlaywrightManager: Browser installation failed: %v", err) + return + } + + // Initialize Playwright + pw, err := playwright.Run() + if err != nil { + pm.mu.Lock() + pm.downloadError = fmt.Errorf("failed to start playwright: %v", err) + pm.mu.Unlock() + log.Printf("PlaywrightManager: Failed to start Playwright: %v", err) + return + } + + pm.mu.Lock() + pm.pw = pw + pm.isAvailable = true + pm.downloadError = nil + pm.mu.Unlock() + + log.Println("PlaywrightManager: Browser download and initialization completed successfully!") +} + +// FetchContent attempts to fetch web content using Playwright +func (pm *PlaywrightManager) FetchContent(targetURL string, config interface{}) (*WebContent, error) { + pm.mu.RLock() + defer pm.mu.RUnlock() + + if !pm.isAvailable || !pm.hasPlaywright || pm.pw == nil { + return nil, fmt.Errorf("playwright not available") + } + + // Use default config if none provided + playwrightConfig := PlaywrightConfig{ + Headless: true, + Timeout: 30 * time.Second, + UserAgent: "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36", + ViewportWidth: 1920, + ViewportHeight: 1080, + } + + // Override with provided config if available + if config != nil { + if cfg, ok := config.(PlaywrightConfig); ok { + playwrightConfig = cfg + } + } + + // Launch browser if not already done + if pm.browser == nil { + browser, err := pm.pw.Chromium.Launch(playwright.BrowserTypeLaunchOptions{ + Headless: playwright.Bool(playwrightConfig.Headless), + }) + if err != nil { + return nil, fmt.Errorf("failed to launch browser: %v", err) + } + pm.browser = browser + } + + // Create new context for this request + context, err := pm.browser.NewContext(playwright.BrowserNewContextOptions{ + UserAgent: playwright.String(playwrightConfig.UserAgent), + Viewport: &playwright.Size{ + Width: playwrightConfig.ViewportWidth, + Height: playwrightConfig.ViewportHeight, + }, + }) + if err != nil { + return nil, fmt.Errorf("failed to create context: %v", err) + } + defer context.Close() + + // Create new page + page, err := context.NewPage() + if err != nil { + return nil, fmt.Errorf("failed to create page: %v", err) + } + + // Navigate to the URL and wait for network to be idle + startTime := time.Now() + response, err := page.Goto(targetURL, playwright.PageGotoOptions{ + WaitUntil: playwright.WaitUntilStateNetworkidle, + Timeout: playwright.Float(float64(playwrightConfig.Timeout.Milliseconds())), + }) + if err != nil { + return nil, fmt.Errorf("failed to navigate: %v", err) + } + + // Wait a bit more for any dynamic content to load + page.WaitForTimeout(2000) // 2 seconds + + // Extract semantic text content from specific HTML elements + title, _ := page.Title() + + // First, remove script and style elements to avoid extracting their content + page.Evaluate("() => { document.querySelectorAll('script, style, noscript').forEach(el => el.remove()); }") + + // Define semantic content selectors for text extraction + contentSelectors := []string{ + "h1, h2, h3, h4, h5, h6", // Headings + "p", // Paragraphs + "span", // Spans + "div", // Divs (common content containers) + "article", // Article content + "section", // Section content + "main", // Main content area + "li", // List items + "td, th", // Table cells + "blockquote", // Quotes + "figcaption", // Figure captions + "address", // Address elements + "time", // Time elements + "strong, b, em, i", // Emphasized text + "code, pre", // Code blocks + "cite", // Citations + "mark", // Highlighted text + "small", // Small text + "sub, sup", // Subscript/superscript + "ins, del", // Inserted/deleted text + "kbd", // Keyboard input + "samp", // Sample output + "var", // Variables + "abbr", // Abbreviations + "dfn", // Definitions + "q", // Inline quotes + "s", // Strikethrough + "u", // Underlined text + "data", // Data values + "output", // Output elements + "a", // Links (text content only) + "button", // Button text + "label", // Form labels + "legend", // Fieldset legends + "caption", // Table captions + "summary", // Details summary + "dt, dd", // Definition terms and descriptions + } + + var contentParts []string + + // Extract text from each type of semantic element + for _, selector := range contentSelectors { + elements, err := page.QuerySelectorAll(selector) + if err != nil { + continue + } + + for _, element := range elements { + // Get text content and clean it up + if textContent, err := element.TextContent(); err == nil && textContent != "" { + trimmed := strings.TrimSpace(textContent) + // Filter out CSS-like content and other noise + if isValidContentText(trimmed) { + contentParts = append(contentParts, trimmed) + } + } + } + } + + // Join all content with newlines and remove duplicates + content := strings.Join(removeDuplicateStrings(contentParts), "\n") + + // Try to get meta description + description := "" + if metaDesc, err := page.GetAttribute("meta[name='description']", "content"); err == nil && metaDesc != "" { + description = metaDesc + } + + statusCode := 200 + if response != nil { + statusCode = response.Status() + } + + return &WebContent{ + URL: targetURL, + Title: title, + Content: content, + Description: description, + StatusCode: statusCode, + IsDynamic: true, + LoadingStrategy: "playwright", + LoadTime: time.Since(startTime), + }, nil +} + +// removeDuplicateStrings removes duplicate strings from a slice while preserving order +func removeDuplicateStrings(slice []string) []string { + seen := make(map[string]bool) + result := make([]string, 0, len(slice)) + + for _, str := range slice { + if !seen[str] { + seen[str] = true + result = append(result, str) + } + } + + return result +} + +// isValidContentText filters out CSS, JavaScript, and other non-content text +func isValidContentText(text string) bool { + // Minimum length check + if len(text) <= 2 { + return false + } + + // Check for CSS-like patterns + cssPatterns := []string{ + "color:", "background:", "border:", "width:", "height:", "margin:", "padding:", + "font-", "text-", "display:", "position:", "top:", "left:", "right:", "bottom:", + "z-index:", "opacity:", "transform:", "transition:", "animation:", + "@media", "@import", "@keyframes", "rgba(", "rgb(", "#", "px", "rem", "em", "%", + "!important", "hover:", "active:", "focus:", "before:", "after:", + "flex", "grid", "absolute", "relative", "fixed", "sticky", + } + + lowerText := strings.ToLower(text) + + // Check if text contains too many CSS indicators + cssCount := 0 + for _, pattern := range cssPatterns { + if strings.Contains(lowerText, pattern) { + cssCount++ + } + } + + // If more than 2 CSS patterns are found, likely CSS content + if cssCount > 2 { + return false + } + + // Check for long strings of CSS-like content (selectors, properties) + if strings.Contains(text, "{") && strings.Contains(text, "}") { + return false + } + + // Check for CSS selector patterns + if strings.Contains(text, ".") && strings.Contains(text, "-") && len(strings.Fields(text)) <= 3 { + return false + } + + // Filter out JavaScript-like content + jsPatterns := []string{ + "function(", "var ", "let ", "const ", "return ", "if(", "for(", "while(", + "document.", "window.", "console.", "alert(", "parseInt(", "parseFloat(", + "getElementById", "querySelector", "addEventListener", + } + + for _, pattern := range jsPatterns { + if strings.Contains(lowerText, pattern) { + return false + } + } + + // Filter out data attributes and technical strings + if strings.HasPrefix(text, "data-") || strings.HasPrefix(text, "aria-") { + return false + } + + // Filter out URLs and file paths + if strings.Contains(text, "http://") || strings.Contains(text, "https://") || + strings.Contains(text, ".com") || strings.Contains(text, ".css") || + strings.Contains(text, ".js") || strings.Contains(text, ".png") || + strings.Contains(text, ".jpg") || strings.Contains(text, ".gif") { + return false + } + + // Accept if it looks like readable content + return true +} + +// GetCapabilities returns information about Playwright capabilities +func (pm *PlaywrightManager) GetCapabilities() map[string]interface{} { + pm.mu.RLock() + defer pm.mu.RUnlock() + + return map[string]interface{}{ + "is_available": pm.isAvailable && pm.hasPlaywright, + "is_downloading": pm.isDownloading, + "has_playwright_lib": pm.hasPlaywright, + "javascript_execution": pm.isAvailable && pm.hasPlaywright, + "network_interception": pm.isAvailable && pm.hasPlaywright, + "screenshot_capture": pm.isAvailable && pm.hasPlaywright, + "download_status": pm.getDownloadStatus(), + "estimated_size": "~50MB (Chromium only)", + "download_error": pm.downloadError, + } +} + +func (pm *PlaywrightManager) getDownloadStatus() string { + if pm.isAvailable && pm.hasPlaywright { + return "installed" + } + if pm.isDownloading { + return "downloading" + } + if !pm.hasPlaywright { + return "library_not_available" + } + return "not_installed" +} + +// GetInstallationStatus returns detailed installation information +func (pm *PlaywrightManager) GetInstallationStatus() map[string]interface{} { + return map[string]interface{}{ + "playwright_library": pm.hasPlaywright, + "browsers_installed": pm.isAvailable, + "currently_downloading": pm.isDownloading, + "last_error": func() string { + if pm.downloadError != nil { + return pm.downloadError.Error() + } + return "" + }(), + "capabilities": pm.GetCapabilities(), + } +} + +// Cleanup performs cleanup operations +func (pm *PlaywrightManager) Cleanup() { + pm.mu.Lock() + defer pm.mu.Unlock() + + if pm.browser != nil { + pm.browser.Close() + pm.browser = nil + } + + if pm.pw != nil { + pm.pw.Stop() + pm.pw = nil + } + + log.Println("PlaywrightManager: Cleanup completed") +} + +// GetUserDataDir returns the user data directory for browser sessions +func (pm *PlaywrightManager) GetUserDataDir() string { + pm.mu.RLock() + defer pm.mu.RUnlock() + return pm.userDataDir +} + +// SetUserDataDir sets the user data directory for browser sessions +func (pm *PlaywrightManager) SetUserDataDir(dir string) { + pm.mu.Lock() + defer pm.mu.Unlock() + pm.userDataDir = dir +} + +// EnablePlaywright enables Playwright functionality when the library becomes available +func (pm *PlaywrightManager) EnablePlaywright() { + pm.mu.Lock() + defer pm.mu.Unlock() + pm.hasPlaywright = true + log.Println("PlaywrightManager: Playwright library enabled") +} + +// Version returns version information +func (pm *PlaywrightManager) Version() map[string]interface{} { + return map[string]interface{}{ + "manager_version": "1.0.0-full", + "playwright_version": "v0.5200.0", + "implementation": "full", + "ready_for_upgrade": false, + } +} diff --git a/clara-mcp/python-mcp-server.go b/clara-mcp/python-mcp-server.go new file mode 100644 index 00000000..6c59d9b7 --- /dev/null +++ b/clara-mcp/python-mcp-server.go @@ -0,0 +1,2830 @@ +package main + +import ( + "bufio" + "encoding/json" + "fmt" + "io" + "io/ioutil" + "log" + "net/http" + "net/url" + "os" + "os/exec" + "path/filepath" + "regexp" + "runtime" + "strings" + "sync" + "time" + + "github.com/jung-kurt/gofpdf" +) + +// SearXNG and Web Content constants +const ( + SearXNGImage = "searxng/searxng:latest" + ContainerName = "clara-searxng" + SearXNGPort = "8080" + SearXNGURL = "http://localhost:8080" + HealthCheckPath = "/healthz" + SearchPath = "/search" + ConfigDirName = "searxng-config" + ConfigEnvVar = "CLARA_SEARXNG_CONFIG_DIR" +) + +// Web content and search types +type SearchResult struct { + URL string `json:"url"` + Title string `json:"title"` + Content string `json:"content"` + PublishedAt string `json:"publishedDate,omitempty"` + Engine string `json:"engine"` +} + +type SearchResponse struct { + Query string `json:"query"` + NumberOfResults int `json:"number_of_results"` + Results []SearchResult `json:"results"` + Infoboxes []interface{} `json:"infoboxes"` + Suggestions []string `json:"suggestions"` + AnswerBox interface{} `json:"answer"` +} + +type WebContent struct { + URL string `json:"url"` + Title string `json:"title"` + Content string `json:"content"` + Description string `json:"description"` + StatusCode int `json:"status_code"` + Error string `json:"error,omitempty"` + // Enhanced fields for smart dynamic detection + IsDynamic bool `json:"is_dynamic"` + LoadingStrategy string `json:"loading_strategy"` // "static", "api_simulation", "fallback" + APIEndpoints []string `json:"api_endpoints,omitempty"` + JavaScriptErrors []string `json:"js_errors,omitempty"` + LoadTime time.Duration `json:"load_time"` +} + +type SearXNGManager struct { + containerID string + isRunning bool + configDir string +} + +type WebContentFetcher struct { + client *http.Client + jsEngine *JSEngine // Self-contained JavaScript engine + smartMode bool // Enable smart dynamic content detection + playwrightManager *PlaywrightManager // Progressive Playwright integration + chromedpManager *ChromeDPManager // Fast chromedp integration +} + +// Self-contained JavaScript engine for basic DOM simulation +type JSEngine struct { + // Will implement lightweight JS execution for basic dynamic content + apiDetector *regexp.Regexp + domSimulator *DOMSimulator +} + +type DOMSimulator struct { + // Lightweight DOM-like structure for content simulation + virtualDOM map[string]interface{} +} + +// Core types +type MCPRequest struct { + JSONRPC string `json:"jsonrpc"` + Method string `json:"method"` + ID interface{} `json:"id"` + Params json.RawMessage `json:"params"` +} + +type MCPResponse struct { + JSONRPC string `json:"jsonrpc"` + ID interface{} `json:"id"` + Result interface{} `json:"result,omitempty"` + Error *MCPError `json:"error,omitempty"` +} + +type MCPError struct { + Code int `json:"code"` + Message string `json:"message"` +} + +type Tool struct { + Name string `json:"name"` + Description string `json:"description"` + InputSchema map[string]interface{} `json:"inputSchema"` +} + +// PythonMCPServer with virtual environment support +type PythonMCPServer struct { + systemPython string // System Python for creating venv + venvPath string // Path to virtual environment + pythonPath string // Python executable in venv + pipPath string // Pip executable in venv + workspaceDir string // Workspace directory + mu sync.RWMutex +} + +// NewPythonMCPServer creates server with dedicated workspace and venv +func NewPythonMCPServer() *PythonMCPServer { + // Create workspace directory with absolute path + // Priority order: CLARA_MCP_WORKSPACE env var -> user home -> /tmp + // NEVER use current working directory as it may be read-only in packaged apps + + var workspace string + var workspaceSource string + + // 1. Try environment variable (set by Electron to writable location) + workspace = os.Getenv("CLARA_MCP_WORKSPACE") + if workspace != "" { + workspaceSource = "CLARA_MCP_WORKSPACE environment variable" + if err := os.MkdirAll(workspace, 0755); err == nil { + log.Printf("Using workspace from %s: %s", workspaceSource, workspace) + } else { + log.Printf("Warning: Failed to create workspace from %s (%s): %v", workspaceSource, workspace, err) + workspace = "" // Clear to try next option + } + } + + // 2. Fallback to user home directory + if workspace == "" { + if homeDir, err := os.UserHomeDir(); err == nil { + workspace = filepath.Join(homeDir, ".clara", "mcp_workspace") + workspaceSource = "user home directory" + if err := os.MkdirAll(workspace, 0755); err == nil { + log.Printf("Using workspace from %s: %s", workspaceSource, workspace) + } else { + log.Printf("Warning: Failed to create workspace in %s (%s): %v", workspaceSource, workspace, err) + workspace = "" // Clear to try next option + } + } else { + log.Printf("Warning: Could not determine user home directory: %v", err) + } + } + + // 3. Last resort: use /tmp (always writable on Unix-like systems) + if workspace == "" { + workspace = filepath.Join(os.TempDir(), "clara_mcp_workspace") + workspaceSource = "temporary directory" + if err := os.MkdirAll(workspace, 0755); err != nil { + // This is critical - if we can't even create in /tmp, something is very wrong + log.Printf("ERROR: Failed to create workspace in %s (%s): %v", workspaceSource, workspace, err) + panic(fmt.Sprintf("Cannot create workspace directory in any location (tried env var, home, and temp): %v", err)) + } + log.Printf("Using workspace from %s: %s", workspaceSource, workspace) + } + + server := &PythonMCPServer{ + workspaceDir: workspace, + venvPath: filepath.Join(workspace, ".venv"), + } + + // Find system Python first + server.systemPython = server.findSystemPython() + log.Printf("Found system Python: %s", server.systemPython) + + // Initialize virtual environment + if err := server.initVirtualEnv(); err != nil { + log.Printf("WARNING: Failed to create virtual environment: %v", err) + log.Printf("Falling back to system Python") + server.pythonPath = server.systemPython + server.pipPath = server.systemPython + } + + log.Printf("Python MCP Server started") + log.Printf("System Python: %s", server.systemPython) + log.Printf("Virtual env: %s", server.venvPath) + log.Printf("Active Python: %s", server.pythonPath) + log.Printf("Workspace: %s", server.workspaceDir) + + // Verify workspace is writable + if err := os.MkdirAll(filepath.Join(server.workspaceDir, "test"), 0755); err != nil { + log.Printf("WARNING: Workspace directory is not writable: %v", err) + } else { + os.RemoveAll(filepath.Join(server.workspaceDir, "test")) + log.Printf("Workspace directory is writable") + } + + // Create README file + server.createReadme() + + return server +} + +// findSystemPython finds the system Python 3 +func (s *PythonMCPServer) findSystemPython() string { + // Try common commands + for _, cmd := range []string{"python3", "python", "py"} { + if path, err := exec.LookPath(cmd); err == nil { + // Verify it's Python 3 + out, err := exec.Command(path, "--version").Output() + if err == nil && strings.Contains(string(out), "Python 3") { + return path + } + } + } + return "python" // fallback +} + +// initVirtualEnv creates and activates a virtual environment +func (s *PythonMCPServer) initVirtualEnv() error { + // Set paths based on OS + if runtime.GOOS == "windows" { + s.pythonPath = filepath.Join(s.venvPath, "Scripts", "python.exe") + s.pipPath = filepath.Join(s.venvPath, "Scripts", "pip.exe") + } else { + s.pythonPath = filepath.Join(s.venvPath, "bin", "python") + s.pipPath = filepath.Join(s.venvPath, "bin", "pip") + } + + // Check if venv already exists + if _, err := os.Stat(s.pythonPath); err == nil { + log.Printf("Virtual environment already exists") + return nil + } + + // Create virtual environment + log.Printf("Creating virtual environment...") + cmd := exec.Command(s.systemPython, "-m", "venv", s.venvPath) + cmd.Dir = s.workspaceDir + output, err := cmd.CombinedOutput() + if err != nil { + return fmt.Errorf("failed to create venv: %v\nOutput: %s", err, output) + } + + // Verify venv was created + if _, err := os.Stat(s.pythonPath); err != nil { + return fmt.Errorf("venv created but Python not found at %s", s.pythonPath) + } + + // Upgrade pip in the venv + log.Printf("Upgrading pip in virtual environment...") + upgradeCmd := exec.Command(s.pythonPath, "-m", "pip", "install", "--upgrade", "pip") + upgradeCmd.Dir = s.workspaceDir + if output, err := upgradeCmd.CombinedOutput(); err != nil { + log.Printf("Warning: Failed to upgrade pip: %v\nOutput: %s", err, output) + } + + log.Printf("Virtual environment created successfully") + return nil +} + +// createReadme creates a comprehensive README file in workspace +func (s *PythonMCPServer) createReadme() { + readmePath := filepath.Join(s.workspaceDir, "README.txt") + if _, err := os.Stat(readmePath); os.IsNotExist(err) { + shellInfo := "Shell: Unix/Linux (/bin/sh)" + if runtime.GOOS == "windows" { + shellInfo = "Shell: Windows PowerShell" + } + + readme := `MCP Workspace Directory - Python Execution Environment +====================================================== + +Welcome to your isolated MCP (Model Context Protocol) workspace! + +OVERVIEW: +This workspace provides a completely isolated Python environment where you can: +- Execute Python code safely without affecting your system +- Install packages that won't interfere with system Python +- Save and load files in a dedicated workspace +- Run shell commands with automatic Python/pip routing + +WORKSPACE FEATURES: +✓ Isolated Python Virtual Environment (.venv/) +✓ Clean workspace for file operations +✓ Cross-platform shell command support +✓ Automatic dependency management +✓ Safe package installation + +AVAILABLE TOOLS: + +1. py(code="...") + - Execute Python code in isolated environment + - Auto-prints last line expressions + - Supports multi-line code, imports, functions + - Examples: + py(code="import math; math.sqrt(16)") + py(code="[x**2 for x in range(5)]") + py(code="def greet(name): return f'Hello {name}!'") + +2. ` + shellInfo + ` + - powershell(cmd="...") on Windows / sh(cmd="...") on Unix + - Execute system commands + - Auto-routes python/pip to virtual environment + - Examples: + powershell(cmd="Get-Process python") + sh(cmd="ps aux | grep python") + +3. pip(pkg="...") + - Install Python packages safely + - Only affects this workspace environment + - Examples: + pip(pkg="requests") + pip(pkg="numpy pandas matplotlib") + pip(pkg="beautifulsoup4==4.9.3") + +4. save(name="...", text="...") + - Save content to workspace files + - Persistent across MCP session + - Examples: + save(name="script.py", text="print('Hello World')") + save(name="data.json", text='{"key": "value"}') + +5. load(name="...") + - Read file content from workspace + - Access previously saved files + - Examples: + load(name="script.py") + load(name="data.json") + +6. ls() + - List all workspace files and directories + - Shows file sizes and types + - Excludes .venv for clarity + +7. open() + - Open workspace in system file manager + - Direct access to workspace folder + - Platform-specific file manager + +GETTING STARTED: +1. Check Python version: py(code="import sys; print(sys.version)") +2. Install a package: pip(pkg="requests") +3. Test the package: py(code="import requests; print('Requests installed!')") +4. Save a script: save(name="test.py", text="print('Hello from saved file')") +5. List files: ls() +6. Load and run: py(code=load(name="test.py")) + +WORKSPACE STRUCTURE: +├── README.txt (this file) +├── .venv/ (Python virtual environment - hidden from ls()) +├── your_files.py (files you save) +├── data_files.json (data you create) +└── any_other_files (content you work with) + +TIPS FOR AI MODELS: +- Use py() for Python calculations, data processing, API calls +- Use pip() to install libraries before using them in py() +- Use save()/load() to persist code and data between operations +- Use ls() to see what files are available +- All operations are isolated and safe to experiment with +- Files persist within the same MCP session + +VIRTUAL ENVIRONMENT DETAILS: +- Location: .venv/ +- Python: Isolated Python 3.x installation +- Packages: Separated from system Python +- Activation: Automatic for all py() and python commands + +This workspace is your sandbox - experiment freely! +` + ioutil.WriteFile(readmePath, []byte(readme), 0644) + } +} + +// getTools returns detailed tool definitions with comprehensive descriptions +func (s *PythonMCPServer) getTools() []Tool { + // Dynamic shell description based on OS + shellName := "sh" + shellDesc := "Execute shell commands in Unix/Linux environment. Runs commands using /bin/sh with full access to system utilities, file operations, and process management. Automatically routes 'python' and 'pip' commands to the isolated virtual environment." + cmdDesc := "Shell command to execute (e.g., 'ls -la', 'grep pattern file.txt', 'curl https://api.example.com')" + + if runtime.GOOS == "windows" { + shellName = "powershell" + shellDesc = "Execute PowerShell commands in Windows environment. Runs commands using PowerShell with full access to Windows utilities, file system operations, registry access, and .NET framework. Automatically routes 'python' and 'pip' commands to the isolated virtual environment. Supports both PowerShell cmdlets and traditional Windows commands." + cmdDesc = "PowerShell command to execute (e.g., 'Get-ChildItem', 'Test-Path C:\\file.txt', 'Invoke-WebRequest https://api.example.com')" + } + + return []Tool{ + { + Name: "py", + Description: "Execute Python code in an isolated virtual environment. Runs Python 3.x code with automatic output handling - expressions on the last line are automatically printed for convenience. All code executes in a dedicated workspace directory with an isolated virtual environment, so system Python and packages remain untouched. Perfect for data analysis, calculations, file processing, API calls, and any Python scripting needs. Supports multi-line code blocks, imports, function definitions, and complex operations.", + InputSchema: map[string]interface{}{ + "type": "object", + "properties": map[string]interface{}{ + "code": map[string]interface{}{ + "type": "string", + "description": "Python code to execute. Can be single expressions (e.g., '2+2'), multi-line scripts, import statements, function definitions, or complex programs. Last line expressions are automatically printed. Examples: 'import requests; requests.get(\"https://api.github.com\").json()', 'def factorial(n): return 1 if n <= 1 else n * factorial(n-1); factorial(5)', '[x**2 for x in range(10)]'", + }, + }, + "required": []string{"code"}, + }, + }, + { + Name: shellName, + Description: shellDesc, + InputSchema: map[string]interface{}{ + "type": "object", + "properties": map[string]interface{}{ + "cmd": map[string]interface{}{ + "type": "string", + "description": cmdDesc, + }, + }, + "required": []string{"cmd"}, + }, + }, + { + Name: "pip", + Description: "Install Python packages into the isolated virtual environment. Safely installs Python packages using pip without affecting the system Python installation. All packages are installed only in the dedicated virtual environment created for this MCP session. Supports installing from PyPI, Git repositories, local files, and specific versions. Use this to add any Python libraries you need for your code execution.", + InputSchema: map[string]interface{}{ + "type": "object", + "properties": map[string]interface{}{ + "pkg": map[string]interface{}{ + "type": "string", + "description": "Package name or specification to install. Examples: 'requests' (latest version), 'numpy==1.21.0' (specific version), 'git+https://github.com/user/repo.git' (from Git), 'package>=1.0,<2.0' (version range), 'requests beautifulsoup4 pandas' (multiple packages)", + }, + }, + "required": []string{"pkg"}, + }, + }, + { + Name: "save", + Description: "Save text content to a file in the MCP workspace directory. Creates or overwrites files with the specified content. All files are saved to the isolated workspace directory and can be accessed later with the 'load' tool or referenced in Python/shell commands. Perfect for saving code, data, configuration files, logs, or any text-based content. Files persist across tool calls within the same MCP session.", + InputSchema: map[string]interface{}{ + "type": "object", + "properties": map[string]interface{}{ + "name": map[string]interface{}{ + "type": "string", + "description": "Filename to save (e.g., 'script.py', 'data.json', 'config.txt', 'analysis.csv'). Extension determines file type. File will be saved in the MCP workspace directory.", + }, + "text": map[string]interface{}{ + "type": "string", + "description": "Complete file content to save. Can be Python code, JSON data, CSV content, configuration text, or any text-based format. Use proper formatting and newlines as needed.", + }, + "auto_open": map[string]interface{}{ + "type": "boolean", + "description": "Whether to automatically open the file after saving (default: true). Set to false when uploading files to workspace to avoid unnecessary file opening.", + "default": true, + }, + }, + "required": []string{"name", "text"}, + }, + }, + { + Name: "load", + Description: "Read and return the complete content of a file from the MCP workspace directory. Retrieves text content from files previously saved with the 'save' tool or placed in the workspace directory. Returns the entire file content as text, which can then be processed, analyzed, or modified. Use this to access saved scripts, data files, configuration files, or any text-based content.", + InputSchema: map[string]interface{}{ + "type": "object", + "properties": map[string]interface{}{ + "name": map[string]interface{}{ + "type": "string", + "description": "Filename to read from the workspace directory (e.g., 'script.py', 'data.json', 'results.txt'). Must be an existing file in the MCP workspace.", + }, + }, + "required": []string{"name"}, + }, + }, + { + Name: "ls", + Description: "List all files and directories in the MCP workspace directory. Shows file names, sizes (in bytes, KB, or MB), and indicates directories with [DIR] prefix. Excludes the .venv virtual environment directory from the listing for clarity. Use this to see what files are available for loading, understand the workspace structure, or verify that files were saved correctly. Helps you navigate and manage workspace contents.", + InputSchema: map[string]interface{}{ + "type": "object", + "properties": map[string]interface{}{}, + }, + }, + { + Name: "open", + Description: "Open the MCP workspace directory in the system file manager for direct access. Launches the default file manager (Windows Explorer on Windows, Finder on macOS, or available file manager on Linux) showing the workspace folder. This allows you to manually inspect files, add external files to the workspace, or perform file operations outside the MCP tools. The workspace contains all saved files and the Python virtual environment.", + InputSchema: map[string]interface{}{ + "type": "object", + "properties": map[string]interface{}{}, + }, + }, + { + Name: "search", + Description: "Search the web using SearXNG private search engine. Performs privacy-focused web searches using a local SearXNG Docker container. Automatically starts the SearXNG service if needed, searches for the specified query, and returns structured results with titles, URLs, content snippets, and source engines. Perfect for research, fact-checking, finding documentation, or gathering information while maintaining privacy. Results include suggestions and can be filtered by various search engines.", + InputSchema: map[string]interface{}{ + "type": "object", + "properties": map[string]interface{}{ + "query": map[string]interface{}{ + "type": "string", + "description": "Search query to execute (e.g., 'golang web scraping', 'machine learning tutorials', 'docker best practices'). Use natural language or specific keywords.", + }, + "num_results": map[string]interface{}{ + "type": "integer", + "description": "Maximum number of search results to return (default: 10, max: 20). More results provide broader coverage but take longer to process.", + "default": 10, + "minimum": 1, + "maximum": 20, + }, + }, + "required": []string{"query"}, + }, + }, + { + Name: "fetch_content", + Description: "Fetch and extract content from web pages using Playwright browser automation as the default standard. Always uses real browser rendering with JavaScript execution for accurate dynamic content extraction. Automatically downloads and installs Playwright browsers (~50MB) on first use.", + InputSchema: map[string]interface{}{ + "type": "object", + "properties": map[string]interface{}{ + "url": map[string]interface{}{ + "type": "string", + "description": "URL to fetch content from (e.g., 'https://example.com/article', 'https://docs.python.org/3/'). All sites including SPAs, React apps, and dynamic content are fully supported through browser automation.", + }, + }, + "required": []string{"url"}, + }, + }, + { + Name: "read_document", + Description: "Read and extract text content from various document formats including PDF, DOCX, XLSX, CSV, PPT, PPTX, TXT, RTF, and more. Supports both local file paths and remote URLs. Automatically detects document type and uses appropriate extraction methods to provide structured, readable text content. Perfect for document analysis, content extraction, and text processing from office files.", + InputSchema: map[string]interface{}{ + "type": "object", + "properties": map[string]interface{}{ + "path": map[string]interface{}{ + "type": "string", + "description": "File path to the document or URL to a remote document. Supports local paths (e.g., 'C:\\Documents\\file.pdf', '/home/user/document.docx') and remote URLs (e.g., 'https://example.com/document.pdf'). Automatically detects format from file extension.", + }, + "extract_metadata": map[string]interface{}{ + "type": "boolean", + "description": "Whether to extract document metadata (author, creation date, etc.) along with content. Default: false", + "default": false, + }, + "page_range": map[string]interface{}{ + "type": "string", + "description": "For PDFs and presentations: specify page range to extract (e.g., '1-5', '2,4,6', 'all'). Default: 'all'", + "default": "all", + }, + "sheet_name": map[string]interface{}{ + "type": "string", + "description": "For Excel files: specify sheet name to extract. If not provided, extracts all sheets.", + }, + }, + "required": []string{"path"}, + }, + }, + { + Name: "create_pdf", + Description: "Create a PDF document from markdown or plain text content. Converts markdown formatting to styled PDF with proper headers, paragraphs, and basic formatting. Saves the PDF to the workspace directory and provides both a file path and clickable file URL that opens the document in the default PDF viewer. Perfect for generating reports, documentation, notes, or any text-based content in PDF format.", + InputSchema: map[string]interface{}{ + "type": "object", + "properties": map[string]interface{}{ + "filename": map[string]interface{}{ + "type": "string", + "description": "Name for the PDF file (e.g., 'report.pdf', 'notes.pdf', 'document.pdf'). Will be saved in the MCP workspace directory. Extension .pdf will be added automatically if not provided.", + }, + "content": map[string]interface{}{ + "type": "string", + "description": "Content to include in the PDF. Supports markdown formatting including headers (# ## ###), paragraphs, bold (**text**), italic (*text*), bullet points (- item), and numbered lists (1. item). Automatically handles line breaks and basic text formatting.", + }, + "title": map[string]interface{}{ + "type": "string", + "description": "Optional title for the PDF document. Will be displayed as the main heading and set as document metadata. If not provided, uses the filename without extension.", + }, + "author": map[string]interface{}{ + "type": "string", + "description": "Optional author name for the PDF metadata. Default: 'Clara MCP'", + "default": "Clara MCP", + }, + }, + "required": []string{"filename", "content"}, + }, + }, + } +} + +// Tool implementations + +func (s *PythonMCPServer) py(params map[string]interface{}) string { + code, ok := params["code"].(string) + if !ok { + return "ERROR: Need 'code'" + } + + // Auto-print last expression if it looks like one + lines := strings.Split(strings.TrimSpace(code), "\n") + if len(lines) > 0 { + last := strings.TrimSpace(lines[len(lines)-1]) + // Simple heuristic: if it doesn't look like a statement, print it + if last != "" && !strings.Contains(last, "=") && !strings.HasPrefix(last, "print") && + !strings.HasPrefix(last, "if") && !strings.HasPrefix(last, "for") && + !strings.HasPrefix(last, "while") && !strings.HasPrefix(last, "def") && + !strings.HasPrefix(last, "class") && !strings.HasPrefix(last, "import") && + !strings.HasPrefix(last, "from") && !strings.HasPrefix(last, "return") && + !strings.HasPrefix(last, "try") && !strings.HasPrefix(last, "except") { + lines[len(lines)-1] = fmt.Sprintf("print(%s)", last) + code = strings.Join(lines, "\n") + } + } + + // Get list of files before execution + filesBefore := make(map[string]bool) + if files, err := ioutil.ReadDir(s.workspaceDir); err == nil { + for _, f := range files { + if !f.IsDir() && f.Name() != ".venv" { + filesBefore[f.Name()] = true + } + } + } + + // Execute Python directly with -c flag using venv Python + cmd := exec.Command(s.pythonPath, "-c", code) + cmd.Dir = s.workspaceDir + + // Set environment to ensure venv is active + cmd.Env = os.Environ() + if runtime.GOOS == "windows" { + cmd.Env = append(cmd.Env, fmt.Sprintf("VIRTUAL_ENV=%s", s.venvPath)) + } + + output, err := cmd.CombinedOutput() + + result := strings.TrimSpace(string(output)) + if err != nil { + if result == "" { + result = err.Error() + } + // Clean up common Python error formats + if strings.Contains(result, "Traceback") { + lines := strings.Split(result, "\n") + if len(lines) > 0 { + lastLine := lines[len(lines)-1] + if strings.Contains(lastLine, "Error:") { + result = lastLine + } + } + } + return fmt.Sprintf("ERROR: %s", result) + } + + if result == "" { + result = "OK (no output)" + } + + // Check for newly created files and auto-open them + if files, err := ioutil.ReadDir(s.workspaceDir); err == nil { + var newFiles []string + for _, f := range files { + if !f.IsDir() && f.Name() != ".venv" && !filesBefore[f.Name()] { + if shouldAutoOpenFile(f.Name()) { + newFiles = append(newFiles, f.Name()) + } + } + } + + // Auto-open new files + if len(newFiles) > 0 { + for _, fileName := range newFiles { + filePath := filepath.Join(s.workspaceDir, fileName) + if err := s.openFile(filePath); err == nil { + result += fmt.Sprintf("\n✓ Opened %s", fileName) + } + } + } + } + + return result +} + +func (s *PythonMCPServer) sh(params map[string]interface{}) string { + command, ok := params["cmd"].(string) + if !ok { + return "ERROR: Need 'cmd'" + } + + var cmd *exec.Cmd + + // Special handling for Python/pip commands to use venv + lowerCmd := strings.ToLower(command) + if strings.HasPrefix(lowerCmd, "python ") || lowerCmd == "python" { + // Replace python with venv python + args := strings.Split(command, " ")[1:] + cmd = exec.Command(s.pythonPath, args...) + } else if strings.HasPrefix(lowerCmd, "pip ") || lowerCmd == "pip" { + // Replace pip with venv pip + args := strings.Split(command, " ")[1:] + cmd = exec.Command(s.pipPath, args...) + } else { + // Regular shell command + switch runtime.GOOS { + case "windows": + cmd = exec.Command("powershell", "-Command", command) + default: + cmd = exec.Command("sh", "-c", command) + } + } + + cmd.Dir = s.workspaceDir + + // Set environment to include venv + cmd.Env = os.Environ() + if runtime.GOOS == "windows" { + cmd.Env = append(cmd.Env, fmt.Sprintf("VIRTUAL_ENV=%s", s.venvPath)) + // Update PATH to include venv Scripts + for i, env := range cmd.Env { + if strings.HasPrefix(env, "PATH=") || strings.HasPrefix(env, "Path=") { + cmd.Env[i] = fmt.Sprintf("%s;%s", env, filepath.Join(s.venvPath, "Scripts")) + break + } + } + } else { + cmd.Env = append(cmd.Env, fmt.Sprintf("VIRTUAL_ENV=%s", s.venvPath)) + // Update PATH to include venv bin + for i, env := range cmd.Env { + if strings.HasPrefix(env, "PATH=") { + cmd.Env[i] = fmt.Sprintf("PATH=%s:%s", filepath.Join(s.venvPath, "bin"), strings.TrimPrefix(env, "PATH=")) + break + } + } + } + + output, err := cmd.CombinedOutput() + + result := strings.TrimSpace(string(output)) + if err != nil { + if result == "" { + result = err.Error() + } + return fmt.Sprintf("ERROR: %s", result) + } + + if result == "" { + result = "OK" + } + return result +} + +func (s *PythonMCPServer) pip(params map[string]interface{}) string { + pkg, ok := params["pkg"].(string) + if !ok { + return "ERROR: Need 'pkg'" + } + + // Use venv pip + cmd := exec.Command(s.pipPath, "install", pkg) + cmd.Dir = s.workspaceDir + output, err := cmd.CombinedOutput() + + if err != nil { + return fmt.Sprintf("ERROR: %s", strings.TrimSpace(string(output))) + } + + return fmt.Sprintf("Installed %s in virtual environment", pkg) +} + +func (s *PythonMCPServer) save(params map[string]interface{}) string { + name, ok := params["name"].(string) + if !ok { + return "ERROR: Need 'name'" + } + + text, ok := params["text"].(string) + if !ok { + return "ERROR: Need 'text'" + } + + // Force save to workspace + fullPath := filepath.Join(s.workspaceDir, filepath.Base(name)) + + // Security check: ensure the file is within workspace directory + workspaceAbs, _ := filepath.Abs(s.workspaceDir) + fullPathAbs, _ := filepath.Abs(fullPath) + if !strings.HasPrefix(fullPathAbs, workspaceAbs) { + return "ERROR: Invalid file path - outside workspace directory" + } + + if err := ioutil.WriteFile(fullPath, []byte(text), 0644); err != nil { + return fmt.Sprintf("ERROR: %v", err) + } + + // Auto-open file if it's a viewable type (unless explicitly disabled) + result := fmt.Sprintf("Saved %s (%d bytes)", filepath.Base(name), len(text)) + + // Check if auto_open parameter is provided (defaults to true for backwards compatibility) + autoOpen := true + if autoOpenParam, ok := params["auto_open"].(bool); ok { + autoOpen = autoOpenParam + } + + if autoOpen && shouldAutoOpenFile(name) { + if err := s.openFile(fullPath); err == nil { + result += fmt.Sprintf("\n✓ Opened %s", filepath.Base(name)) + } + } + + return result +} + +func (s *PythonMCPServer) load(params map[string]interface{}) string { + name, ok := params["name"].(string) + if !ok { + return "ERROR: Need 'name'" + } + + // Look in workspace + fullPath := filepath.Join(s.workspaceDir, filepath.Base(name)) + + content, err := ioutil.ReadFile(fullPath) + if err != nil { + return fmt.Sprintf("ERROR: %v", err) + } + + return string(content) +} + +func (s *PythonMCPServer) ls(params map[string]interface{}) string { + files, err := ioutil.ReadDir(s.workspaceDir) + if err != nil { + return fmt.Sprintf("ERROR: %v", err) + } + + if len(files) == 0 { + return "No files in workspace" + } + + var items []string + for _, f := range files { + // Skip .venv directory in listing + if f.Name() == ".venv" { + continue + } + + if f.IsDir() { + items = append(items, fmt.Sprintf("[DIR] %s", f.Name())) + } else { + size := f.Size() + unit := "B" + if size > 1024*1024 { + size = size / (1024 * 1024) + unit = "MB" + } else if size > 1024 { + size = size / 1024 + unit = "KB" + } + items = append(items, fmt.Sprintf("%s (%d%s)", f.Name(), size, unit)) + } + } + + if len(items) == 0 { + return "No files in workspace (excluding .venv)" + } + + return strings.Join(items, "\n") +} + +func (s *PythonMCPServer) open(params map[string]interface{}) string { + var cmd *exec.Cmd + + switch runtime.GOOS { + case "windows": + cmd = exec.Command("explorer", s.workspaceDir) + case "darwin": + cmd = exec.Command("open", s.workspaceDir) + case "linux": + if _, err := exec.LookPath("xdg-open"); err == nil { + cmd = exec.Command("xdg-open", s.workspaceDir) + } else if _, err := exec.LookPath("nautilus"); err == nil { + cmd = exec.Command("nautilus", s.workspaceDir) + } else if _, err := exec.LookPath("dolphin"); err == nil { + cmd = exec.Command("dolphin", s.workspaceDir) + } else if _, err := exec.LookPath("thunar"); err == nil { + cmd = exec.Command("thunar", s.workspaceDir) + } else { + return "ERROR: No file manager found. Workspace at: " + s.workspaceDir + } + default: + return "ERROR: Unsupported OS. Workspace at: " + s.workspaceDir + } + + if err := cmd.Start(); err != nil { + return fmt.Sprintf("ERROR: Failed to open folder: %v\nWorkspace at: %s", err, s.workspaceDir) + } + + go func() { + cmd.Wait() + }() + + return fmt.Sprintf("Opened workspace folder: %s", s.workspaceDir) +} + +// openFile opens a specific file with the default system application +func (s *PythonMCPServer) openFile(filePath string) error { + var cmd *exec.Cmd + + switch runtime.GOOS { + case "windows": + // Use 'start' command to open file with default application + cmd = exec.Command("cmd", "/c", "start", "", filePath) + case "darwin": + cmd = exec.Command("open", filePath) + case "linux": + if _, err := exec.LookPath("xdg-open"); err == nil { + cmd = exec.Command("xdg-open", filePath) + } else { + return fmt.Errorf("no file opener found") + } + default: + return fmt.Errorf("unsupported OS") + } + + if err := cmd.Start(); err != nil { + return err + } + + go func() { + cmd.Wait() + }() + + return nil +} + +// shouldAutoOpenFile determines if a file should be auto-opened based on extension +func shouldAutoOpenFile(filename string) bool { + ext := strings.ToLower(filepath.Ext(filename)) + autoOpenExts := []string{ + ".pdf", ".csv", ".xlsx", ".xls", + ".png", ".jpg", ".jpeg", ".gif", ".bmp", ".svg", + ".html", ".htm", + ".txt", ".md", + } + + for _, e := range autoOpenExts { + if ext == e { + return true + } + } + return false +} + +// Embedded SearXNG and Web Content functionality + +// NewSearXNGManager creates a new SearXNG manager +func NewSearXNGManager() *SearXNGManager { + return &SearXNGManager{} +} + +func (sm *SearXNGManager) getConfigDir() (string, error) { + if sm.configDir != "" { + return sm.configDir, nil + } + + dir, err := resolveSearXNGConfigDir() + if err != nil { + return "", err + } + + sm.configDir = dir + return sm.configDir, nil +} + +func resolveSearXNGConfigDir() (string, error) { + var ( + errorsList []string + tried = make(map[string]struct{}) + successDir string + ) + + tryPath := func(path string) bool { + path = strings.TrimSpace(path) + if path == "" { + return false + } + if _, seen := tried[path]; seen { + return false + } + tried[path] = struct{}{} + + abs, err := filepath.Abs(filepath.Clean(path)) + if err != nil { + errorsList = append(errorsList, fmt.Sprintf("%s (abs: %v)", path, err)) + return false + } + + if err := os.MkdirAll(abs, 0755); err != nil { + errorsList = append(errorsList, fmt.Sprintf("%s (%v)", abs, err)) + return false + } + + successDir = abs + return true + } + + // 1. User override via environment variable + if envDir := os.Getenv(ConfigEnvVar); tryPath(envDir) { + return successDir, nil + } + + // 2. Legacy current working directory path (for development setups) + if cwd, err := os.Getwd(); err == nil { + tryPath(filepath.Join(cwd, ConfigDirName)) + if successDir != "" { + return successDir, nil + } + } + + // 3. Platform-specific config locations + if homeDir, err := os.UserHomeDir(); err == nil && homeDir != "" { + switch runtime.GOOS { + case "darwin": + if tryPath(filepath.Join(homeDir, "Library", "Application Support", "ClaraVerse", ConfigDirName)) { + return successDir, nil + } + case "windows": + if localAppData := os.Getenv("LOCALAPPDATA"); strings.TrimSpace(localAppData) != "" { + if tryPath(filepath.Join(localAppData, "ClaraVerse", ConfigDirName)) { + return successDir, nil + } + } + if tryPath(filepath.Join(homeDir, "AppData", "Local", "ClaraVerse", ConfigDirName)) { + return successDir, nil + } + default: + if xdgConfig := os.Getenv("XDG_CONFIG_HOME"); strings.TrimSpace(xdgConfig) != "" { + if tryPath(filepath.Join(xdgConfig, "claraverse", ConfigDirName)) { + return successDir, nil + } + } + if tryPath(filepath.Join(homeDir, ".config", "claraverse", ConfigDirName)) { + return successDir, nil + } + } + + if tryPath(filepath.Join(homeDir, ".claraverse", ConfigDirName)) { + return successDir, nil + } + } + + // 4. System temporary directory fallback + if tryPath(filepath.Join(os.TempDir(), "claraverse", ConfigDirName)) { + return successDir, nil + } + + if len(errorsList) == 0 { + return "", fmt.Errorf("unable to determine SearXNG config directory: no valid locations") + } + + return "", fmt.Errorf("unable to create SearXNG config directory (tried: %s)", strings.Join(errorsList, "; ")) +} + +// NewWebContentFetcher creates a new web content fetcher with smart dynamic detection +func NewWebContentFetcher() *WebContentFetcher { + return &WebContentFetcher{ + client: &http.Client{ + Timeout: 30 * time.Second, + CheckRedirect: func(req *http.Request, via []*http.Request) error { + if len(via) >= 10 { + return fmt.Errorf("too many redirects") + } + return nil + }, + }, + jsEngine: &JSEngine{ + apiDetector: regexp.MustCompile(`(?i)(fetch\(|XMLHttpRequest|axios\.|$.ajax|$.get|$.post|api/|/api/|graphql)`), + domSimulator: &DOMSimulator{ + virtualDOM: make(map[string]interface{}), + }, + }, + smartMode: true, + playwrightManager: NewPlaywrightManager(), + chromedpManager: NewChromeDPManager(), + } +} + +// CheckDockerInstalled checks if Docker is available +func (sm *SearXNGManager) CheckDockerInstalled() error { + cmd := exec.Command("docker", "--version") + if err := cmd.Run(); err != nil { + return fmt.Errorf("Docker is not installed or not running. Please install Docker Desktop and ensure it's running") + } + return nil +} + +// CheckContainerExists checks if the SearXNG container exists +func (sm *SearXNGManager) CheckContainerExists() bool { + cmd := exec.Command("docker", "ps", "-a", "--filter", fmt.Sprintf("name=%s", ContainerName), "--format", "{{.Names}}") + output, err := cmd.Output() + if err != nil { + return false + } + return strings.TrimSpace(string(output)) == ContainerName +} + +// CheckContainerRunning checks if the SearXNG container is running +func (sm *SearXNGManager) CheckContainerRunning() bool { + cmd := exec.Command("docker", "ps", "--filter", fmt.Sprintf("name=%s", ContainerName), "--format", "{{.Names}}") + output, err := cmd.Output() + if err != nil { + return false + } + return strings.TrimSpace(string(output)) == ContainerName +} + +// CreateSearXNGConfig creates a proper SearXNG configuration +func (sm *SearXNGManager) CreateSearXNGConfig() error { + configDir, err := sm.getConfigDir() + if err != nil { + return fmt.Errorf("failed to resolve config directory: %v", err) + } + + if err := os.MkdirAll(configDir, 0755); err != nil { + return fmt.Errorf("failed to create config directory: %v", err) + } + + settingsContent := `# SearXNG settings for Clara MCP +use_default_settings: true + +general: + debug: false + instance_name: "Clara SearXNG" + contact_url: false + enable_metrics: false + +search: + safe_search: 0 + autocomplete: "" + default_lang: "en" + ban_time_on_fail: 5 + max_ban_time_on_fail: 120 + formats: + - html + - json + +server: + port: 8080 + bind_address: "0.0.0.0" + secret_key: "clara-secret-key-for-searxng" + base_url: false + image_proxy: true + static_use_hash: false + +ui: + static_use_hash: false + default_locale: "en" + query_in_title: false + infinite_scroll: false + center_alignment: false + cache_url: "https://web.archive.org/web/" + search_on_category_select: true + hotkeys: default + +# Disable bot detection for local use +botdetection: + ip_limit: + filter_link_local: false + link_token: false + ip_lists: + pass_searx_org: false + pass_ip: [] + block_ip: [] + +# Enable all default engines +engines: + - name: google + engine: google + use_mobile_ui: false + + - name: bing + engine: bing + + - name: duckduckgo + engine: duckduckgo + + - name: wikipedia + engine: wikipedia + + - name: github + engine: github + +enabled_plugins: + - 'Hash plugin' + - 'Search on category select' + - 'Self Information' + - 'Tracker URL remover' + - 'Ahmia blacklist' +` + + settingsPath := filepath.Join(configDir, "settings.yml") + if err := ioutil.WriteFile(settingsPath, []byte(settingsContent), 0644); err != nil { + return fmt.Errorf("failed to write settings.yml: %v", err) + } + + // Create limiter.toml to avoid warnings + limiterContent := `# SearXNG limiter configuration +[botdetection.ip_limit] +# Disable aggressive bot detection for local use +filter_link_local = false + +[botdetection.ip_lists] +pass_ip = ["127.0.0.1", "::1", "192.168.0.0/16", "10.0.0.0/8", "172.16.0.0/12"] +` + + limiterPath := filepath.Join(configDir, "limiter.toml") + if err := ioutil.WriteFile(limiterPath, []byte(limiterContent), 0644); err != nil { + return fmt.Errorf("failed to write limiter.toml: %v", err) + } + + fmt.Printf("✅ Created SearXNG configuration in %s\n", configDir) + return nil +} + +// StartContainer starts the SearXNG container +func (sm *SearXNGManager) StartContainer() error { + if err := sm.CheckDockerInstalled(); err != nil { + return err + } + + // Check if container is already running + if sm.CheckContainerRunning() { + fmt.Println("✅ SearXNG container is already running") + sm.isRunning = true + return nil + } + + // Create configuration first + if err := sm.CreateSearXNGConfig(); err != nil { + return fmt.Errorf("failed to create SearXNG config: %v", err) + } + + // Pull image if not exists + fmt.Println("🐳 Pulling SearXNG Docker image...") + pullCmd := exec.Command("docker", "pull", SearXNGImage) + if err := pullCmd.Run(); err != nil { + return fmt.Errorf("failed to pull SearXNG image: %v", err) + } + + configDir, err := sm.getConfigDir() + if err != nil { + return fmt.Errorf("failed to resolve config directory: %v", err) + } + + if err := os.MkdirAll(configDir, 0755); err != nil { + return fmt.Errorf("failed to prepare config directory: %v", err) + } + + // Check if container exists but is stopped + if sm.CheckContainerExists() { + fmt.Println("🚀 Starting existing SearXNG container...") + cmd := exec.Command("docker", "start", ContainerName) + if err := cmd.Run(); err != nil { + return fmt.Errorf("failed to start existing container: %v", err) + } + } else { + // Create and start new container with proper configuration and volume mount + fmt.Println("🚀 Creating and starting SearXNG container...") + cmd := exec.Command("docker", "run", "-d", + "--name", ContainerName, + "-p", fmt.Sprintf("%s:8080", SearXNGPort), + "-v", fmt.Sprintf("%s:/etc/searxng", configDir), + "-e", "SEARXNG_BASE_URL=http://localhost:8080/", + "-e", "SEARXNG_SECRET=clara-secret-key-for-searxng", + "--add-host=host.docker.internal:host-gateway", // For localhost access + SearXNGImage) + + output, err := cmd.Output() + if err != nil { + return fmt.Errorf("failed to create container: %v", err) + } + sm.containerID = strings.TrimSpace(string(output)) + } + + // Wait for container to be healthy + return sm.WaitForHealthy() +} + +// WaitForHealthy waits for the SearXNG container to be ready +func (sm *SearXNGManager) WaitForHealthy() error { + fmt.Println("⏳ Waiting for SearXNG to be ready...") + + timeout := 60 * time.Second + start := time.Now() + + for time.Since(start) < timeout { + if !sm.CheckContainerRunning() { + time.Sleep(2 * time.Second) + continue + } + + client := &http.Client{Timeout: 5 * time.Second} + resp, err := client.Get(SearXNGURL + "/") + if err == nil && resp.StatusCode == 200 { + resp.Body.Close() + fmt.Println("✅ SearXNG is ready!") + sm.isRunning = true + return nil + } + if resp != nil { + resp.Body.Close() + } + + time.Sleep(2 * time.Second) + } + + return fmt.Errorf("timeout waiting for SearXNG to be ready") +} + +// SearchSearXNG performs a search using SearXNG +func (sm *SearXNGManager) SearchSearXNG(query string, numResults int) (*SearchResponse, error) { + if !sm.isRunning && !sm.CheckContainerRunning() { + return nil, fmt.Errorf("SearXNG container is not running") + } + + if numResults <= 0 { + numResults = 10 + } + + // Build search URL + searchURL := fmt.Sprintf("%s%s?q=%s&format=json&pageno=1", + SearXNGURL, SearchPath, + strings.ReplaceAll(query, " ", "+")) + + fmt.Printf("🔍 Searching for: %s\n", query) + + // Perform HTTP request with proper headers for SearXNG + client := &http.Client{Timeout: 30 * time.Second} + req, err := http.NewRequest("GET", searchURL, nil) + if err != nil { + return nil, fmt.Errorf("failed to create search request: %v", err) + } + + // Add headers to satisfy SearXNG bot detection + req.Header.Set("User-Agent", "Clara-MCP-Client/1.0") + req.Header.Set("Accept", "application/json, text/html") + req.Header.Set("X-Forwarded-For", "127.0.0.1") + req.Header.Set("X-Real-IP", "127.0.0.1") + + resp, err := client.Do(req) + if err != nil { + return nil, fmt.Errorf("search request failed: %v", err) + } + defer resp.Body.Close() + + if resp.StatusCode != 200 { + body, _ := io.ReadAll(resp.Body) + return nil, fmt.Errorf("search failed with status %d: %s", resp.StatusCode, string(body)) + } + + // Parse JSON response + body, err := io.ReadAll(resp.Body) + if err != nil { + return nil, fmt.Errorf("failed to read response body: %v", err) + } + + var searchResp SearchResponse + if err := json.Unmarshal(body, &searchResp); err != nil { + return nil, fmt.Errorf("failed to parse search response: %v", err) + } + + // Limit results + if len(searchResp.Results) > numResults { + searchResp.Results = searchResp.Results[:numResults] + } + searchResp.NumberOfResults = len(searchResp.Results) + + fmt.Printf("✅ Found %d results\n", searchResp.NumberOfResults) + return &searchResp, nil +} + +// FetchContent fetches and extracts content from a URL using chromedp for dynamic content +func (wf *WebContentFetcher) FetchContent(targetURL string) *WebContent { + result := &WebContent{ + URL: targetURL, + LoadingStrategy: "chromedp", + } + startTime := time.Now() + + // Validate URL + parsedURL, err := url.Parse(targetURL) + if err != nil { + result.Error = fmt.Sprintf("Invalid URL: %v", err) + return result + } + + // Ensure URL has scheme + if parsedURL.Scheme == "" { + targetURL = "https://" + targetURL + result.URL = targetURL + } + + // Use chromedp for fast dynamic content fetching + log.Printf("Fetching content using chromedp for %s", targetURL) + + chromedpResult, err := wf.chromedpManager.FetchContent(targetURL, 10*time.Second) + if err != nil { + // Fallback to static fetch if chromedp fails + log.Printf("ChromeDP failed, falling back to HTTP: %v", err) + result = wf.fetchStatic(targetURL) + result.LoadTime = time.Since(startTime) + result.LoadingStrategy = "http-fallback" + return result + } + + // Update timing and return successful result + chromedpResult.LoadTime = time.Since(startTime) + log.Printf("Successfully extracted content using chromedp for %s (took %v)", targetURL, chromedpResult.LoadTime) + + return chromedpResult +} + +// fetchStatic performs standard static HTML fetching +func (wf *WebContentFetcher) fetchStatic(targetURL string) *WebContent { + result := &WebContent{URL: targetURL} + + // Create request with proper headers + req, err := http.NewRequest("GET", targetURL, nil) + if err != nil { + result.Error = fmt.Sprintf("Failed to create request: %v", err) + return result + } + + // Set realistic browser headers + req.Header.Set("User-Agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36") + req.Header.Set("Accept", "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8") + req.Header.Set("Accept-Language", "en-US,en;q=0.5") + req.Header.Set("Connection", "keep-alive") + + // Perform request + resp, err := wf.client.Do(req) + if err != nil { + result.Error = fmt.Sprintf("Request failed: %v", err) + return result + } + defer resp.Body.Close() + + result.StatusCode = resp.StatusCode + + // Check if response is successful + if resp.StatusCode >= 400 { + result.Error = fmt.Sprintf("HTTP error: %d %s", resp.StatusCode, resp.Status) + return result + } + + // Check content type + contentType := resp.Header.Get("Content-Type") + if !strings.Contains(strings.ToLower(contentType), "text/html") { + result.Error = fmt.Sprintf("Content type not supported: %s", contentType) + return result + } + + // Read response body + body, err := io.ReadAll(resp.Body) + if err != nil { + result.Error = fmt.Sprintf("Failed to read response: %v", err) + return result + } + + html := string(body) + + // Extract content + result.Title = wf.extractTitle(html) + result.Description = wf.extractDescription(html) + result.Content = wf.extractTextContent(html) + + return result +} + +// detectDynamicContent analyzes HTML for dynamic content indicators +func (wf *WebContentFetcher) detectDynamicContent(html string) bool { + // Check for common SPA indicators + spaIndicators := []string{ + "react", "vue", "angular", "svelte", "ember", + "ng-app", "data-react", "v-if", "v-for", + "useEffect", "useState", "componentDidMount", + "spa-", "_next/", "__nuxt", "@angular", + } + + htmlLower := strings.ToLower(html) + for _, indicator := range spaIndicators { + if strings.Contains(htmlLower, indicator) { + return true + } + } + + // Check for AJAX/API calls + if wf.jsEngine.apiDetector.MatchString(html) { + return true + } + + // Check for empty containers that typically get populated + emptyContainers := regexp.MustCompile(`]*(?:id|class)=['"](?:app|root|main|content|container)['"][^>]*>\s*`) + if emptyContainers.MatchString(html) { + return true + } + + // Check for loading indicators + loadingIndicators := regexp.MustCompile(`(?i)(loading|spinner|skeleton|placeholder)`) + if loadingIndicators.MatchString(html) { + return true + } + + return false +} + +// simulateDynamicContent attempts to extract content by finding and calling APIs +func (wf *WebContentFetcher) simulateDynamicContent(baseURL, html string) *WebContent { + result := &WebContent{URL: baseURL, IsDynamic: true} + + // Extract potential API endpoints + apiEndpoints := wf.extractAPIEndpoints(baseURL, html) + result.APIEndpoints = apiEndpoints + + // Try to fetch content from discovered APIs + var additionalContent []string + + for _, endpoint := range apiEndpoints { + if apiContent := wf.fetchAPIContent(endpoint); apiContent != "" { + additionalContent = append(additionalContent, apiContent) + } + } + + // Combine static content with API content + staticContent := wf.extractTextContent(html) + if len(additionalContent) > 0 { + allContent := append([]string{staticContent}, additionalContent...) + result.Content = strings.Join(allContent, "\n\n--- API Content ---\n\n") + } else { + result.Content = staticContent + } + + result.Title = wf.extractTitle(html) + result.Description = wf.extractDescription(html) + + return result +} + +// extractAPIEndpoints finds potential API endpoints in the HTML/JavaScript +func (wf *WebContentFetcher) extractAPIEndpoints(baseURL string, html string) []string { + var endpoints []string + + // Parse base URL + parsedBase, err := url.Parse(baseURL) + if err != nil { + return endpoints + } + + // Common API patterns + apiPatterns := []*regexp.Regexp{ + regexp.MustCompile(`['"]([^'"]*/?api/[^'"]*?)['"]`), + regexp.MustCompile(`['"]([^'"]*?/v\d+/[^'"]*?)['"]`), + regexp.MustCompile(`['"]([^'"]*?\.json[^'"]*?)['"]`), + regexp.MustCompile(`fetch\(['"]([^'"]+?)['"]`), + regexp.MustCompile(`axios\.get\(['"]([^'"]+?)['"]`), + } + + for _, pattern := range apiPatterns { + matches := pattern.FindAllStringSubmatch(html, -1) + for _, match := range matches { + if len(match) > 1 { + endpoint := match[1] + // Convert relative URLs to absolute + if strings.HasPrefix(endpoint, "/") { + endpoint = parsedBase.Scheme + "://" + parsedBase.Host + endpoint + } else if !strings.HasPrefix(endpoint, "http") { + continue // Skip relative paths that aren't root-relative + } + endpoints = append(endpoints, endpoint) + } + } + } + + // Remove duplicates + uniqueEndpoints := make(map[string]bool) + var result []string + for _, ep := range endpoints { + if !uniqueEndpoints[ep] && len(result) < 5 { // Limit to 5 endpoints + uniqueEndpoints[ep] = true + result = append(result, ep) + } + } + + return result +} + +// fetchAPIContent attempts to fetch content from an API endpoint +func (wf *WebContentFetcher) fetchAPIContent(endpoint string) string { + req, err := http.NewRequest("GET", endpoint, nil) + if err != nil { + return "" + } + + req.Header.Set("Accept", "application/json, text/plain, */*") + req.Header.Set("User-Agent", "Mozilla/5.0 (compatible; Clara-MCP)") + + client := &http.Client{Timeout: 10 * time.Second} + resp, err := client.Do(req) + if err != nil { + return "" + } + defer resp.Body.Close() + + if resp.StatusCode != 200 { + return "" + } + + body, err := io.ReadAll(resp.Body) + if err != nil { + return "" + } + + // Try to extract meaningful text from JSON responses + content := string(body) + if strings.HasPrefix(strings.TrimSpace(content), "{") || strings.HasPrefix(strings.TrimSpace(content), "[") { + // Basic JSON content extraction + content = wf.extractTextFromJSON(content) + } + + return content +} + +// extractTextFromJSON extracts readable text from JSON responses +func (wf *WebContentFetcher) extractTextFromJSON(jsonStr string) string { + var data interface{} + if err := json.Unmarshal([]byte(jsonStr), &data); err != nil { + return "" + } + + var textParts []string + wf.extractTextFromInterface(data, &textParts) + + return strings.Join(textParts, " ") +} + +// extractTextFromInterface recursively extracts text from JSON interface +func (wf *WebContentFetcher) extractTextFromInterface(data interface{}, textParts *[]string) { + switch v := data.(type) { + case string: + if len(v) > 10 && len(v) < 1000 { // Reasonable text length + *textParts = append(*textParts, v) + } + case map[string]interface{}: + for _, value := range v { + wf.extractTextFromInterface(value, textParts) + } + case []interface{}: + for _, item := range v { + wf.extractTextFromInterface(item, textParts) + } + } +} + +// enhancedStaticExtraction performs enhanced static content extraction +func (wf *WebContentFetcher) enhancedStaticExtraction(result *WebContent) *WebContent { + // This could include more sophisticated text extraction, + // meta tag analysis, structured data extraction, etc. + return result +} + +// extractTitle extracts the page title +func (wf *WebContentFetcher) extractTitle(html string) string { + titleRegex := regexp.MustCompile(`(?i)]*>([^<]*)`) + matches := titleRegex.FindStringSubmatch(html) + if len(matches) > 1 { + return strings.TrimSpace(wf.cleanText(matches[1])) + } + return "" +} + +// extractDescription extracts meta description +func (wf *WebContentFetcher) extractDescription(html string) string { + // Try meta description + descRegex := regexp.MustCompile(`(?i)]*name=["\']description["\'][^>]*content=["\']([^"\']*)["\']`) + matches := descRegex.FindStringSubmatch(html) + if len(matches) > 1 { + return strings.TrimSpace(wf.cleanText(matches[1])) + } + + // Try og:description + ogDescRegex := regexp.MustCompile(`(?i)]*property=["\']og:description["\'][^>]*content=["\']([^"\']*)["\']`) + matches = ogDescRegex.FindStringSubmatch(html) + if len(matches) > 1 { + return strings.TrimSpace(wf.cleanText(matches[1])) + } + + return "" +} + +// extractTextContent extracts main text content +func (wf *WebContentFetcher) extractTextContent(html string) string { + // Remove script and style tags + scriptRegex := regexp.MustCompile(`(?i)]*>.*?`) + html = scriptRegex.ReplaceAllString(html, "") + + styleRegex := regexp.MustCompile(`(?i)]*>.*?`) + html = styleRegex.ReplaceAllString(html, "") + + // Try to extract content from common content containers + contentSelectors := []string{ + `(?i)]*>(.*?)`, + `(?i)]*>(.*?)`, + `(?i)]*class=["\'][^"\']*content[^"\']*["\'][^>]*>(.*?)`, + `(?i)]*>(.*?)

`, + } + + var extractedText []string + + for _, selector := range contentSelectors { + regex := regexp.MustCompile(selector) + matches := regex.FindAllStringSubmatch(html, -1) + for _, match := range matches { + if len(match) > 1 { + text := wf.cleanText(match[1]) + if len(text) > 50 { + extractedText = append(extractedText, text) + } + } + } + if len(extractedText) > 0 { + break + } + } + + // Fallback: extract all text content + if len(extractedText) == 0 { + tagRegex := regexp.MustCompile(`<[^>]*>`) + text := tagRegex.ReplaceAllString(html, " ") + extractedText = append(extractedText, wf.cleanText(text)) + } + + // Join and limit content + content := strings.Join(extractedText, "\n\n") + if len(content) > 2000 { + content = content[:2000] + "..." + } + + return content +} + +// cleanText cleans extracted text +func (wf *WebContentFetcher) cleanText(text string) string { + // Decode HTML entities + text = strings.ReplaceAll(text, "&", "&") + text = strings.ReplaceAll(text, "<", "<") + text = strings.ReplaceAll(text, ">", ">") + text = strings.ReplaceAll(text, """, "\"") + text = strings.ReplaceAll(text, "'", "'") + text = strings.ReplaceAll(text, " ", " ") + + // Remove excessive whitespace + spaceRegex := regexp.MustCompile(`\s+`) + text = spaceRegex.ReplaceAllString(text, " ") + + return strings.TrimSpace(text) +} + +// search performs web search using embedded SearXNG functionality +func (s *PythonMCPServer) search(params map[string]interface{}) string { + query, ok := params["query"].(string) + if !ok { + return "ERROR: Need 'query' parameter" + } + + // Get optional num_results parameter + numResults := 10 + if nr, ok := params["num_results"].(float64); ok { + numResults = int(nr) + if numResults < 1 { + numResults = 1 + } else if numResults > 20 { + numResults = 20 + } + } + + // Create SearXNG manager + manager := NewSearXNGManager() + + // Start SearXNG container if not running + wasRunning := manager.CheckContainerRunning() + if !wasRunning { + if err := manager.StartContainer(); err != nil { + return fmt.Sprintf("ERROR: Failed to start SearXNG: %v", err) + } + } + + // Perform search + results, err := manager.SearchSearXNG(query, numResults) + if err != nil { + return fmt.Sprintf("ERROR: Search failed: %v", err) + } + + // Format results + var output strings.Builder + output.WriteString(fmt.Sprintf("🔍 SEARCH RESULTS for: %s\n", query)) + output.WriteString(fmt.Sprintf("📊 Found %d results\n", results.NumberOfResults)) + output.WriteString("⚡ Privacy-focused search via SearXNG\n\n") + + for i, result := range results.Results { + output.WriteString(fmt.Sprintf("%d. %s\n", i+1, result.Title)) + output.WriteString(fmt.Sprintf(" 🔗 %s\n", result.URL)) + if result.Content != "" { + content := strings.TrimSpace(result.Content) + if len(content) > 150 { + content = content[:150] + "..." + } + output.WriteString(fmt.Sprintf(" 📝 %s\n", content)) + } + if result.Engine != "" { + output.WriteString(fmt.Sprintf(" 🔍 Source: %s\n", result.Engine)) + } + output.WriteString("\n") + } + + if len(results.Suggestions) > 0 { + output.WriteString(fmt.Sprintf("� Suggestions: %s\n", strings.Join(results.Suggestions, ", "))) + } + + return output.String() +} + +// fetchContent fetches and extracts content from a web page using Playwright as the default standard +func (s *PythonMCPServer) fetchContent(params map[string]interface{}) string { + url, ok := params["url"].(string) + if !ok { + return "ERROR: Need 'url' parameter" + } + + // Create web content fetcher (always uses Playwright as default) + fetcher := NewWebContentFetcher() + + // Fetch content using Playwright + result := fetcher.FetchContent(url) + + if result.Error != "" { + return fmt.Sprintf("ERROR: %s", result.Error) + } + + // Format output + var output strings.Builder + output.WriteString(fmt.Sprintf("🌐 WEB CONTENT EXTRACTED from: %s\n", result.URL)) + output.WriteString(fmt.Sprintf("✅ Successfully fetched content (Status: %d)\n", result.StatusCode)) + + // Show loading strategy used + strategyIcon := "📄" + if result.IsDynamic { + strategyIcon = "⚡" + } + output.WriteString(fmt.Sprintf("%s Strategy: %s (Load time: %v)\n", strategyIcon, result.LoadingStrategy, result.LoadTime)) + + if result.IsDynamic { + output.WriteString("🔍 Dynamic content detected - used intelligent API simulation\n") + if len(result.APIEndpoints) > 0 { + output.WriteString(fmt.Sprintf("🔗 Discovered %d API endpoints\n", len(result.APIEndpoints))) + } + } + output.WriteString("\n") + + if result.Title != "" { + output.WriteString(fmt.Sprintf("📰 Title: %s\n\n", result.Title)) + } + + if result.Description != "" { + output.WriteString(fmt.Sprintf("📝 Description: %s\n\n", result.Description)) + } + + if result.Content != "" { + output.WriteString(fmt.Sprintf("📄 Content:\n%s\n", result.Content)) + } + + return output.String() +} + +// readDocument reads and extracts content from various document formats +func (s *PythonMCPServer) readDocument(params map[string]interface{}) string { + path, ok := params["path"].(string) + if !ok { + return "ERROR: Need 'path' parameter" + } + + // Check optional parameters + extractMetadata := false + if meta, ok := params["extract_metadata"].(bool); ok { + extractMetadata = meta + } + + pageRange := "all" + if pr, ok := params["page_range"].(string); ok { + pageRange = pr + } + + sheetName := "" + if sn, ok := params["sheet_name"].(string); ok { + sheetName = sn + } + + // Prepare Python script for document reading + scriptTemplate := ` +import sys +import os +import json +import traceback +from pathlib import Path +import urllib.request +import tempfile + +# Document processing libraries will be imported as needed +def safe_import(module_name, pip_name=None): + try: + return __import__(module_name) + except ImportError: + if pip_name: + import subprocess + subprocess.check_call([sys.executable, '-m', 'pip', 'install', pip_name]) + return __import__(module_name) + return None + +def read_document(file_path, extract_metadata=False, page_range="all", sheet_name=None): + result = { + "content": "", + "metadata": {}, + "format": "", + "pages": 0, + "error": None + } + + try: + # Handle URLs by downloading to temp file + temp_file = None + if file_path.startswith(('http://', 'https://')): + print(f"Downloading document from URL: {file_path}") + # Create temp file in current working directory (workspace) + temp_file = tempfile.NamedTemporaryFile(delete=False, suffix=Path(file_path).suffix, dir=os.getcwd()) + urllib.request.urlretrieve(file_path, temp_file.name) + file_path = temp_file.name + + # Check if file exists + if not os.path.exists(file_path): + result["error"] = f"File not found: {file_path}" + return result + + # Detect file format + file_ext = Path(file_path).suffix.lower() + result["format"] = file_ext + + print(f"Processing {file_ext} document: {os.path.basename(file_path)}") + + # Process based on file type + if file_ext == '.pdf': + result = read_pdf(file_path, extract_metadata, page_range, result) + elif file_ext in ['.docx', '.doc']: + result = read_word(file_path, extract_metadata, result) + elif file_ext in ['.xlsx', '.xls']: + result = read_excel(file_path, extract_metadata, sheet_name, result) + elif file_ext == '.csv': + result = read_csv(file_path, result) + elif file_ext in ['.pptx', '.ppt']: + result = read_powerpoint(file_path, extract_metadata, page_range, result) + elif file_ext == '.txt': + result = read_text(file_path, result) + elif file_ext == '.rtf': + result = read_rtf(file_path, result) + elif file_ext in ['.odt', '.ods', '.odp']: + result = read_libreoffice(file_path, extract_metadata, result) + elif file_ext == '.json': + result = read_json(file_path, result) + elif file_ext in ['.xml', '.html', '.htm']: + result = read_markup(file_path, result) + else: + result["error"] = f"Unsupported file format: {file_ext}. Supported formats: PDF, DOCX, DOC, XLSX, XLS, CSV, PPTX, PPT, TXT, RTF, ODT, ODS, ODP, JSON, XML, HTML" + + # Cleanup temp file + if temp_file: + os.unlink(temp_file.name) + + except Exception as e: + result["error"] = f"Error processing document: {str(e)}\n{traceback.format_exc()}" + + return result + +def read_pdf(file_path, extract_metadata, page_range, result): + try: + # Try PyPDF2 first, fallback to pdfplumber + PyPDF2 = safe_import('PyPDF2', 'PyPDF2') + if not PyPDF2: + pdfplumber = safe_import('pdfplumber', 'pdfplumber') + if pdfplumber: + return read_pdf_pdfplumber(file_path, extract_metadata, page_range, result) + else: + result["error"] = "Could not import PDF processing libraries" + return result + + with open(file_path, 'rb') as file: + pdf_reader = PyPDF2.PdfReader(file) + result["pages"] = len(pdf_reader.pages) + + if extract_metadata and pdf_reader.metadata: + result["metadata"] = { + "title": pdf_reader.metadata.get('/Title', ''), + "author": pdf_reader.metadata.get('/Author', ''), + "subject": pdf_reader.metadata.get('/Subject', ''), + "creator": pdf_reader.metadata.get('/Creator', ''), + "creation_date": str(pdf_reader.metadata.get('/CreationDate', '')), + "modification_date": str(pdf_reader.metadata.get('/ModDate', '')) + } + + # Parse page range + pages_to_extract = parse_page_range(page_range, result["pages"]) + + content_parts = [] + for page_num in pages_to_extract: + if 0 <= page_num < len(pdf_reader.pages): + page = pdf_reader.pages[page_num] + text = page.extract_text() + if text.strip(): + content_parts.append(f"--- Page {page_num + 1} ---\n{text}") + + result["content"] = "\n\n".join(content_parts) + + except Exception as e: + result["error"] = f"PDF processing error: {str(e)}" + + return result + +def read_pdf_pdfplumber(file_path, extract_metadata, page_range, result): + try: + pdfplumber = safe_import('pdfplumber', 'pdfplumber') + + with pdfplumber.open(file_path) as pdf: + result["pages"] = len(pdf.pages) + + if extract_metadata and pdf.metadata: + result["metadata"] = dict(pdf.metadata) + + pages_to_extract = parse_page_range(page_range, result["pages"]) + + content_parts = [] + for page_num in pages_to_extract: + if 0 <= page_num < len(pdf.pages): + page = pdf.pages[page_num] + text = page.extract_text() + if text and text.strip(): + content_parts.append(f"--- Page {page_num + 1} ---\n{text}") + + result["content"] = "\n\n".join(content_parts) + + except Exception as e: + result["error"] = f"PDF processing error: {str(e)}" + + return result + +def read_word(file_path, extract_metadata, result): + try: + python_docx = safe_import('docx', 'python-docx') + if not python_docx: + result["error"] = "Could not import python-docx library" + return result + + doc = python_docx.Document(file_path) + + if extract_metadata: + props = doc.core_properties + result["metadata"] = { + "title": props.title or '', + "author": props.author or '', + "subject": props.subject or '', + "created": str(props.created) if props.created else '', + "modified": str(props.modified) if props.modified else '', + "keywords": props.keywords or '' + } + + content_parts = [] + for para in doc.paragraphs: + text = para.text.strip() + if text: + content_parts.append(text) + + result["content"] = "\n\n".join(content_parts) + result["pages"] = len(doc.paragraphs) + + except Exception as e: + result["error"] = f"Word document processing error: {str(e)}" + + return result + +def read_excel(file_path, extract_metadata, sheet_name, result): + try: + pandas = safe_import('pandas', 'pandas openpyxl xlrd') + if not pandas: + result["error"] = "Could not import pandas library" + return result + + # Read Excel file + if sheet_name: + df = pandas.read_excel(file_path, sheet_name=sheet_name) + content_parts = [f"Sheet: {sheet_name}\n{df.to_string(index=False)}"] + else: + excel_file = pandas.ExcelFile(file_path) + content_parts = [] + for sheet in excel_file.sheet_names: + df = pandas.read_excel(file_path, sheet_name=sheet) + content_parts.append(f"Sheet: {sheet}\n{df.to_string(index=False)}") + + result["content"] = "\n\n" + "="*50 + "\n\n".join(content_parts) + result["pages"] = len(content_parts) + + if extract_metadata: + result["metadata"] = { + "sheets": excel_file.sheet_names if not sheet_name else [sheet_name], + "total_sheets": len(excel_file.sheet_names) + } + + except Exception as e: + result["error"] = f"Excel processing error: {str(e)}" + + return result + +def read_csv(file_path, result): + try: + pandas = safe_import('pandas', 'pandas') + if not pandas: + # Fallback to built-in csv + import csv + content_parts = [] + with open(file_path, 'r', encoding='utf-8', newline='') as file: + csv_reader = csv.reader(file) + for i, row in enumerate(csv_reader): + content_parts.append(" | ".join(row)) + result["content"] = "\n".join(content_parts) + else: + df = pandas.read_csv(file_path) + result["content"] = df.to_string(index=False) + + result["pages"] = 1 + + except Exception as e: + result["error"] = f"CSV processing error: {str(e)}" + + return result + +def read_powerpoint(file_path, extract_metadata, page_range, result): + try: + python_pptx = safe_import('pptx', 'python-pptx') + if not python_pptx: + result["error"] = "Could not import python-pptx library" + return result + + from pptx import Presentation + + prs = Presentation(file_path) + result["pages"] = len(prs.slides) + + if extract_metadata: + props = prs.core_properties + result["metadata"] = { + "title": props.title or '', + "author": props.author or '', + "subject": props.subject or '', + "created": str(props.created) if props.created else '', + "modified": str(props.modified) if props.modified else '' + } + + pages_to_extract = parse_page_range(page_range, result["pages"]) + + content_parts = [] + for slide_num in pages_to_extract: + if 0 <= slide_num < len(prs.slides): + slide = prs.slides[slide_num] + slide_text = [] + + for shape in slide.shapes: + if hasattr(shape, "text") and shape.text.strip(): + slide_text.append(shape.text) + + if slide_text: + content_parts.append(f"--- Slide {slide_num + 1} ---\n" + "\n".join(slide_text)) + + result["content"] = "\n\n".join(content_parts) + + except Exception as e: + result["error"] = f"PowerPoint processing error: {str(e)}" + + return result + +def read_text(file_path, result): + try: + with open(file_path, 'r', encoding='utf-8') as file: + result["content"] = file.read() + result["pages"] = 1 + except UnicodeDecodeError: + try: + with open(file_path, 'r', encoding='latin-1') as file: + result["content"] = file.read() + result["pages"] = 1 + except Exception as e: + result["error"] = f"Text file processing error: {str(e)}" + except Exception as e: + result["error"] = f"Text file processing error: {str(e)}" + + return result + +def read_rtf(file_path, result): + try: + striprtf = safe_import('striprtf', 'striprtf') + if not striprtf: + result["error"] = "Could not import striprtf library" + return result + + from striprtf.striprtf import rtf_to_text + + with open(file_path, 'r', encoding='utf-8') as file: + rtf_content = file.read() + + result["content"] = rtf_to_text(rtf_content) + result["pages"] = 1 + + except Exception as e: + result["error"] = f"RTF processing error: {str(e)}" + + return result + +def read_libreoffice(file_path, extract_metadata, result): + try: + # Try to use python-uno for LibreOffice files + result["error"] = "LibreOffice document support requires additional setup. Please convert to DOCX/PDF format." + return result + except Exception as e: + result["error"] = f"LibreOffice processing error: {str(e)}" + + return result + +def read_json(file_path, result): + try: + with open(file_path, 'r', encoding='utf-8') as file: + data = json.load(file) + + # Pretty print JSON content + result["content"] = json.dumps(data, indent=2, ensure_ascii=False) + result["pages"] = 1 + + except Exception as e: + result["error"] = f"JSON processing error: {str(e)}" + + return result + +def read_markup(file_path, result): + try: + BeautifulSoup = safe_import('bs4', 'beautifulsoup4').BeautifulSoup + + with open(file_path, 'r', encoding='utf-8') as file: + content = file.read() + + if file_path.lower().endswith(('.html', '.htm')): + soup = BeautifulSoup(content, 'html.parser') + # Extract text content + result["content"] = soup.get_text(separator='\n', strip=True) + else: + # For XML, just return pretty-printed content + soup = BeautifulSoup(content, 'xml') + result["content"] = soup.prettify() + + result["pages"] = 1 + + except Exception as e: + # Fallback to raw text + try: + with open(file_path, 'r', encoding='utf-8') as file: + result["content"] = file.read() + result["pages"] = 1 + except Exception as e2: + result["error"] = f"Markup processing error: {str(e2)}" + + return result + +def parse_page_range(page_range, total_pages): + if page_range == "all" or not page_range: + return list(range(total_pages)) + + pages = [] + for part in page_range.split(','): + part = part.strip() + if '-' in part: + start, end = map(int, part.split('-')) + pages.extend(range(start - 1, min(end, total_pages))) + else: + page_num = int(part) - 1 + if 0 <= page_num < total_pages: + pages.append(page_num) + + return sorted(set(pages)) + +# Main execution +try: + result = read_document("%s", %s, "%s", %s) + print("RESULT_START") + print(json.dumps(result, indent=2, ensure_ascii=False)) + print("RESULT_END") +except Exception as e: + error_result = {"error": f"Script execution error: {str(e)}", "content": "", "metadata": {}, "format": "", "pages": 0} + print("RESULT_START") + print(json.dumps(error_result, indent=2, ensure_ascii=False)) + print("RESULT_END") +` + + // Format the script with parameters + metadataStr := "False" + if extractMetadata { + metadataStr = "True" + } + + sheetNameParam := "None" + if sheetName != "" { + sheetNameParam = fmt.Sprintf(`"%s"`, sheetName) + } + + // Escape the path for Python string literal (handle Windows backslashes) + escapedPath := strings.ReplaceAll(path, `\`, `\\`) + + script := fmt.Sprintf(scriptTemplate, escapedPath, metadataStr, pageRange, sheetNameParam) + + // Execute the Python script + cmd := exec.Command(s.pythonPath, "-c", script) + cmd.Dir = s.workspaceDir + cmd.Env = os.Environ() + if runtime.GOOS == "windows" { + cmd.Env = append(cmd.Env, fmt.Sprintf("VIRTUAL_ENV=%s", s.venvPath)) + } + + // Ensure working directory is set correctly + if cmd.Dir == "" || cmd.Dir == "/" { + cmd.Dir = s.workspaceDir + } + + output, err := cmd.CombinedOutput() + outputStr := string(output) + + if err != nil { + // Try to extract any error information from the output + if strings.Contains(outputStr, "RESULT_START") { + // Script ran but had an error in document processing + start := strings.Index(outputStr, "RESULT_START") + len("RESULT_START") + end := strings.Index(outputStr, "RESULT_END") + if end > start { + jsonStr := strings.TrimSpace(outputStr[start:end]) + var result map[string]interface{} + if json.Unmarshal([]byte(jsonStr), &result) == nil { + if errorMsg, ok := result["error"].(string); ok && errorMsg != "" { + return fmt.Sprintf("ERROR: %s", errorMsg) + } + } + } + } + return fmt.Sprintf("ERROR: Script execution failed: %v\nOutput: %s", err, outputStr) + } + + // Parse the result from the script output + start := strings.Index(outputStr, "RESULT_START") + end := strings.Index(outputStr, "RESULT_END") + + if start == -1 || end == -1 { + return fmt.Sprintf("ERROR: Could not parse script output\nOutput: %s", outputStr) + } + + start += len("RESULT_START") + jsonStr := strings.TrimSpace(outputStr[start:end]) + + var result map[string]interface{} + if err := json.Unmarshal([]byte(jsonStr), &result); err != nil { + return fmt.Sprintf("ERROR: Could not parse JSON result: %v\nJSON: %s", err, jsonStr) + } + + // Check for errors in the result + if errorMsg, ok := result["error"].(string); ok && errorMsg != "" { + return fmt.Sprintf("ERROR: %s", errorMsg) + } + + // Format the successful result + var output_builder strings.Builder + + format, _ := result["format"].(string) + pages := int(result["pages"].(float64)) + content, _ := result["content"].(string) + metadata, _ := result["metadata"].(map[string]interface{}) + + output_builder.WriteString("📄 DOCUMENT READER RESULTS\n") + output_builder.WriteString("==========================\n\n") + output_builder.WriteString(fmt.Sprintf("📂 File: %s\n", filepath.Base(path))) + output_builder.WriteString(fmt.Sprintf("📋 Format: %s\n", strings.ToUpper(strings.TrimPrefix(format, ".")))) + output_builder.WriteString(fmt.Sprintf("📊 Pages/Sections: %d\n", pages)) + output_builder.WriteString(fmt.Sprintf("📝 Content Length: %d characters\n", len(content))) + + if extractMetadata && len(metadata) > 0 { + output_builder.WriteString("\n📋 METADATA\n") + output_builder.WriteString("===========\n") + for key, value := range metadata { + if valueStr, ok := value.(string); ok && valueStr != "" { + output_builder.WriteString(fmt.Sprintf("%s: %s\n", strings.Title(key), valueStr)) + } + } + } + + output_builder.WriteString("\n📄 CONTENT\n") + output_builder.WriteString("==========\n") + + if len(content) > 0 { + // Limit content display if too long + if len(content) > 10000 { + output_builder.WriteString(content[:10000]) + output_builder.WriteString(fmt.Sprintf("\n\n... [Content truncated - showing first 10,000 characters of %d total]", len(content))) + } else { + output_builder.WriteString(content) + } + } else { + output_builder.WriteString("No readable content found in the document.") + } + + return output_builder.String() +} + +// createPDF creates a PDF document from markdown content using Go PDF library +func (s *PythonMCPServer) createPDF(params map[string]interface{}) string { + filename, ok := params["filename"].(string) + if !ok { + return "ERROR: Need 'filename' parameter" + } + + content, ok := params["content"].(string) + if !ok { + return "ERROR: Need 'content' parameter" + } + + // Get optional parameters + title := "" + if t, ok := params["title"].(string); ok { + title = t + } + if title == "" { + // Use filename without extension as title + title = strings.TrimSuffix(filepath.Base(filename), filepath.Ext(filename)) + } + + author := "Clara MCP" + if a, ok := params["author"].(string); ok && a != "" { + author = a + } + + // Ensure filename has .pdf extension + if !strings.HasSuffix(strings.ToLower(filename), ".pdf") { + filename = filename + ".pdf" + } + + // Create full path in workspace - ensure it's within workspace + fullPath := filepath.Join(s.workspaceDir, filepath.Base(filename)) + + // Security check: ensure the file is within workspace directory + workspaceAbs, _ := filepath.Abs(s.workspaceDir) + fullPathAbs, _ := filepath.Abs(fullPath) + if !strings.HasPrefix(fullPathAbs, workspaceAbs) { + return "ERROR: Invalid file path - outside workspace directory" + } + + // Create PDF with UTF-8 support + pdf := gofpdf.New("P", "mm", "A4", "") + + // Enable UTF-8 support for proper bullet point rendering + tr := pdf.UnicodeTranslatorFromDescriptor("") + pdf.SetCreator("Clara MCP", true) + pdf.SetAuthor(tr(author), true) + pdf.SetTitle(tr(title), true) + pdf.SetSubject("Document created by Clara MCP", true) + + // Add page + pdf.AddPage() + + // Set font + pdf.SetFont("Arial", "", 12) + + // Add title if provided + if title != "" { + pdf.SetFont("Arial", "B", 16) + pdf.Cell(0, 10, tr(title)) + pdf.Ln(15) + pdf.SetFont("Arial", "", 12) + } + + // Process content - convert markdown-like formatting to PDF + lines := strings.Split(content, "\n") + + for _, line := range lines { + line = strings.TrimSpace(line) + + if line == "" { + pdf.Ln(5) // Empty line spacing + continue + } + + // Handle markdown headers + if strings.HasPrefix(line, "# ") { + pdf.Ln(5) + pdf.SetFont("Arial", "B", 14) + pdf.Cell(0, 8, tr(strings.TrimPrefix(line, "# "))) + pdf.Ln(10) + pdf.SetFont("Arial", "", 12) + } else if strings.HasPrefix(line, "## ") { + pdf.Ln(3) + pdf.SetFont("Arial", "B", 13) + pdf.Cell(0, 8, tr(strings.TrimPrefix(line, "## "))) + pdf.Ln(8) + pdf.SetFont("Arial", "", 12) + } else if strings.HasPrefix(line, "### ") { + pdf.Ln(2) + pdf.SetFont("Arial", "B", 12) + pdf.Cell(0, 7, tr(strings.TrimPrefix(line, "### "))) + pdf.Ln(7) + pdf.SetFont("Arial", "", 12) + } else if strings.HasPrefix(line, "- ") || strings.HasPrefix(line, "* ") { + // Bullet points with proper UTF-8 encoding + pdf.Cell(10, 6, tr("• ")) + s.addTextWithFormatting(pdf, tr, strings.TrimPrefix(strings.TrimPrefix(line, "- "), "* ")) + pdf.Ln(6) + } else if regexp.MustCompile(`^\d+\.\s`).MatchString(line) { + // Numbered lists + parts := regexp.MustCompile(`^(\d+\.\s)(.*)`).FindStringSubmatch(line) + if len(parts) == 3 { + pdf.Cell(10, 6, parts[1]) + s.addTextWithFormatting(pdf, tr, parts[2]) + pdf.Ln(6) + } + } else { + // Regular paragraph + s.addTextWithFormatting(pdf, tr, line) + pdf.Ln(6) + } + } + + // Save PDF + err := pdf.OutputFileAndClose(fullPath) + if err != nil { + return fmt.Sprintf("ERROR: Failed to save PDF: %v", err) + } + + // Get file info + fileInfo, err := os.Stat(fullPath) + if err != nil { + return fmt.Sprintf("ERROR: Failed to get file info: %v", err) + } + + // Create file URL for different operating systems + var fileURL string + switch runtime.GOOS { + case "windows": + fileURL = "file:///" + strings.ReplaceAll(fullPath, "\\", "/") + default: + fileURL = "file://" + fullPath + } + + // Auto-open the PDF file + if err := s.openFile(fullPath); err != nil { + log.Printf("Warning: Failed to auto-open PDF: %v", err) + } + + // Format result + var output strings.Builder + output.WriteString("📄 PDF CREATED SUCCESSFULLY\n") + output.WriteString("==========================\n\n") + output.WriteString(fmt.Sprintf("📁 File: %s\n", filepath.Base(fullPath))) + output.WriteString(fmt.Sprintf("📍 Location: %s\n", fullPath)) + output.WriteString(fmt.Sprintf("🔗 Click to open: %s\n", fileURL)) + output.WriteString(fmt.Sprintf("📊 Size: %.2f KB\n", float64(fileInfo.Size())/1024)) + output.WriteString(fmt.Sprintf("📝 Title: %s\n", title)) + output.WriteString(fmt.Sprintf("👤 Author: %s\n", author)) + output.WriteString(fmt.Sprintf("📅 Created: %s\n", fileInfo.ModTime().Format("2006-01-02 15:04:05"))) + output.WriteString("\n✓ PDF opened automatically in your default viewer") + + return output.String() +} + +// addTextWithFormatting adds text to PDF with basic markdown formatting and UTF-8 support +func (s *PythonMCPServer) addTextWithFormatting(pdf *gofpdf.Fpdf, tr func(string) string, text string) { + // Handle bold **text** + boldRegex := regexp.MustCompile(`\*\*(.*?)\*\*`) + // Handle italic *text* + italicRegex := regexp.MustCompile(`\*(.*?)\*`) + + // Simple approach: for now, just remove formatting markers + // A more sophisticated implementation would properly handle formatting + text = boldRegex.ReplaceAllString(text, "$1") + text = italicRegex.ReplaceAllString(text, "$1") + + // Split long lines to fit page width + maxWidth := 180.0 // mm + words := strings.Fields(text) + + currentLine := "" + for _, word := range words { + testLine := currentLine + if testLine != "" { + testLine += " " + } + testLine += word + + // Check if line fits (using translated text for width calculation) + lineWidth := pdf.GetStringWidth(tr(testLine)) + if lineWidth > maxWidth && currentLine != "" { + // Output current line and start new one (with UTF-8 encoding) + pdf.Cell(0, 6, tr(currentLine)) + pdf.Ln(6) + currentLine = word + } else { + currentLine = testLine + } + } + + // Output remaining text (with UTF-8 encoding) + if currentLine != "" { + pdf.Cell(0, 6, tr(currentLine)) + } +} + +// playwrightStatus shows the current status of Playwright integration +func (s *PythonMCPServer) playwrightStatus(params map[string]interface{}) string { + // Create web content fetcher to access Playwright manager + fetcher := NewWebContentFetcher() + + var output strings.Builder + output.WriteString("🎭 PLAYWRIGHT STATUS\n") + output.WriteString("==================\n\n") + + // Get Playwright capabilities + capabilities := fetcher.playwrightManager.GetCapabilities() + + // Parse capabilities + isAvailable, _ := capabilities["is_available"].(bool) + isDownloading, _ := capabilities["is_downloading"].(bool) + hasLibrary, _ := capabilities["has_playwright_lib"].(bool) + downloadStatus, _ := capabilities["download_status"].(string) + estimatedSize, _ := capabilities["estimated_size"].(string) + + // Status overview + statusIcon := "❌" + if isAvailable { + statusIcon = "✅" + } else if isDownloading { + statusIcon = "⬇️" + } + + output.WriteString(fmt.Sprintf("%s Overall Status: %s\n", statusIcon, downloadStatus)) + output.WriteString(fmt.Sprintf("📚 Playwright Library: %s\n", map[bool]string{true: "Available", false: "Not Available"}[hasLibrary])) + + if estimatedSize != "" { + output.WriteString(fmt.Sprintf("💾 Download Size: %s\n", estimatedSize)) + } + + output.WriteString("\n🔧 CAPABILITIES\n") + output.WriteString("===============\n") + + jsExecution, _ := capabilities["javascript_execution"].(bool) + networkInterception, _ := capabilities["network_interception"].(bool) + screenshotCapture, _ := capabilities["screenshot_capture"].(bool) + + output.WriteString(fmt.Sprintf("⚡ JavaScript Execution: %s\n", map[bool]string{true: "✅ Available", false: "❌ Not Available"}[jsExecution])) + output.WriteString(fmt.Sprintf("🌐 Network Interception: %s\n", map[bool]string{true: "✅ Available", false: "❌ Not Available"}[networkInterception])) + output.WriteString(fmt.Sprintf("📸 Screenshot Capture: %s\n", map[bool]string{true: "✅ Available", false: "❌ Not Available"}[screenshotCapture])) + + output.WriteString("\n🎯 PLAYWRIGHT STANDARD\n") + output.WriteString("=========================\n") + + if isAvailable { + output.WriteString("✅ Playwright is ready and active\n") + output.WriteString("✅ All web content uses full browser automation\n") + output.WriteString("✅ JavaScript execution and dynamic content fully supported\n") + } else if isDownloading { + output.WriteString("⬇️ Playwright is installing automatically...\n") + output.WriteString("⏳ Content requests will wait for installation to complete\n") + output.WriteString("� Once ready, all content will use browser automation\n") + } else { + output.WriteString("� Playwright installation required for content fetching\n") + output.WriteString("� Will auto-install on first content request (~50MB)\n") + output.WriteString("� No fallback modes - Playwright is the standard\n") + if !hasLibrary { + output.WriteString("📦 Playwright library needs to be installed\n") + } + } + + // Error information + if downloadErrorRaw := capabilities["download_error"]; downloadErrorRaw != nil { + output.WriteString(fmt.Sprintf("\n⚠️ NOTICE\n========\n%v\n", downloadErrorRaw)) + } + + // Installation information + installStatus := fetcher.playwrightManager.GetInstallationStatus() + version := fetcher.playwrightManager.Version() + + output.WriteString("\n📋 TECHNICAL DETAILS\n") + output.WriteString("==================\n") + + managerVersion, _ := version["manager_version"].(string) + implementation, _ := version["implementation"].(string) + + output.WriteString(fmt.Sprintf("Manager Version: %s\n", managerVersion)) + output.WriteString(fmt.Sprintf("Implementation: %s\n", implementation)) + + currentlyDownloading, _ := installStatus["currently_downloading"].(bool) + browsersInstalled, _ := installStatus["browsers_installed"].(bool) + + output.WriteString(fmt.Sprintf("Browsers Installed: %s\n", map[bool]string{true: "Yes", false: "No"}[browsersInstalled])) + output.WriteString(fmt.Sprintf("Currently Downloading: %s\n", map[bool]string{true: "Yes", false: "No"}[currentlyDownloading])) + + return output.String() +} + +// handleRequest processes requests +func (s *PythonMCPServer) handleRequest(req MCPRequest) MCPResponse { + resp := MCPResponse{ + JSONRPC: "2.0", + ID: req.ID, + } + + switch req.Method { + case "initialize": + resp.Result = map[string]interface{}{ + "protocolVersion": "2024-11-05", + "serverInfo": map[string]interface{}{ + "name": "python-mcp", + "version": "5.1.0", + }, + "capabilities": map[string]interface{}{ + "tools": map[string]interface{}{}, + }, + } + + case "tools/list": + resp.Result = map[string]interface{}{ + "tools": s.getTools(), + } + + case "tools/call": + var params struct { + Name string `json:"name"` + Arguments map[string]interface{} `json:"arguments"` + } + + if err := json.Unmarshal(req.Params, ¶ms); err != nil { + resp.Error = &MCPError{Code: -32602, Message: "Invalid params"} + return resp + } + + var result string + switch params.Name { + case "py": + result = s.py(params.Arguments) + case "sh", "powershell": // Support both for compatibility + result = s.sh(params.Arguments) + case "pip": + result = s.pip(params.Arguments) + case "save": + result = s.save(params.Arguments) + case "load": + result = s.load(params.Arguments) + case "ls": + result = s.ls(params.Arguments) + case "open": + result = s.open(params.Arguments) + case "search": + result = s.search(params.Arguments) + case "fetch_content": + result = s.fetchContent(params.Arguments) + case "read_document": + result = s.readDocument(params.Arguments) + case "create_pdf": + result = s.createPDF(params.Arguments) + default: + resp.Error = &MCPError{Code: -32603, Message: "Unknown tool"} + return resp + } + + resp.Result = map[string]interface{}{ + "content": []map[string]interface{}{ + { + "type": "text", + "text": result, + }, + }, + } + + default: + resp.Error = &MCPError{Code: -32601, Message: "Method not found"} + } + + return resp +} + +// runServer main loop +func (s *PythonMCPServer) runServer() { + scanner := bufio.NewScanner(os.Stdin) + encoder := json.NewEncoder(os.Stdout) + + for scanner.Scan() { + var req MCPRequest + if err := json.Unmarshal(scanner.Bytes(), &req); err != nil { + log.Printf("Parse error: %v", err) + continue + } + + resp := s.handleRequest(req) + if err := encoder.Encode(resp); err != nil { + log.Printf("Encode error: %v", err) + } + } + + if err := scanner.Err(); err != nil && err != io.EOF { + log.Printf("Scanner error: %v", err) + } +} + +func main() { + log.SetOutput(os.Stderr) + server := NewPythonMCPServer() + server.runServer() +} diff --git a/clara-mcp/simple-test.json b/clara-mcp/simple-test.json new file mode 100644 index 00000000..d2b4e3bd --- /dev/null +++ b/clara-mcp/simple-test.json @@ -0,0 +1 @@ +{"jsonrpc": "2.0", "id": 1, "method": "tools/call", "params": {"name": "create_pdf", "arguments": {"filename": "test.pdf", "title": "Test Document", "content": "# Hello World\n\nThis is a test PDF created by Clara MCP.\n\n## Features\n\n- PDF creation from markdown\n- Automatic formatting\n- File system integration\n\nThank you for testing!"}}} diff --git a/clara-mcp/test-pdf-creation.json b/clara-mcp/test-pdf-creation.json new file mode 100644 index 00000000..ac3b0a53 --- /dev/null +++ b/clara-mcp/test-pdf-creation.json @@ -0,0 +1,14 @@ +{ + "jsonrpc": "2.0", + "id": 1, + "method": "tools/call", + "params": { + "name": "create_pdf", + "arguments": { + "filename": "test-document.pdf", + "title": "Clara MCP Test Document", + "author": "Test User", + "content": "# Welcome to Clara MCP PDF Creation\n\nThis is a test document created using the new PDF creation tool in Clara MCP.\n\n## Features\n\n- **Markdown formatting** support\n- Automatic *text formatting*\n- Header hierarchy\n- Bullet points:\n - Item 1\n - Item 2\n - Item 3\n\n### Numbered Lists\n\n1. First item\n2. Second item\n3. Third item\n\n## Conclusion\n\nThis PDF was generated from markdown content using Go and the gofpdf library. The file will be saved to the MCP workspace and can be opened directly from the file system.\n\nThank you for testing Clara MCP's PDF creation capabilities!" + } + } +} diff --git a/claraverse-agent-deployement/QUICKSTART.md b/claraverse-agent-deployement/QUICKSTART.md new file mode 100644 index 00000000..3f2c8647 --- /dev/null +++ b/claraverse-agent-deployement/QUICKSTART.md @@ -0,0 +1,139 @@ +# ClaraVerse Remote Server - Quick Start + +Deploy ClaraVerse services (ComfyUI + Python Backend + n8n) on any server with one command! + +## 🚀 Super Quick Start + +**Any Platform (Windows, Linux, macOS):** +```bash +python install.py +``` + +The installer will: +1. ✅ Check your system (Docker, GPU, etc.) +2. 🎯 Let you choose which services to deploy +3. 📦 Download and start the selected services +4. 🏥 Monitor health and show you the URLs +5. 🌐 Open your browser to the services + +## 📱 Installation Examples + +### Deploy Everything (Recommended) +```bash +python install.py +# When prompted, select: A +``` + +### Deploy Only ComfyUI +```bash +python install.py +# When prompted, select: 1 +``` + +### Deploy ComfyUI + n8n Automation +```bash +python install.py +# When prompted, select: 1,3 +``` + +## 📦 What Gets Deployed + +- **ComfyUI**: AI image generation interface (Port 8188) +- **Python Backend**: Core AI processing APIs (Port 5001) +- **n8n**: Workflow automation platform (Port 5678) + +## 🌐 Access Points + +After deployment, access your services directly at: + +- **ComfyUI**: `http://your-server-ip:8188` +- **Python Backend**: `http://your-server-ip:5001` +- **n8n**: `http://your-server-ip:5678` + +## ⚙️ Configuration + +### Default Credentials +- **n8n**: `admin / clara123` (change in docker-compose.yml) + +### GPU Support +- Automatically detects NVIDIA GPUs +- Falls back to CPU if no GPU available +- Works on Windows, Linux, and macOS + +### Ports +- **ComfyUI**: 8188 +- **Python Backend**: 5001 +- **n8n**: 5678 + +## 🔧 Management + +### Check Status +```bash +docker compose ps +``` + +### View Logs +```bash +docker compose logs -f +``` + +### Restart Services +```bash +docker compose restart +``` + +### Stop All Services +```bash +docker compose down +``` + +### Update Images +```bash +docker compose pull +docker compose up -d +``` + +## 📋 System Requirements + +### Minimum +- 4GB RAM +- 20GB storage +- Docker with Compose support + +### Recommended +- 8GB+ RAM +- 50GB+ storage (for AI models) +- NVIDIA GPU with 6GB+ VRAM +- Ubuntu 20.04+ or Windows 10+ with WSL2 + +## 🐛 Troubleshooting + +### Services Won't Start +1. Check Docker is running: `docker info` +2. Check logs: `docker compose logs` +3. Verify ports aren't in use: `netstat -tulpn` + +### GPU Not Detected +1. Install NVIDIA drivers +2. Install NVIDIA Container Toolkit +3. Restart Docker daemon +4. Test: `docker run --rm --gpus all nvidia/cuda:11.8-base-ubuntu20.04 nvidia-smi` + +### Can't Access Services +1. Check firewall settings +2. Verify service health: `curl http://localhost/health` +3. Check container status: `docker compose ps` + +## 🔒 Security Notes + +- Change default n8n password +- Use HTTPS in production +- Configure firewall rules +- Regular security updates + +## 📞 Support + +For issues and support: +1. Check logs: `docker compose logs` +2. Verify system requirements +3. Check firewall and network settings \ No newline at end of file diff --git a/claraverse-agent-deployement/README.md b/claraverse-agent-deployement/README.md new file mode 100644 index 00000000..7769672b --- /dev/null +++ b/claraverse-agent-deployement/README.md @@ -0,0 +1,85 @@ +# ClaraVerse Remote Server + +A simple deployment package for hosting ClaraVerse services on remote servers with direct port access. + +## 🚀 One-Command Installation + +**Windows, Linux, or macOS:** +```bash +python install.py +``` + +That's it! The installer will guide you through everything. + +## 📋 Prerequisites + +- **Python 3.6+** (comes pre-installed on most systems) +- **Docker Desktop** (Windows/Mac) or **Docker Engine** (Linux) +- **Internet connection** (for downloading containers) + +## 🎯 What You Get + +The installer lets you choose which services to deploy: + +- **ComfyUI** (Port 8188): AI image generation and workflows +- **Python Backend** (Port 5001): Core AI processing and APIs +- **n8n** (Port 5678): Workflow automation platform + +## 🎨 Beautiful Interactive Installation + +The Python installer provides: +- ✨ **Colorful terminal interface** with progress bars +- 🔍 **Automatic system detection** (Docker, GPU, OS) +- ⚙️ **Interactive service selection** - pick what you need +- 🏥 **Health monitoring** - ensures everything starts correctly +- 🌐 **Auto browser opening** to your deployed services + +## 🏗️ Architecture + +``` +┌─────────────────────────────────────────┐ +│ Remote Server │ +│ │ +│ ComfyUI → Port 8188 │ +│ Python Backend → Port 5001 │ +│ n8n → Port 5678 │ +│ │ +│ Direct access to each service: │ +│ http://server:8188 (ComfyUI) │ +│ http://server:5001 (Python API) │ +│ http://server:5678 (n8n) │ +└─────────────────────────────────────────┘ +``` + +## 💡 Usage Examples + +**Deploy everything:** +```bash +python install.py +# Select "A" for all services +``` + +**Deploy only ComfyUI:** +```bash +python install.py +# Select "1" when prompted +``` + +**Deploy ComfyUI + n8n:** +```bash +python install.py +# Select "1,3" when prompted +``` + +## GPU Support + +- **NVIDIA CUDA**: Automatically detected and enabled +- **CPU Fallback**: Works without GPU +- **Windows**: Supports both CUDA and CPU modes + +## Requirements + +- Docker with GPU support (for NVIDIA) +- 8GB+ RAM (16GB+ recommended with GPU) +- 50GB+ storage for models +- Open ports: 80, 443 (or custom ports) \ No newline at end of file diff --git a/claraverse-agent-deployement/docker-compose.yml b/claraverse-agent-deployement/docker-compose.yml new file mode 100644 index 00000000..f4f49eee --- /dev/null +++ b/claraverse-agent-deployement/docker-compose.yml @@ -0,0 +1,224 @@ +services: + # ComfyUI - AI Image Generation + clara_comfyui: + image: clara17verse/clara-comfyui:with-custom-nodes + container_name: clara_comfyui + ports: + - "8188:8188" + environment: + # GPU Environment (will be enabled if NVIDIA GPU detected) + - NVIDIA_VISIBLE_DEVICES=all + - CUDA_VISIBLE_DEVICES=0 + - PYTORCH_CUDA_ALLOC_CONF=max_split_size_mb:2048,expandable_segments:True + - CUDA_LAUNCH_BLOCKING=0 + - TORCH_CUDNN_V8_API_ENABLED=1 + - CUDA_MODULE_LOADING=LAZY + - XFORMERS_MORE_DETAILS=0 + - COMFYUI_FORCE_FP16=1 + - COMFYUI_DISABLE_XFORMERS_WARNING=1 + - COMFYUI_HIGHVRAM=1 + - COMFYUI_DISABLE_MODEL_OFFLOAD=1 + - COMFYUI_VRAM_USAGE=gpu-only + volumes: + - comfyui_models:/app/ComfyUI/models + - comfyui_output:/app/ComfyUI/output + - comfyui_input:/app/ComfyUI/input + - comfyui_custom_nodes:/app/ComfyUI/custom_nodes + - comfyui_temp:/app/ComfyUI/temp + - comfyui_user:/app/ComfyUI/user + networks: + - clara_network + runtime: nvidia # Will be removed if no GPU detected + restart: unless-stopped + healthcheck: + test: ["CMD", "curl", "-f", "http://localhost:8188/"] + interval: 30s + timeout: 10s + retries: 3 + + # Python Backend - Core AI Processing + clara_python: + image: clara17verse/clara-backend:latest + container_name: clara_python + ports: + - "5001:5000" + environment: + - PYTHONUNBUFFERED=1 + - TOKENIZERS_PARALLELISM=false + - OMP_NUM_THREADS=1 + # GPU Environment (will be enabled if NVIDIA GPU detected) + - NVIDIA_VISIBLE_DEVICES=all + - CUDA_VISIBLE_DEVICES=0 + - PYTORCH_CUDA_ALLOC_CONF=max_split_size_mb:512,expandable_segments:True + - CUDA_LAUNCH_BLOCKING=0 + - TORCH_CUDNN_V8_API_ENABLED=1 + - CUDA_MODULE_LOADING=LAZY + - CUDA_CACHE_DISABLE=0 + - WHISPER_CUDA=1 + - FASTER_WHISPER_DEVICE=cuda + volumes: + - python_data:/home/clara + - python_models:/app/models + networks: + - clara_network + runtime: nvidia # Will be removed if no GPU detected + restart: unless-stopped + healthcheck: + test: ["CMD", "curl", "-f", "http://localhost:5000/health"] + interval: 30s + timeout: 10s + retries: 3 + + # n8n - Workflow Automation + clara_n8n: + image: n8nio/n8n:latest + container_name: clara_n8n + ports: + - "5678:5678" + environment: + - N8N_BASIC_AUTH_ACTIVE=true + - N8N_BASIC_AUTH_USER=admin + - N8N_BASIC_AUTH_PASSWORD=clara123 # Change this! + - N8N_HOST=0.0.0.0 + - N8N_PORT=5678 + - N8N_PROTOCOL=http + - WEBHOOK_URL=http://localhost:5678/ + volumes: + - n8n_data:/home/node/.n8n + networks: + - clara_network + restart: unless-stopped + healthcheck: + test: ["CMD", "curl", "-f", "http://localhost:5678/healthz"] + interval: 30s + timeout: 10s + retries: 3 + + # PostgreSQL - Database for Agent Runner + clara_postgres: + image: postgres:16-alpine + container_name: clara_postgres + ports: + - "5432:5432" + environment: + - POSTGRES_DB=clara_workflows + - POSTGRES_USER=clara + - POSTGRES_PASSWORD=clara123 # Change this! + - PGDATA=/var/lib/postgresql/data/pgdata + volumes: + - postgres_data:/var/lib/postgresql/data + - ../sdk/migrations:/docker-entrypoint-initdb.d + networks: + - clara_network + restart: unless-stopped + healthcheck: + test: ["CMD-SHELL", "pg_isready -U clara"] + interval: 10s + timeout: 5s + retries: 5 + + # Agent Runner - Workflow Deployment Service + clara_agent_runner: + build: + context: ../sdk + dockerfile: Dockerfile + container_name: clara_agent_runner + ports: + - "3000:3000" + environment: + # Server Configuration + - PORT=3000 + - NODE_ENV=production + - HOST=0.0.0.0 + - BASE_URL=http://localhost:3000 + + # Database Configuration + - DATABASE_URL=postgresql://clara:clara123@clara_postgres:5432/clara_workflows + + # External Services (connecting to other Clara services) + - COMFYUI_URL=http://clara_comfyui:8188 + - PYTHON_BACKEND_URL=http://clara_python:5000 + - CLARA_ASSISTANT_URL=http://localhost:8069 + - OLLAMA_URL=http://localhost:11434 + + # API Keys (set your actual keys here or in .env file) + - OPENAI_API_KEY=${OPENAI_API_KEY:-} + - ANTHROPIC_API_KEY=${ANTHROPIC_API_KEY:-} + - OPENROUTER_API_KEY=${OPENROUTER_API_KEY:-} + + # Execution Limits + - MAX_EXECUTION_TIME=300000 + - MAX_CONCURRENT_EXECUTIONS=10 + - MAX_INPUT_SIZE=10485760 + + # Rate Limiting + - RATE_LIMIT_ENABLED=true + - RATE_LIMIT_WINDOW_MS=3600000 + - RATE_LIMIT_MAX_REQUESTS=100 + + # Security + - CORS_ORIGINS=* + - TRUST_PROXY=false + + # Logging + - LOG_LEVEL=info + - ENABLE_EXECUTION_LOGGING=true + volumes: + - agent_runner_logs:/app/logs + networks: + - clara_network + depends_on: + clara_postgres: + condition: service_healthy + clara_comfyui: + condition: service_healthy + clara_python: + condition: service_healthy + restart: unless-stopped + healthcheck: + test: ["CMD", "curl", "-f", "http://localhost:3000/health"] + interval: 30s + timeout: 10s + retries: 3 + start_period: 40s + +# Docker Networks +networks: + clara_network: + driver: bridge + ipam: + config: + - subnet: 172.25.0.0/16 + +# Docker Volumes for Data Persistence +volumes: + # ComfyUI Data + comfyui_models: + driver: local + comfyui_output: + driver: local + comfyui_input: + driver: local + comfyui_custom_nodes: + driver: local + comfyui_temp: + driver: local + comfyui_user: + driver: local + + # Python Backend Data + python_data: + driver: local + python_models: + driver: local + + # n8n Data + n8n_data: + driver: local + # PostgreSQL Data + postgres_data: + driver: local + + # Agent Runner Data + agent_runner_logs: + driver: local diff --git a/claraverse-agent-deployement/install.py b/claraverse-agent-deployement/install.py new file mode 100644 index 00000000..c1479813 --- /dev/null +++ b/claraverse-agent-deployement/install.py @@ -0,0 +1,461 @@ +#!/usr/bin/env python3 +""" +ClaraVerse Remote Server Installer +A beautiful, cross-platform installer for ClaraVerse AI services +No external dependencies required - uses only Python standard library +""" + +import os +import sys +import subprocess +import json +import time +import platform +import shutil +from typing import List, Dict, Optional + +class Colors: + """ANSI color codes for beautiful terminal output""" + RED = '\033[31m' + GREEN = '\033[32m' + YELLOW = '\033[33m' + BLUE = '\033[34m' + MAGENTA = '\033[35m' + CYAN = '\033[36m' + WHITE = '\033[37m' + BOLD = '\033[1m' + RESET = '\033[0m' + + @staticmethod + def disable_on_windows(): + """Enable ANSI colors on Windows""" + if platform.system() == "Windows": + try: + # Enable ANSI escape sequences on Windows 10+ + import ctypes + kernel32 = ctypes.windll.kernel32 + kernel32.SetConsoleMode(kernel32.GetStdHandle(-11), 7) + except: + pass + +class ClaraVerseInstaller: + def __init__(self): + Colors.disable_on_windows() + self.services = { + 'clara_comfyui': { + 'name': 'ComfyUI', + 'description': 'AI Image Generation Powerhouse', + 'port': 8188, + 'health_path': '/', + 'selected': False + }, + 'clara_python': { + 'name': 'Python Backend', + 'description': 'Advanced AI Processing APIs', + 'port': 5001, + 'health_path': '/health', + 'selected': False + }, + 'clara_n8n': { + 'name': 'n8n Workflows', + 'description': 'Automation Made Simple', + 'port': 5678, + 'health_path': '/healthz', + 'selected': False + } + } + self.docker_available = False + self.gpu_available = False + self.os_type = platform.system().lower() + + def clear_screen(self): + """Clear the terminal screen""" + os.system('cls' if self.os_type == 'windows' else 'clear') + + def print_banner(self): + """Display beautiful ASCII banner""" + banner = f""" +{Colors.CYAN} + ##### # ##### ###### ##### # # ####### ###### ##### ####### + # # # # # # # # # # # # # # # # # + # # ####### ###### ####### # # ##### ###### ##### ##### + # # # # # # # # # # # # # # # + # # # # # # # # # # # # # # # # # + ##### ####### # # # # # # ### ####### # # ##### ####### +{Colors.RESET} + +{Colors.BOLD}{Colors.MAGENTA} REMOTE SERVER DEPLOYMENT{Colors.RESET} +{Colors.CYAN} The AI-Powered Creative Suite{Colors.RESET} + +{Colors.YELLOW} +==================================================================+{Colors.RESET} +{Colors.YELLOW} : :{Colors.RESET} +{Colors.YELLOW} : {Colors.WHITE}ComfyUI - AI Image Generation Powerhouse{Colors.YELLOW} :{Colors.RESET} +{Colors.YELLOW} : {Colors.WHITE}Python Backend - Advanced AI Processing APIs{Colors.YELLOW} :{Colors.RESET} +{Colors.YELLOW} : {Colors.WHITE}n8n Workflows - Automation Made Simple{Colors.YELLOW} :{Colors.RESET} +{Colors.YELLOW} : :{Colors.RESET} +{Colors.YELLOW} +==================================================================+{Colors.RESET} + +{Colors.GREEN} Auto-GPU Detection - Secure Deployment - Production Ready{Colors.RESET} +""" + print(banner) + time.sleep(2) + + def print_section_header(self, title: str): + """Print a beautiful section header""" + print(f"\n{Colors.BLUE}{'='*80}{Colors.RESET}") + print(f"{Colors.BOLD}{Colors.WHITE} [{title}] {Colors.RESET}") + print(f"{Colors.BLUE}{'='*80}{Colors.RESET}\n") + + def print_progress(self, step: int, total: int, message: str): + """Print a progress indicator""" + percentage = (step / total) * 100 + filled = int(50 * step // total) + bar = '█' * filled + '-' * (50 - filled) + print(f"\r{Colors.CYAN}[{bar}] {percentage:.1f}% {message}{Colors.RESET}", end='', flush=True) + + def log_info(self, message: str): + """Log an info message""" + print(f"{Colors.GREEN}[✓] {message}{Colors.RESET}") + + def log_warning(self, message: str): + """Log a warning message""" + print(f"{Colors.YELLOW}[!] {message}{Colors.RESET}") + + def log_error(self, message: str): + """Log an error message""" + print(f"{Colors.RED}[✗] {message}{Colors.RESET}") + + def run_command(self, command: List[str], capture_output: bool = True) -> tuple: + """Run a shell command and return (success, output)""" + try: + if capture_output: + result = subprocess.run(command, capture_output=True, text=True, timeout=30) + return result.returncode == 0, result.stdout.strip() + else: + result = subprocess.run(command, timeout=60) + return result.returncode == 0, "" + except subprocess.TimeoutExpired: + return False, "Command timed out" + except Exception as e: + return False, str(e) + + def check_docker(self) -> bool: + """Check if Docker is installed and running""" + print(f"{Colors.YELLOW}[*] Checking Docker installation...{Colors.RESET}") + + # Check if docker command exists + success, output = self.run_command(['docker', '--version']) + if not success: + self.log_error("Docker is not installed or not in PATH") + self.print_docker_install_instructions() + return False + + # Check if Docker daemon is running + success, output = self.run_command(['docker', 'ps']) + if not success: + self.log_error("Docker daemon is not running") + self.log_warning("Please start Docker Desktop and try again") + return False + + self.log_info(f"Docker is available: {output}") + return True + + def print_docker_install_instructions(self): + """Print Docker installation instructions""" + print(f"\n{Colors.YELLOW}Please install Docker:{Colors.RESET}") + if self.os_type == 'windows': + print(f"{Colors.WHITE} 1. Download Docker Desktop for Windows{Colors.RESET}") + print(f"{Colors.WHITE} 2. https://docs.docker.com/desktop/windows/install/{Colors.RESET}") + elif self.os_type == 'darwin': + print(f"{Colors.WHITE} 1. Download Docker Desktop for Mac{Colors.RESET}") + print(f"{Colors.WHITE} 2. https://docs.docker.com/desktop/mac/install/{Colors.RESET}") + else: + print(f"{Colors.WHITE} 1. Install Docker Engine: sudo apt install docker.io{Colors.RESET}") + print(f"{Colors.WHITE} 2. Start Docker: sudo systemctl start docker{Colors.RESET}") + + def check_gpu(self) -> bool: + """Check if NVIDIA GPU is available""" + print(f"{Colors.YELLOW}[*] Checking GPU availability...{Colors.RESET}") + + # Check for nvidia-smi + success, output = self.run_command(['nvidia-smi']) + if success and 'NVIDIA' in output: + self.log_info("NVIDIA GPU detected - CUDA acceleration will be enabled") + return True + else: + self.log_warning("No NVIDIA GPU detected - using CPU mode") + return False + + def select_services(self): + """Interactive service selection""" + self.print_section_header("SERVICE SELECTION") + + print(f"{Colors.WHITE}Please select which services you want to deploy:{Colors.RESET}\n") + + for i, (service_id, service) in enumerate(self.services.items(), 1): + print(f"{Colors.CYAN} [{i}] {service['name']:<15} {Colors.WHITE}- {service['description']} (Port {service['port']}){Colors.RESET}") + print(f"{Colors.CYAN} [A] All Services {Colors.WHITE}- Deploy everything (Recommended){Colors.RESET}\n") + + while True: + choice = input(f"{Colors.YELLOW}Select services (e.g., 1,3 or A for all): {Colors.RESET}").strip() + + if choice.upper() == 'A': + for service in self.services.values(): + service['selected'] = True + self.log_info("All services selected for deployment") + break + elif choice: + # Reset selections + for service in self.services.values(): + service['selected'] = False + + # Parse selections + service_list = list(self.services.keys()) + for char in choice.replace(',', '').replace(' ', ''): + if char.isdigit(): + idx = int(char) - 1 + if 0 <= idx < len(service_list): + self.services[service_list[idx]]['selected'] = True + + # Check if any service selected + if any(service['selected'] for service in self.services.values()): + break + else: + self.log_error("No valid services selected. Please try again.") + else: + self.log_error("Please enter a selection.") + + # Show selected services + print(f"\n{Colors.WHITE}Selected services:{Colors.RESET}") + for service in self.services.values(): + if service['selected']: + print(f"{Colors.GREEN} ✓ {service['name']}{Colors.RESET}") + print() + time.sleep(2) + + def deploy_services(self): + """Deploy selected services using Docker Compose""" + self.print_section_header("DEPLOYMENT") + + # Build service list + selected_services = [service_id for service_id, service in self.services.items() if service['selected']] + + if not selected_services: + self.log_error("No services selected for deployment") + return False + + print(f"{Colors.YELLOW}[*] Deploying selected services: {', '.join(selected_services)}{Colors.RESET}") + + # Pull images + print(f"{Colors.YELLOW}[*] Pulling Docker images (this may take several minutes)...{Colors.RESET}") + pull_cmd = ['docker', 'compose', 'pull'] + selected_services + success, output = self.run_command(pull_cmd, capture_output=False) + + if not success: + self.log_error("Failed to pull Docker images") + return False + + # Start services + print(f"\n{Colors.YELLOW}[*] Starting services...{Colors.RESET}") + up_cmd = ['docker', 'compose', 'up', '-d'] + selected_services + success, output = self.run_command(up_cmd, capture_output=False) + + if not success: + self.log_error("Failed to start services") + return False + + self.log_info("Services started successfully") + return True + + def check_service_health(self, port: int, path: str) -> bool: + """Check if a service is healthy using HTTP request""" + try: + import urllib.request + import urllib.error + + url = f'http://localhost:{port}{path}' + request = urllib.request.Request(url) + response = urllib.request.urlopen(request, timeout=5) + return response.getcode() == 200 + except: + return False + + def wait_for_services(self): + """Wait for services to become healthy""" + self.print_section_header("HEALTH CHECK") + + print(f"{Colors.YELLOW}[*] Waiting for services to become healthy...{Colors.RESET}") + max_attempts = 24 + + for attempt in range(max_attempts): + all_healthy = True + + for service_id, service in self.services.items(): + if not service['selected']: + continue + + port = service['port'] + health_path = service['health_path'] + + # Check service health using pure Python HTTP request + if not self.check_service_health(port, health_path): + all_healthy = False + break + + if all_healthy: + print(f"\n{Colors.GREEN}[✓] All selected services are healthy and ready!{Colors.RESET}") + return True + + self.print_progress(attempt + 1, max_attempts, f"Checking service health (attempt {attempt + 1}/{max_attempts})") + time.sleep(5) + + print(f"\n{Colors.YELLOW}[!] Services are taking longer than expected to start{Colors.RESET}") + self.log_info("You can check status with: docker compose ps") + return True + + def show_success_message(self): + """Display final success message""" + self.clear_screen() + + print(f""" +{Colors.GREEN} + ###### ####### ###### # ####### # # ####### ###### + # # # # # # # # # # # # # + # # ##### ###### # # # # # ##### # # + # # # # # # # # # # # + # # # # # # # # # # # + ###### ####### # ####### ####### # ####### ###### +{Colors.RESET} + +{Colors.BOLD}{Colors.MAGENTA} +=================================================================+{Colors.RESET} +{Colors.BOLD}{Colors.MAGENTA} : :{Colors.RESET} +{Colors.BOLD}{Colors.MAGENTA} : CLARAVERSE DEPLOYMENT SUCCESSFUL! :{Colors.RESET} +{Colors.BOLD}{Colors.MAGENTA} : :{Colors.RESET} +{Colors.BOLD}{Colors.MAGENTA} +=================================================================+{Colors.RESET} +""") + + print(f"{Colors.CYAN}Your selected ClaraVerse services are now running:{Colors.RESET}\n") + + # Get local IP + try: + import socket + hostname = socket.gethostname() + local_ip = socket.gethostbyname(hostname) + except: + local_ip = 'localhost' + + print(f"{Colors.WHITE} +-----------------------------------------------------------------+{Colors.RESET}") + print(f"{Colors.WHITE} : :{Colors.RESET}") + + for service in self.services.values(): + if service['selected']: + service_name = service['name'] + port = service['port'] + print(f"{Colors.WHITE} : {Colors.BOLD}{Colors.BLUE}{service_name:<13}{Colors.RESET} {Colors.YELLOW}http://{local_ip}:{port}{Colors.RESET}") + + print(f"{Colors.WHITE} : :{Colors.RESET}") + print(f"{Colors.WHITE} +-----------------------------------------------------------------+{Colors.RESET}\n") + + # Management commands + print(f"{Colors.GREEN}Quick Management Commands:{Colors.RESET}") + print(f"{Colors.CYAN} Check Status: {Colors.WHITE}docker compose ps{Colors.RESET}") + print(f"{Colors.CYAN} View Logs: {Colors.WHITE}docker compose logs -f{Colors.RESET}") + print(f"{Colors.CYAN} Restart: {Colors.WHITE}docker compose restart{Colors.RESET}") + print(f"{Colors.CYAN} Stop: {Colors.WHITE}docker compose down{Colors.RESET}\n") + + # System info + print(f"{Colors.BOLD}{Colors.GREEN}System Specifications:{Colors.RESET}") + if self.gpu_available: + print(f"{Colors.WHITE} • GPU: NVIDIA GPU with CUDA support{Colors.RESET}") + else: + print(f"{Colors.WHITE} • Mode: CPU-only deployment{Colors.RESET}") + + print(f"{Colors.WHITE} • Services: {', '.join([s['name'] for s in self.services.values() if s['selected']])}{Colors.RESET}") + print(f"{Colors.WHITE} • Platform: {platform.system()} {platform.release()}{Colors.RESET}\n") + + print(f"{Colors.BLUE}{'='*80}{Colors.RESET}") + print(f"{Colors.BOLD}{Colors.WHITE}Thank you for using ClaraVerse! Happy creating!{Colors.RESET}") + print(f"{Colors.BLUE}{'='*80}{Colors.RESET}\n") + + # Open browser to first service + first_service = next((s for s in self.services.values() if s['selected']), None) + if first_service: + try: + import webbrowser + url = f"http://{local_ip}:{first_service['port']}" + input(f"{Colors.YELLOW}Press Enter to open {first_service['name']} in your browser...{Colors.RESET}") + webbrowser.open(url) + self.log_info(f"Browser opened to {first_service['name']}") + except: + pass + + def run(self): + """Main installation flow""" + try: + self.clear_screen() + self.print_banner() + + # System checks + self.print_section_header("SYSTEM VERIFICATION") + + # Check Docker + self.docker_available = self.check_docker() + if not self.docker_available: + return False + + # Check GPU + self.gpu_available = self.check_gpu() + + # Service selection + self.select_services() + + # Deploy services + if not self.deploy_services(): + return False + + # Wait for health + self.wait_for_services() + + # Show success + self.show_success_message() + + return True + + except KeyboardInterrupt: + print(f"\n{Colors.YELLOW}Installation cancelled by user{Colors.RESET}") + return False + except Exception as e: + self.log_error(f"Unexpected error: {e}") + return False + +def main(): + """Main entry point""" + if len(sys.argv) > 1 and sys.argv[1] in ['-h', '--help']: + print(""" +ClaraVerse Remote Server Installer + +A beautiful, cross-platform installer for ClaraVerse AI services. +This installer will help you deploy ComfyUI, Python Backend, and n8n services. + +Usage: + python install.py # Interactive installation + python install.py --help # Show this help + +Requirements: + - Python 3.6+ + - Docker or Docker Desktop + - Internet connection + +Services: + - ComfyUI: AI Image Generation (Port 8188) + - Python Backend: AI Processing APIs (Port 5001) + - n8n: Workflow Automation (Port 5678) + """) + return + + installer = ClaraVerseInstaller() + success = installer.run() + sys.exit(0 if success else 1) + +if __name__ == "__main__": + main() \ No newline at end of file diff --git a/claraverse-agent-deployement/nginx/nginx.conf b/claraverse-agent-deployement/nginx/nginx.conf new file mode 100644 index 00000000..d58effbc --- /dev/null +++ b/claraverse-agent-deployement/nginx/nginx.conf @@ -0,0 +1,469 @@ +events {events {events { + + worker_connections 1024; + +} worker_connections 1024; worker_connections 1024; + + + +http {}} + + include /etc/nginx/mime.types; + + default_type application/octet-stream; + + + + sendfile on;http {http { + + tcp_nopush on; + + tcp_nodelay on; include /etc/nginx/mime.types; upstream comfyui { + + keepalive_timeout 65; + + default_type application/octet-stream; server clara_comfyui:8188; + + # WebSocket upgrade headers + + map $http_upgrade $connection_upgrade { } + + default upgrade; + + '' close; # Log format + + } + + log_format main '$remote_addr - $remote_user [$time_local] "$request" ' upstream python_backend { + + # Upstream definitions + + upstream comfyui_backend { '$status $body_bytes_sent "$http_referer" ' server clara_python:5000; + + server clara_comfyui:8188; + + } '"$http_user_agent" "$http_x_forwarded_for"'; } + + + + upstream n8n_backend { + + server clara_n8n:5678; + + } access_log /var/log/nginx/access.log main; upstream n8n { + + + + upstream python_backend { error_log /var/log/nginx/error.log warn; server clara_n8n:5678; + + server clara_python:8000; + + } } + + + + server { sendfile on; + + listen 80; + + server_name _; tcp_nopush on; # WebSocket support + + + + # Increase upload limits for ComfyUI tcp_nodelay on; map $http_upgrade $connection_upgrade { + + client_max_body_size 100M; + + keepalive_timeout 65; default upgrade; + + # Health check endpoint + + location /health { types_hash_max_size 2048; '' close; + + add_header Content-Type "application/json"; + + return 200 '{"status":"healthy","services":["comfyui","n8n","python-backend"]}'; } + + } + + # Gzip Settings + + # ComfyUI service + + location /comfyui/ { gzip on; server { + + proxy_pass http://comfyui_backend/; + + proxy_http_version 1.1; gzip_vary on; listen 80; + + proxy_set_header Upgrade $http_upgrade; + + proxy_set_header Connection $connection_upgrade; gzip_proxied any; server_name _; + + proxy_set_header Host $http_host; + + proxy_set_header X-Real-IP $remote_addr; gzip_comp_level 6; + + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + + proxy_set_header X-Forwarded-Proto $scheme; gzip_types # Increase client body size for file uploads + + proxy_cache_bypass $http_upgrade; + + proxy_read_timeout 86400; text/plain client_max_body_size 100M; + + } + + text/css + + # n8n service + + location /n8n/ { text/xml # Timeout settings + + proxy_pass http://n8n_backend/; + + proxy_http_version 1.1; text/javascript proxy_read_timeout 300s; + + proxy_set_header Upgrade $http_upgrade; + + proxy_set_header Connection $connection_upgrade; application/json proxy_connect_timeout 75s; + + proxy_set_header Host $http_host; + + proxy_set_header X-Real-IP $remote_addr; application/javascript proxy_send_timeout 300s; + + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + + proxy_set_header X-Forwarded-Proto $scheme; application/xml+rss + + proxy_cache_bypass $http_upgrade; + + } application/atom+xml # ComfyUI - AI Image Generation (with WebSocket support) + + + + # Python backend API image/svg+xml; location /comfyui/ { + + location /api/ { + + proxy_pass http://python_backend/; proxy_pass http://comfyui/; + + proxy_set_header Host $http_host; + + proxy_set_header X-Real-IP $remote_addr; # Upstream definitions proxy_set_header Host $host; + + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + + proxy_set_header X-Forwarded-Proto $scheme; upstream comfyui_backend { proxy_set_header X-Real-IP $remote_addr; + + } + + server clara_comfyui:8188; proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + + # Root - ClaraVerse Server Dashboard + + location / { } proxy_set_header X-Forwarded-Proto $scheme; + + root /usr/share/nginx/html; + + index dashboard.html; + + try_files $uri $uri/ /dashboard.html; + + } upstream n8n_backend { # WebSocket support for ComfyUI + + } + +} server clara_n8n:5678; proxy_http_version 1.1; + + } proxy_set_header Upgrade $http_upgrade; + + proxy_set_header Connection $connection_upgrade; + + upstream python_backend { proxy_cache_bypass $http_upgrade; + + server clara_python:8000; } + + } + + # Python Backend - Core AI APIs + + server { location /api/ { + + listen 80; proxy_pass http://python_backend/; + + server_name _; proxy_set_header Host $host; + + proxy_set_header X-Real-IP $remote_addr; + + # Increase upload limits for ComfyUI proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + + client_max_body_size 100M; proxy_set_header X-Forwarded-Proto $scheme; + + } + + # WebSocket upgrade headers + + map $http_upgrade $connection_upgrade { # n8n - Workflow Automation + + default upgrade; location /n8n/ { + + '' close; proxy_pass http://n8n/; + + } proxy_set_header Host $host; + + proxy_set_header X-Real-IP $remote_addr; + + # Health check endpoint proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + + location /health { proxy_set_header X-Forwarded-Proto $scheme; + + add_header Content-Type "application/json"; + + return 200 '{"status":"healthy","services":["comfyui","n8n","python-backend"],"timestamp":"$time_iso8601"}'; # WebSocket support for n8n + + } proxy_http_version 1.1; + + proxy_set_header Upgrade $http_upgrade; + + # ComfyUI service proxy_set_header Connection $connection_upgrade; + + location /comfyui/ { proxy_cache_bypass $http_upgrade; + + proxy_pass http://comfyui_backend/; } + + proxy_http_version 1.1; + + proxy_set_header Upgrade $http_upgrade; # Health Check Endpoint + + proxy_set_header Connection $connection_upgrade; location /health { + + proxy_set_header Host $http_host; add_header Content-Type application/json; + + proxy_set_header X-Real-IP $remote_addr; return 200 '{"status":"ok","services":{"comfyui":"http://localhost/comfyui/","python":"http://localhost/api/","n8n":"http://localhost/n8n/"}}'; + + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; } + + proxy_set_header X-Forwarded-Proto $scheme; + + proxy_cache_bypass $http_upgrade; # Root - ClaraVerse Server Dashboard + + proxy_read_timeout 86400; location / { + + } root /usr/share/nginx/html; + + index dashboard.html; + + # n8n service try_files $uri $uri/ /dashboard.html; + + location /n8n/ { } + + proxy_pass http://n8n_backend/; + + proxy_http_version 1.1; + + proxy_set_header Upgrade $http_upgrade; + + proxy_set_header Connection $connection_upgrade; + + proxy_set_header Host $http_host; + + proxy_set_header X-Real-IP $remote_addr; ClaraVerse Remote Server + + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + + +
+
+ +

ClaraVerse Remote Server

+

Your AI-Powered Creative Suite is Ready!

+
+ +
+
+ 🎨 +

ComfyUI - AI Image Generation

+

Create stunning AI-generated images with advanced workflows and custom nodes

+ Open ComfyUI → +
+ +
+ 🔧 +

n8n - Workflow Automation

+

Automate tasks and create powerful workflows with visual programming

+ Open n8n → +
+ +
+ 🐍 +

Python Backend - AI APIs

+

Core AI processing and REST APIs for advanced machine learning tasks

+ View API Docs → +
+
+ +
+ +
+
+ + +
+ + + +'; + } + } +} \ No newline at end of file diff --git a/claraverse-agent-deployement/scripts/deploy-remote.sh b/claraverse-agent-deployement/scripts/deploy-remote.sh new file mode 100644 index 00000000..81b17a4a --- /dev/null +++ b/claraverse-agent-deployement/scripts/deploy-remote.sh @@ -0,0 +1,256 @@ +#!/bin/bash + +# ClaraVerse Remote Deployment Script +# Deploys ClaraVerse server package to a remote server via SSH + +set -e + +# Colors for output +RED='\033[0;31m' +GREEN='\033[0;32m' +YELLOW='\033[1;33m' +BLUE='\033[0;34m' +NC='\033[0m' + +# Default values +REMOTE_USER="" +REMOTE_HOST="" +REMOTE_PORT="22" +SSH_KEY="" +DEPLOYMENT_PATH="/opt/claraverse" + +# Usage information +usage() { + echo "Usage: $0 -u USER -h HOST [-p PORT] [-k SSH_KEY] [-d DEPLOYMENT_PATH]" + echo "" + echo "Options:" + echo " -u USER Remote username" + echo " -h HOST Remote server hostname or IP" + echo " -p PORT SSH port (default: 22)" + echo " -k SSH_KEY Path to SSH private key (optional)" + echo " -d DEPLOYMENT_PATH Remote deployment path (default: /opt/claraverse)" + echo "" + echo "Example:" + echo " $0 -u ubuntu -h 192.168.1.100 -k ~/.ssh/id_rsa" + echo " $0 -u root -h my-server.com -p 2222" + exit 1 +} + +# Parse command line arguments +while getopts "u:h:p:k:d:?" opt; do + case $opt in + u) REMOTE_USER="$OPTARG" ;; + h) REMOTE_HOST="$OPTARG" ;; + p) REMOTE_PORT="$OPTARG" ;; + k) SSH_KEY="$OPTARG" ;; + d) DEPLOYMENT_PATH="$OPTARG" ;; + ?) usage ;; + esac +done + +# Validate required parameters +if [ -z "$REMOTE_USER" ] || [ -z "$REMOTE_HOST" ]; then + echo -e "${RED}Error: Remote user and host are required${NC}" + usage +fi + +# Build SSH command +SSH_CMD="ssh -p $REMOTE_PORT" +SCP_CMD="scp -P $REMOTE_PORT" +if [ -n "$SSH_KEY" ]; then + SSH_CMD="$SSH_CMD -i $SSH_KEY" + SCP_CMD="$SCP_CMD -i $SSH_KEY" +fi + +SSH_TARGET="$REMOTE_USER@$REMOTE_HOST" + +log() { + echo -e "${GREEN}[$(date +'%Y-%m-%d %H:%M:%S')] $1${NC}" +} + +warn() { + echo -e "${YELLOW}[$(date +'%Y-%m-%d %H:%M:%S')] WARNING: $1${NC}" +} + +error() { + echo -e "${RED}[$(date +'%Y-%m-%d %H:%M:%S')] ERROR: $1${NC}" +} + +# Test SSH connection +test_ssh_connection() { + log "Testing SSH connection to $SSH_TARGET..." + + if $SSH_CMD $SSH_TARGET "echo 'SSH connection successful'" &> /dev/null; then + log "✅ SSH connection successful" + else + error "❌ Failed to connect to $SSH_TARGET" + echo "Please check:" + echo " - Server is running and accessible" + echo " - SSH credentials are correct" + echo " - SSH key permissions (chmod 600 if using key)" + exit 1 + fi +} + +# Create deployment directory on remote server +create_remote_directory() { + log "Creating deployment directory: $DEPLOYMENT_PATH" + + $SSH_CMD $SSH_TARGET "sudo mkdir -p $DEPLOYMENT_PATH && sudo chown $REMOTE_USER:$REMOTE_USER $DEPLOYMENT_PATH" + + if [ $? -eq 0 ]; then + log "✅ Deployment directory created" + else + error "❌ Failed to create deployment directory" + exit 1 + fi +} + +# Copy ClaraVerse server package to remote server +copy_server_package() { + log "Copying ClaraVerse server package..." + + # Create a temporary archive of the server package + local temp_archive="/tmp/claraverse-server-$(date +%s).tar.gz" + + # Get the directory containing this script + local script_dir="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)" + local server_dir="$(dirname "$script_dir")" + + log "Creating archive from: $server_dir" + tar -czf "$temp_archive" -C "$(dirname "$server_dir")" "$(basename "$server_dir")" --exclude="*.git*" --exclude="node_modules" --exclude="*.log" + + # Copy archive to remote server + $SCP_CMD "$temp_archive" "$SSH_TARGET:$temp_archive" + + # Extract on remote server + $SSH_CMD $SSH_TARGET "cd $DEPLOYMENT_PATH && tar -xzf $temp_archive --strip-components=1 && rm $temp_archive" + + # Clean up local archive + rm "$temp_archive" + + if [ $? -eq 0 ]; then + log "✅ Server package copied successfully" + else + error "❌ Failed to copy server package" + exit 1 + fi +} + +# Install Docker and dependencies on remote server +install_dependencies() { + log "Installing Docker and dependencies on remote server..." + + $SSH_CMD $SSH_TARGET "cd $DEPLOYMENT_PATH && chmod +x scripts/install.sh && sudo scripts/install.sh" + + if [ $? -eq 0 ]; then + log "✅ Dependencies installed successfully" + else + error "❌ Failed to install dependencies" + exit 1 + fi +} + +# Start ClaraVerse services +start_services() { + log "Starting ClaraVerse services..." + + $SSH_CMD $SSH_TARGET "cd $DEPLOYMENT_PATH && sudo docker compose up -d" + + if [ $? -eq 0 ]; then + log "✅ Services started successfully" + else + error "❌ Failed to start services" + exit 1 + fi +} + +# Wait for services to be ready +wait_for_services() { + log "Waiting for services to become ready..." + + local max_attempts=30 + local attempt=0 + + while [ $attempt -lt $max_attempts ]; do + if $SSH_CMD $SSH_TARGET "curl -f http://localhost/health" &> /dev/null; then + log "✅ All services are ready!" + return 0 + fi + + echo -n "." + sleep 10 + attempt=$((attempt + 1)) + done + + warn "Services are taking longer than expected to start" + return 1 +} + +# Get remote server info +get_server_info() { + log "Getting server information..." + + # Get server IP + SERVER_IP=$($SSH_CMD $SSH_TARGET "curl -s https://api.ipify.org" 2>/dev/null || echo "$REMOTE_HOST") + + # Get GPU info + GPU_INFO=$($SSH_CMD $SSH_TARGET "nvidia-smi --query-gpu=name --format=csv,noheader,nounits" 2>/dev/null || echo "No GPU detected") + + # Get Docker info + DOCKER_VERSION=$($SSH_CMD $SSH_TARGET "docker --version" 2>/dev/null || echo "Unknown") + + # Get container status + CONTAINER_STATUS=$($SSH_CMD $SSH_TARGET "cd $DEPLOYMENT_PATH && docker compose ps --format table" 2>/dev/null || echo "Unable to get status") +} + +# Display deployment results +show_deployment_results() { + echo "" + echo -e "${GREEN}🎉 ClaraVerse Remote Deployment Complete!${NC}" + echo "==============================================" + echo "" + echo -e "${BLUE}Server Information:${NC}" + echo " 🖥️ Server IP: $SERVER_IP" + echo " 🎮 GPU: $GPU_INFO" + echo " 🐳 Docker: $DOCKER_VERSION" + echo "" + echo -e "${BLUE}Access your ClaraVerse server at:${NC}" + echo " 🌐 Dashboard: http://$SERVER_IP" + echo " 🎨 ComfyUI: http://$SERVER_IP/comfyui/" + echo " 🔧 n8n: http://$SERVER_IP/n8n/" + echo " 🐍 API Docs: http://$SERVER_IP/api/docs" + echo "" + echo -e "${YELLOW}Remote Management Commands:${NC}" + echo " 📊 Check status: ssh $SSH_TARGET 'cd $DEPLOYMENT_PATH && docker compose ps'" + echo " 📝 View logs: ssh $SSH_TARGET 'cd $DEPLOYMENT_PATH && docker compose logs -f'" + echo " 🔄 Restart: ssh $SSH_TARGET 'cd $DEPLOYMENT_PATH && docker compose restart'" + echo " ⏹️ Stop: ssh $SSH_TARGET 'cd $DEPLOYMENT_PATH && docker compose down'" + echo "" + echo -e "${GREEN}Container Status:${NC}" + echo "$CONTAINER_STATUS" + echo "" +} + +# Main deployment flow +main() { + echo -e "${BLUE}🚀 ClaraVerse Remote Deployment${NC}" + echo "=================================" + echo "Target: $SSH_TARGET:$REMOTE_PORT" + echo "Path: $DEPLOYMENT_PATH" + echo "" + + test_ssh_connection + create_remote_directory + copy_server_package + install_dependencies + start_services + wait_for_services + get_server_info + show_deployment_results + + log "Deployment completed successfully!" +} + +# Run main function +main "$@" \ No newline at end of file diff --git a/claraverse-agent-deployement/scripts/install-simple.bat b/claraverse-agent-deployement/scripts/install-simple.bat new file mode 100644 index 00000000..8664e585 --- /dev/null +++ b/claraverse-agent-deployement/scripts/install-simple.bat @@ -0,0 +1,258 @@ +@echo off +cls +echo. +echo ██████╗██╗ █████╗ ██████╗ █████╗ ██╗ ██╗███████╗██████╗ ███████╗███████╗ +echo ██╔════╝██║ ██╔══██╗██╔══██╗██╔══██╗██║ ██║██╔════╝██╔══██╗██╔════╝██╔════╝ +echo ██║ ██║ ███████║██████╔╝███████║██║ ██║█████╗ ██████╔╝███████╗█████╗ +echo ██║ ██║ ██╔══██║██╔══██╗██╔══██║╚██╗ ██╔╝██╔══╝ ██╔══██╗╚════██║██╔══╝ +echo ╚██████╗███████╗██║ ██║██║ ██║██║ ██║ ╚████╔╝ ███████╗██║ ██║███████║███████╗ +echo ╚═════╝╚══════╝╚═╝ ╚═╝╚═╝ ╚═╝╚═╝ ╚═╝ ╚═══╝ ╚══════╝╚═╝ ╚═╝╚══════╝╚══════╝ +echo. +echo 🚀 REMOTE SERVER DEPLOYMENT 🚀 +echo The AI-Powered Creative Suite +echo. +echo ╔══════════════════════════════════════════════════════════════════╗ +echo ║ 🎨 ComfyUI - AI Image Generation Powerhouse ║ +echo ║ 🐍 Python Backend - Advanced AI Processing APIs ║ +echo ║ 🔧 n8n Workflows - Automation Made Simple ║ +echo ║ 🌐 Nginx Proxy - Professional Reverse Proxy ║ +echo ╚══════════════════════════════════════════════════════════════════╝ +echo. +echo ⚡ Auto-GPU Detection 🔒 Secure Deployment 📈 Production Ready +echo. +echo ═══════════════════════════════════════════════════════════════════════════════ +echo 🔍 SYSTEM VERIFICATION +echo ═══════════════════════════════════════════════════════════════════════════════ +echo. + +REM Step 1: Check Docker Desktop +echo [1/8] 🔍 Checking Docker Desktop installation... +docker --version >nul 2>&1 +if %errorLevel% neq 0 ( + echo. + echo ❌ ERROR: Docker Desktop is not installed or not in PATH + echo. + echo Please install Docker Desktop for Windows: + echo 1. Download from: https://docs.docker.com/desktop/windows/install/ + echo 2. Install Docker Desktop + echo 3. Start Docker Desktop and wait for it to be ready + echo 4. Run this script again + echo. + pause + exit /b 1 +) + +for /f "tokens=3" %%v in ('docker --version') do set "DOCKER_VERSION=%%v" +echo ✅ Docker Desktop %DOCKER_VERSION% detected + +REM Step 2: Check Docker daemon +echo [2/8] 🔍 Verifying Docker daemon status... +docker info >nul 2>&1 +if %errorLevel% neq 0 ( + echo ❌ Docker daemon not responding + echo 💡 Please start Docker Desktop and wait for it to be ready + echo. + pause + exit /b 1 +) +echo ✅ Docker daemon is running + +REM Step 3: GPU Detection +echo [3/8] 🎮 Scanning for NVIDIA GPU... +nvidia-smi >nul 2>&1 +if %errorLevel% equ 0 ( + for /f "skip=8 tokens=2" %%g in ('nvidia-smi --query-gpu=name --format=csv,noheader 2^>nul') do ( + echo ✅ GPU Detected: %%g + set "GPU_AVAILABLE=true" + goto gpu_check_done + ) +) else ( + echo ⚠️ No NVIDIA GPU detected - CPU mode will be used + set "GPU_AVAILABLE=false" +) + +:gpu_check_done + +REM Step 4: Test GPU Docker support +echo [4/8] 🧪 Testing Docker GPU integration... +if "%GPU_AVAILABLE%"=="true" ( + docker run --rm --gpus all --pull=always nvidia/cuda:12.0-base-ubuntu20.04 nvidia-smi >nul 2>&1 + if %errorLevel% equ 0 ( + echo ✅ GPU Docker acceleration ready + set "GPU_DOCKER=true" + ) else ( + echo ⚠️ GPU available but Docker GPU support not working + echo 💡 Continuing with CPU mode + set "GPU_DOCKER=false" + set "GPU_AVAILABLE=false" + ) +) else ( + echo 🖥️ Configuring for CPU-only deployment + set "GPU_DOCKER=false" +) + +echo. +echo ═══════════════════════════════════════════════════════════════════════════════ +echo 🛠️ DEPLOYMENT CONFIGURATION +echo ═══════════════════════════════════════════════════════════════════════════════ +echo. + +REM Step 5: Create directories +echo [5/8] 📁 Setting up data directories... +if not exist "data" mkdir data +if not exist "data\comfyui" mkdir data\comfyui +if not exist "data\python" mkdir data\python +if not exist "data\n8n" mkdir data\n8n +if not exist "logs" mkdir logs +echo ✅ Directory structure created + +REM Step 6: Configure for GPU/CPU +echo [6/8] ⚙️ Generating optimized configuration... +if "%GPU_AVAILABLE%"=="false" ( + echo 🖥️ Creating CPU-optimized configuration... + + REM Create CPU-only version + powershell -Command "(Get-Content 'docker-compose.yml') | Where-Object { $_ -notmatch 'runtime: nvidia' -and $_ -notmatch 'NVIDIA_VISIBLE_DEVICES' -and $_ -notmatch 'CUDA_VISIBLE_DEVICES' -and $_ -notmatch 'PYTORCH_CUDA_ALLOC_CONF' -and $_ -notmatch 'CUDA_LAUNCH_BLOCKING' -and $_ -notmatch 'TORCH_CUDNN_V8_API_ENABLED' -and $_ -notmatch 'CUDA_MODULE_LOADING' -and $_ -notmatch 'CUDA_CACHE_DISABLE' -and $_ -notmatch 'WHISPER_CUDA' -and $_ -notmatch 'COMFYUI_' } | ForEach-Object { $_ -replace 'FASTER_WHISPER_DEVICE=cuda', 'FASTER_WHISPER_DEVICE=cpu' } | Set-Content 'docker-compose-active.yml'" + + echo ✅ CPU-optimized configuration ready +) else ( + echo 🚀 Creating GPU-accelerated configuration... + copy docker-compose.yml docker-compose-active.yml >nul + echo ✅ GPU-accelerated configuration ready +) + +echo. +echo ═══════════════════════════════════════════════════════════════════════════════ +echo 📥 DOWNLOADING COMPONENTS +echo ═══════════════════════════════════════════════════════════════════════════════ +echo. + +REM Step 7: Pull images +echo [7/8] 📥 Downloading ClaraVerse containers... +echo This may take several minutes for first-time setup +echo. +docker compose -f docker-compose-active.yml pull +if %errorLevel% neq 0 ( + echo ❌ Failed to download container images + echo Please check your internet connection and try again + pause + exit /b 1 +) +echo ✅ All container images downloaded successfully + +echo. +echo ═══════════════════════════════════════════════════════════════════════════════ +echo 🚀 LAUNCHING CLARAVERSE +echo ═══════════════════════════════════════════════════════════════════════════════ +echo. + +REM Step 8: Deploy services +echo [8/8] 🚀 Starting all ClaraVerse services... +docker compose -f docker-compose-active.yml up -d +if %errorLevel% neq 0 ( + echo ❌ Failed to start services + echo Check Docker Desktop and try again + pause + exit /b 1 +) +echo ✅ All services launched successfully + +REM Wait for services +echo. +echo ⏳ Waiting for services to initialize... +set /a "attempts=0" +set /a "max_attempts=24" + +:health_check_loop +set /a "attempts+=1" +echo [%attempts%/%max_attempts%] Checking service health... + +curl -f http://localhost/health >nul 2>&1 +if %errorLevel% equ 0 ( + echo ✅ All services are healthy and ready! + goto services_ready +) + +if %attempts% geq %max_attempts% ( + echo ⚠️ Services are taking longer than expected to start + echo 💡 You can check status with: docker compose -f docker-compose-active.yml ps + goto services_ready +) + +timeout /t 5 /nobreak >nul +goto health_check_loop + +:services_ready + +REM Get local IP +for /f "tokens=2 delims=:" %%a in ('ipconfig ^| findstr /c:"IPv4 Address" ^| findstr "192.168\|10\.\|172\."') do ( + set "LOCAL_IP=%%a" + set "LOCAL_IP=!LOCAL_IP: =!" + goto ip_found +) +set "LOCAL_IP=localhost" + +:ip_found + +cls +echo. +echo ██████╗ ███████╗██████╗ ██╗ ██████╗ ██╗ ██╗███████╗██████╗ +echo ██╔══██╗██╔════╝██╔══██╗██║ ██╔═══██╗╚██╗ ██╔╝██╔════╝██╔══██╗ +echo ██║ ██║█████╗ ██████╔╝██║ ██║ ██║ ╚████╔╝ █████╗ ██║ ██║ +echo ██║ ██║██╔══╝ ██╔═══╝ ██║ ██║ ██║ ╚██╔╝ ██╔══╝ ██║ ██║ +echo ██████╔╝███████╗██║ ███████╗╚██████╔╝ ██║ ███████╗██████╔╝ +echo ╚═════╝ ╚══════╝╚═╝ ╚══════╝ ╚═════╝ ╚═╝ ╚══════╝╚═════╝ +echo. +echo ╔═══════════════════════════════════════════════════════════════╗ +echo ║ ║ +echo ║ 🎉 CLARAVERSE DEPLOYMENT SUCCESSFUL! 🎉 ║ +echo ║ ║ +echo ╚═══════════════════════════════════════════════════════════════╝ +echo. +echo 🌐 Your ClaraVerse server is now running at: +echo. +echo ┌─────────────────────────────────────────────────────────────────┐ +echo │ │ +echo │ 📊 Main Dashboard: http://%LOCAL_IP% │ +echo │ 🎨 ComfyUI: http://%LOCAL_IP%/comfyui/ │ +echo │ 🔧 n8n Workflows: http://%LOCAL_IP%/n8n/ │ +echo │ 🐍 API Docs: http://%LOCAL_IP%/api/docs │ +echo │ │ +echo └─────────────────────────────────────────────────────────────────┘ +echo. +echo 💡 Quick Management Commands: +echo 📊 Check Status: docker compose -f docker-compose-active.yml ps +echo 📝 View Logs: docker compose -f docker-compose-active.yml logs -f +echo 🔄 Restart: docker compose -f docker-compose-active.yml restart +echo ⏹️ Stop: docker compose -f docker-compose-active.yml down +echo. + +if "%GPU_AVAILABLE%"=="true" ( + echo 🚀 GPU Acceleration: ENABLED ^(NVIDIA CUDA^) +) else ( + echo 🖥️ Running Mode: CPU Only ^(No GPU detected^) +) + +echo. +echo System Specifications: +echo • Docker: %DOCKER_VERSION% +if "%GPU_AVAILABLE%"=="true" ( + echo • GPU: NVIDIA GPU with CUDA support +) else ( + echo • GPU: CPU-only mode +) +echo • Services: ComfyUI, Python Backend, n8n, Nginx +echo • Architecture: Windows with Docker Desktop +echo. +echo ═══════════════════════════════════════════════════════════════════════════════ +echo Thank you for using ClaraVerse! Happy creating! 🎨✨ +echo ═══════════════════════════════════════════════════════════════════════════════ +echo. +echo Press any key to open the dashboard in your browser... +pause >nul + +REM Open browser +start http://%LOCAL_IP% + +echo Browser opened! Installation complete. +timeout /t 3 /nobreak >nul \ No newline at end of file diff --git a/claraverse-agent-deployement/scripts/install.bat b/claraverse-agent-deployement/scripts/install.bat new file mode 100644 index 00000000..92d0c3dc --- /dev/null +++ b/claraverse-agent-deployement/scripts/install.bat @@ -0,0 +1,511 @@ +@echo off +REM ClaraVerse Remote Server Installer - Windows +REM Beautiful ASCII art deployment script with full automation + +setlocal enabledelayedexpansion + +REM Enable color support +for /F %%a in ('echo prompt $E ^| cmd') do set "ESC=%%a" + +REM Color definitions +set "RED=%ESC%[31m" +set "GREEN=%ESC%[32m" +set "YELLOW=%ESC%[33m" +set "BLUE=%ESC%[34m" +set "MAGENTA=%ESC%[35m" +set "CYAN=%ESC%[36m" +set "WHITE=%ESC%[37m" +set "BOLD=%ESC%[1m"echo %WHITE% • Services: %RESET% +if "!DEPLOY_COMFYUI!"=="true" echo %WHITE% - ComfyUI%RESET% +if "!DEPLOY_PYTHON!"=="true" echo %WHITE% - Python Backend%RESET% +if "!DEPLOY_N8N!"=="true" echo %WHITE% - n8n%RESET% +echo %WHITE% • Architecture: Windows with Docker Desktop%RESET% +echo. +echo %BLUE%=================================================================================%RESET% +echo %BOLD%%WHITE%Thank you for using ClaraVerse! Happy creating!%RESET% +echo %BLUE%=================================================================================%RESET% +echo. + +REM Determine which service to open in browser +set "BROWSER_URL=" +if "!DEPLOY_COMFYUI!"=="true" ( + set "BROWSER_URL=http://!LOCAL_IP!:8188" + set "BROWSER_SERVICE=ComfyUI" +) else if "!DEPLOY_N8N!"=="true" ( + set "BROWSER_URL=http://!LOCAL_IP!:5678" + set "BROWSER_SERVICE=n8n" +) else if "!DEPLOY_PYTHON!"=="true" ( + set "BROWSER_URL=http://!LOCAL_IP!:5001" + set "BROWSER_SERVICE=Python Backend" +) + +if not "!BROWSER_URL!"=="" ( + echo %YELLOW%Press any key to open !BROWSER_SERVICE! in your browser...%RESET% + pause >nul + start !BROWSER_URL! + echo %GREEN%Browser opened! Installation complete.%RESET% +) else ( + echo %GREEN%Installation complete!%RESET% +) + +echo %WHITE%Access your services at:%RESET% +if "!DEPLOY_COMFYUI!"=="true" echo %WHITE% ComfyUI: http://!LOCAL_IP!:8188%RESET% +if "!DEPLOY_PYTHON!"=="true" echo %WHITE% Python Backend: http://!LOCAL_IP!:5001%RESET% +if "!DEPLOY_N8N!"=="true" echo %WHITE% n8n: http://!LOCAL_IP!:5678%RESET%[0m" + +REM Progress bar settings +set "PROGRESS_WIDTH=50" +set /a "step=0" +set /a "total_steps=8" + +cls +echo. +echo %CYAN% +echo ##### # ##### ###### ##### # # ####### ###### ##### ####### +echo # # # # # # # # # # # # # # # # # +echo # # ####### ###### ####### # # ##### ###### ##### ##### +echo # # # # # # # # # # # # # # # +echo # # # # # # # # # # # # # # # # # +echo ##### ####### # # # # # # ### ####### # # ##### ####### +echo %RESET% +echo. +echo %BOLD%%MAGENTA% REMOTE SERVER DEPLOYMENT%RESET% +echo %CYAN% The AI-Powered Creative Suite%RESET% +echo. +echo %YELLOW% +==================================================================+%RESET% +echo %YELLOW% : :%RESET% +echo %YELLOW% : %WHITE%ComfyUI - AI Image Generation Powerhouse%YELLOW% :%RESET% +echo %YELLOW% : %WHITE%Python Backend - Advanced AI Processing APIs%YELLOW% :%RESET% +echo %YELLOW% : %WHITE%n8n Workflows - Automation Made Simple%YELLOW% :%RESET% +echo %YELLOW% : :%RESET% +echo %YELLOW% +==================================================================+%RESET% +echo. +echo %GREEN% Auto-GPU Detection - Secure Deployment - Production Ready%RESET% +echo. +timeout /t 2 /nobreak >nul + +echo %BLUE%=================================================================================%RESET% +echo %BOLD%%WHITE% [SERVICE SELECTION] %RESET% +echo %BLUE%=================================================================================%RESET% +echo. +echo %WHITE%Please select which services you want to deploy:%RESET% +echo. +echo %CYAN% [1] ComfyUI %WHITE%- AI Image Generation (Port 8188)%RESET% +echo %CYAN% [2] Python Backend %WHITE%- Core AI Processing APIs (Port 5001)%RESET% +echo %CYAN% [3] n8n Workflows %WHITE%- Automation Platform (Port 5678)%RESET% +echo %CYAN% [A] All Services %WHITE%- Deploy everything (Recommended)%RESET% +echo. + +set "DEPLOY_COMFYUI=false" +set "DEPLOY_PYTHON=false" +set "DEPLOY_N8N=false" + +:service_selection +echo %YELLOW%Select services (e.g., 1,3 or A for all): %RESET% +set /p "SERVICE_CHOICE=" + +if /i "!SERVICE_CHOICE!"=="A" ( + set "DEPLOY_COMFYUI=true" + set "DEPLOY_PYTHON=true" + set "DEPLOY_N8N=true" + echo %GREEN%[*] All services selected for deployment%RESET% + goto selection_complete +) + +if "!SERVICE_CHOICE!"=="" goto service_selection + +REM Parse individual service selections +echo !SERVICE_CHOICE! | findstr "1" >nul +if !errorlevel! equ 0 set "DEPLOY_COMFYUI=true" + +echo !SERVICE_CHOICE! | findstr "2" >nul +if !errorlevel! equ 0 set "DEPLOY_PYTHON=true" + +echo !SERVICE_CHOICE! | findstr "3" >nul +if !errorlevel! equ 0 set "DEPLOY_N8N=true" + +REM Check if any service was selected +if "!DEPLOY_COMFYUI!"=="false" ( + if "!DEPLOY_PYTHON!"=="false" ( + if "!DEPLOY_N8N!"=="false" ( + echo %RED%[!] No valid services selected. Please try again.%RESET% + goto service_selection + ) + ) +) + +:selection_complete +echo. +echo %WHITE%Selected services:%RESET% +if "!DEPLOY_COMFYUI!"=="true" echo %GREEN% ✓ ComfyUI%RESET% +if "!DEPLOY_PYTHON!"=="true" echo %GREEN% ✓ Python Backend%RESET% +if "!DEPLOY_N8N!"=="true" echo %GREEN% ✓ n8n Workflows%RESET% +echo. +timeout /t 2 /nobreak >nul + +echo %BLUE%=================================================================================%RESET% +echo %BOLD%%WHITE% [SYSTEM VERIFICATION] %RESET% +echo %BLUE%=================================================================================%RESET% +echo. + +REM Step 1: Check Docker Desktop +set /a "step=1" +call :show_progress +echo %YELLOW%[*] Checking Docker Desktop installation...%RESET% +docker --version >nul 2>&1 +if %errorLevel% neq 0 ( + echo. + echo %RED%[X] ERROR: Docker Desktop is not installed or not in PATH%RESET% + echo. + echo %YELLOW%Please install Docker Desktop for Windows:%RESET% + echo %WHITE% 1. Download from: https://docs.docker.com/desktop/windows/install/%RESET% + echo %WHITE% 2. Install Docker Desktop%RESET% + echo %WHITE% 3. Start Docker Desktop and wait for it to be ready%RESET% + echo %WHITE% 4. Run this script again%RESET% + echo. + pause + exit /b 1 +) + +REM Get Docker version safely +set "DOCKER_VERSION=unknown" +for /f "tokens=3 delims= " %%v in ('docker --version 2^>nul') do ( + set "DOCKER_VERSION=%%v" + goto version_found +) +:version_found +echo %GREEN%[OK] Docker Desktop %DOCKER_VERSION% detected%RESET% + +REM Step 2: Check Docker daemon +set /a "step=2" +call :show_progress +echo %YELLOW%[*] Verifying Docker daemon status...%RESET% +docker info >nul 2>&1 +if %errorLevel% neq 0 ( + echo %RED%[X] Docker daemon not responding%RESET% + echo %YELLOW%[i] Starting Docker Desktop automatically...%RESET% + + REM Try to find and start Docker Desktop + if exist "C:\Program Files\Docker\Docker\Docker Desktop.exe" ( + start "" "C:\Program Files\Docker\Docker\Docker Desktop.exe" + ) else if exist "%LOCALAPPDATA%\Docker\Docker Desktop.exe" ( + start "" "%LOCALAPPDATA%\Docker\Docker Desktop.exe" + ) else ( + echo %RED%[X] Cannot find Docker Desktop executable%RESET% + echo %YELLOW%Please start Docker Desktop manually and run this script again%RESET% + pause + exit /b 1 + ) + + echo %CYAN%[*] Waiting for Docker Desktop to start (60 seconds)...%RESET% + + set /a "wait_count=0" + :wait_docker + timeout /t 5 /nobreak >nul + docker info >nul 2>&1 + if %errorLevel% equ 0 goto docker_ready + + set /a "wait_count+=1" + if %wait_count% lss 12 ( + echo %CYAN% [*] Still waiting... (!wait_count!/12)%RESET% + goto wait_docker + ) + + echo %RED%[X] Docker Desktop failed to start automatically%RESET% + echo %YELLOW%Please start Docker Desktop manually and run this script again%RESET% + pause + exit /b 1 +) + +:docker_ready +echo %GREEN%[OK] Docker daemon is running%RESET% + +REM Step 3: GPU Detection +set /a "step=3" +call :show_progress +echo %YELLOW%[*] Scanning for NVIDIA GPU...%RESET% +set "GPU_AVAILABLE=false" + +where nvidia-smi >nul 2>&1 +if %errorLevel% equ 0 ( + nvidia-smi --query-gpu=name --format=csv,noheader >temp_gpu.txt 2>nul + if %errorLevel% equ 0 ( + for /f "tokens=*" %%g in (temp_gpu.txt) do ( + echo %GREEN%[OK] GPU Detected: %%g%RESET% + set "GPU_AVAILABLE=true" + del temp_gpu.txt >nul 2>&1 + goto gpu_check_done + ) + ) + del temp_gpu.txt >nul 2>&1 +) +echo %YELLOW%[!] No NVIDIA GPU detected - CPU mode will be used%RESET% + +:gpu_check_done + +REM Step 4: Test GPU Docker support +if "%GPU_AVAILABLE%"=="true" ( + set /a "step=4" + call :show_progress + echo %YELLOW%[*] Testing NVIDIA Docker integration...%RESET% + docker run --rm --gpus all nvidia/cuda:11.8.0-base-ubuntu22.04 nvidia-smi >nul 2>&1 + if %errorLevel% equ 0 ( + echo %GREEN%[OK] GPU Docker acceleration ready%RESET% + set "GPU_DOCKER=true" + ) else ( + echo %YELLOW%[!] GPU available but Docker GPU support not working%RESET% + echo %CYAN%[i] Continuing with CPU mode%RESET% + set "GPU_DOCKER=false" + set "GPU_AVAILABLE=false" + ) +) else ( + set /a "step=4" + call :show_progress + echo %CYAN%[i] Configuring for CPU-only deployment%RESET% + set "GPU_DOCKER=false" +) + +echo. +echo %BLUE%=================================================================================%RESET% +echo %BOLD%%WHITE% [DEPLOYMENT CONFIGURATION] %RESET% +echo %BLUE%=================================================================================%RESET% +echo. + +REM Step 5: Create directories +set /a "step=5" +call :show_progress +echo %YELLOW%[*] Setting up data directories...%RESET% +if not exist "data" mkdir data +if not exist "data\comfyui" mkdir data\comfyui +if not exist "data\python" mkdir data\python +if not exist "data\n8n" mkdir data\n8n +if not exist "logs" mkdir logs +echo %GREEN%[OK] Directory structure created%RESET% + +REM Step 6: Configure for GPU/CPU +set /a "step=6" +call :show_progress +echo %YELLOW%[*] Generating optimized configuration...%RESET% + +if "%GPU_AVAILABLE%"=="false" ( + echo %CYAN%[i] CPU-only deployment (GPU options will be ignored by Docker)%RESET% +) else ( + echo %CYAN%[i] GPU-accelerated deployment ready%RESET% +) + +echo. +echo %BLUE%=================================================================================%RESET% +echo %BOLD%%WHITE% [DOWNLOADING COMPONENTS] %RESET% +echo %BLUE%=================================================================================%RESET% +echo. + +REM Step 7: Pull images +set /a "step=7" +call :show_progress +echo %YELLOW%[*] Downloading selected ClaraVerse containers...%RESET% +echo %CYAN% This may take several minutes for first-time setup%RESET% +echo. + +REM Build service list for pulling +set "SERVICES=" +if "!DEPLOY_COMFYUI!"=="true" set "SERVICES=!SERVICES! clara_comfyui" +if "!DEPLOY_PYTHON!"=="true" set "SERVICES=!SERVICES! clara_python" +if "!DEPLOY_N8N!"=="true" set "SERVICES=!SERVICES! clara_n8n" + +docker compose pull!SERVICES! +if %errorLevel% neq 0 ( + echo %RED%[X] Failed to download container images%RESET% + echo %YELLOW%Please check your internet connection and try again%RESET% + pause + exit /b 1 +) +echo %GREEN%[OK] All container images downloaded successfully%RESET% + +echo. +echo %BLUE%=================================================================================%RESET% +echo %BOLD%%WHITE% [LAUNCHING CLARAVERSE] %RESET% +echo %BLUE%=================================================================================%RESET% +echo. + +REM Step 8: Deploy services +set /a "step=8" +call :show_progress +echo %YELLOW%[*] Starting selected ClaraVerse services...%RESET% + +REM Build service list based on selections +set "SERVICES=" +if "!DEPLOY_COMFYUI!"=="true" set "SERVICES=!SERVICES! clara_comfyui" +if "!DEPLOY_PYTHON!"=="true" set "SERVICES=!SERVICES! clara_python" +if "!DEPLOY_N8N!"=="true" set "SERVICES=!SERVICES! clara_n8n" + +echo %CYAN%[*] Starting services:!SERVICES!%RESET% + +docker compose up -d!SERVICES! +if %errorLevel% neq 0 ( + echo %RED%[X] Failed to start services%RESET% + echo %YELLOW%Check Docker Desktop and try again%RESET% + pause + exit /b 1 +) + +echo %GREEN%[OK] All services launched successfully%RESET% +echo. + +REM Wait for services with animated progress +echo %YELLOW%[*] Waiting for services to initialize...%RESET% +set /a "health_attempts=0" +set /a "max_health_attempts=24" + +:health_check_loop +set /a "health_attempts+=1" +set /a "health_percent=health_attempts*100/max_health_attempts" + +REM Create animated dots +set "dots=..." +set /a "dot_count=health_attempts%%4" +if !dot_count! equ 0 set "dots= " +if !dot_count! equ 1 set "dots=. " +if !dot_count! equ 2 set "dots=.. " +if !dot_count! equ 3 set "dots=..." + +echo %CYAN% [!health_percent!%%] Checking service health!dots! %RESET% + +REM Check health of selected services +set "HEALTH_OK=true" + +if "!DEPLOY_COMFYUI!"=="true" ( + curl -f http://localhost:8188/ >nul 2>&1 + if !errorLevel! neq 0 set "HEALTH_OK=false" +) + +if "!DEPLOY_PYTHON!"=="true" ( + curl -f http://localhost:5001/health >nul 2>&1 + if !errorLevel! neq 0 set "HEALTH_OK=false" +) + +if "!DEPLOY_N8N!"=="true" ( + curl -f http://localhost:5678/healthz >nul 2>&1 + if !errorLevel! neq 0 set "HEALTH_OK=false" +) + +if "!HEALTH_OK!"=="true" ( + echo %GREEN%[OK] All selected services are healthy and ready!%RESET% + goto services_ready +) + +REM Check if containers are at least running +docker compose ps --format json >nul 2>&1 +if %errorLevel% equ 0 ( + if !health_attempts! geq 6 ( + echo %GREEN%[OK] Services are running (health endpoint not available yet)%RESET% + goto services_ready + ) +) + +if !health_attempts! geq !max_health_attempts! ( + echo %YELLOW%[!] Services are taking longer than expected to start%RESET% + echo %CYAN%[i] You can check status with: docker compose ps%RESET% + goto services_ready +) + +timeout /t 5 /nobreak >nul +goto health_check_loop + +:services_ready + +REM Get local IP safely +set "LOCAL_IP=localhost" +ipconfig >temp_ip.txt 2>nul +for /f "tokens=2 delims=:" %%a in ('findstr /c:"IPv4" temp_ip.txt ^| findstr "192.168 10. 172."') do ( + for /f "tokens=*" %%b in ("%%a") do ( + set "LOCAL_IP=%%b" + goto ip_found + ) +) +del temp_ip.txt >nul 2>&1 + +:ip_found + +cls +echo. +echo %GREEN% +echo ###### ####### ###### # ####### # # ####### ###### +echo # # # # # # # # # # # # # +echo # # ##### ###### # # # # # ##### # # +echo # # # # # # # # # # # +echo # # # # # # # # # # # +echo ###### ####### # ####### ####### # ####### ###### +echo %RESET% +echo. +echo %BOLD%%MAGENTA% +=================================================================+%RESET% +echo %BOLD%%MAGENTA% : :%RESET% +echo %BOLD%%MAGENTA% : CLARAVERSE DEPLOYMENT SUCCESSFUL! :%RESET% +echo %BOLD%%MAGENTA% : :%RESET% +echo %BOLD%%MAGENTA% +=================================================================+%RESET% +echo. +echo %CYAN%Your selected ClaraVerse services are now running:%RESET% +echo. +echo %WHITE% +-----------------------------------------------------------------+%RESET% +echo %WHITE% : :%RESET% +if "!DEPLOY_COMFYUI!"=="true" echo %WHITE% : %BOLD%%BLUE%ComfyUI: %RESET% %YELLOW%http://!LOCAL_IP!:8188%RESET% +if "!DEPLOY_PYTHON!"=="true" echo %WHITE% : %BOLD%%BLUE%Python API: %RESET% %YELLOW%http://!LOCAL_IP!:5001%RESET% +if "!DEPLOY_N8N!"=="true" echo %WHITE% : %BOLD%%BLUE%n8n Workflows:%RESET% %YELLOW%http://!LOCAL_IP!:5678%RESET% +echo %WHITE% : :%RESET% +echo %WHITE% +-----------------------------------------------------------------+%RESET% +echo. +echo %GREEN%Quick Management Commands:%RESET% +echo %CYAN% Check Status: %WHITE%docker compose ps%RESET% +echo %CYAN% View Logs: %WHITE%docker compose logs -f%RESET% +echo %CYAN% Restart: %WHITE%docker compose restart%RESET% +echo %CYAN% Stop: %WHITE%docker compose down%RESET% +echo. + +if "%GPU_AVAILABLE%"=="true" ( + echo %GREEN%GPU Acceleration: %BOLD%ENABLED%RESET% %GREEN%(NVIDIA CUDA)%RESET% +) else ( + echo %YELLOW%Running Mode: %BOLD%CPU Only%RESET% %YELLOW%(No GPU detected)%RESET% +) + +echo. +echo %BOLD%%GREEN%System Specifications:%RESET% +echo %WHITE% • Docker: %DOCKER_VERSION%%RESET% +if "%GPU_AVAILABLE%"=="true" ( + echo %WHITE% • GPU: NVIDIA GPU with CUDA support%RESET% +) else ( + echo %WHITE% • Mode: CPU-only deployment%RESET% +) +echo %WHITE% • Services: ComfyUI, Python Backend, n8n%RESET% +echo %WHITE% • Architecture: Windows with Docker Desktop%RESET% +echo. +echo %BLUE%=================================================================================%RESET% +echo %BOLD%%WHITE%Thank you for using ClaraVerse! Happy creating!%RESET% +echo %BLUE%=================================================================================%RESET% +echo. +echo %YELLOW%Press any key to open ComfyUI in your browser...%RESET% +pause >nul + +REM Open browser to ComfyUI +start http://!LOCAL_IP!:8188 + +echo %GREEN%Browser opened! Installation complete.%RESET% +echo %WHITE%Access your services at:%RESET% +echo %WHITE% ComfyUI: http://!LOCAL_IP!:8188%RESET% +echo %WHITE% n8n: http://!LOCAL_IP!:5678%RESET% +echo %WHITE% Python Backend: http://!LOCAL_IP!:5001%RESET% +echo. +timeout /t 3 /nobreak >nul +exit /b 0 + +REM ============= FUNCTIONS ============= + +:show_progress +set /a "percent=!step!*100/!total_steps!" +set /a "filled=!step!*!PROGRESS_WIDTH!/!total_steps!" +set /a "empty=!PROGRESS_WIDTH!-!filled!" + +set "bar=" +for /l %%i in (1,1,!filled!) do set "bar=!bar!#" +for /l %%i in (1,1,!empty!) do set "bar=!bar!-" + +echo %CYAN%[!bar!] !percent!%% %RESET% +exit /b 0 \ No newline at end of file diff --git a/claraverse-agent-deployement/scripts/install.sh b/claraverse-agent-deployement/scripts/install.sh new file mode 100644 index 00000000..70ccafa8 --- /dev/null +++ b/claraverse-agent-deployement/scripts/install.sh @@ -0,0 +1,382 @@ +#!/bin/bash + +# ClaraVerse Remote Server Installer - Linux/Mac +# This script installs Docker, sets up GPU support, and deploys ClaraVerse services + +set -e + +echo "🚀 ClaraVerse Remote Server Installation" +echo "=======================================" + +# Colors for output +RED='\033[0;31m' +GREEN='\033[0;32m' +YELLOW='\033[1;33m' +BLUE='\033[0;34m' +CYAN='\033[0;36m' +NC='\033[0m' # No Color + +echo "" +echo -e "${BLUE}=================================================================================${NC}" +echo -e "${BLUE} [SERVICE SELECTION] ${NC}" +echo -e "${BLUE}=================================================================================${NC}" +echo "" +echo -e "Please select which services you want to deploy:" +echo "" +echo -e "${CYAN} [1] ComfyUI ${NC}- AI Image Generation (Port 8188)" +echo -e "${CYAN} [2] Python Backend ${NC}- Core AI Processing APIs (Port 5001)" +echo -e "${CYAN} [3] n8n Workflows ${NC}- Automation Platform (Port 5678)" +echo -e "${CYAN} [A] All Services ${NC}- Deploy everything (Recommended)" +echo "" + +DEPLOY_COMFYUI=false +DEPLOY_PYTHON=false +DEPLOY_N8N=false + +while true; do + echo -e "${YELLOW}Select services (e.g., 1,3 or A for all): ${NC}" + read -r SERVICE_CHOICE + + if [[ "${SERVICE_CHOICE^^}" == "A" ]]; then + DEPLOY_COMFYUI=true + DEPLOY_PYTHON=true + DEPLOY_N8N=true + echo -e "${GREEN}[*] All services selected for deployment${NC}" + break + fi + + if [[ -z "$SERVICE_CHOICE" ]]; then + continue + fi + + # Parse individual service selections + if [[ "$SERVICE_CHOICE" == *"1"* ]]; then + DEPLOY_COMFYUI=true + fi + if [[ "$SERVICE_CHOICE" == *"2"* ]]; then + DEPLOY_PYTHON=true + fi + if [[ "$SERVICE_CHOICE" == *"3"* ]]; then + DEPLOY_N8N=true + fi + + if [[ "$DEPLOY_COMFYUI" == false && "$DEPLOY_PYTHON" == false && "$DEPLOY_N8N" == false ]]; then + echo -e "${RED}[!] No valid services selected. Please try again.${NC}" + continue + fi + + break +done + +echo "" +echo "Selected services:" +if [[ "$DEPLOY_COMFYUI" == true ]]; then + echo -e "${GREEN} ✓ ComfyUI${NC}" +fi +if [[ "$DEPLOY_PYTHON" == true ]]; then + echo -e "${GREEN} ✓ Python Backend${NC}" +fi +if [[ "$DEPLOY_N8N" == true ]]; then + echo -e "${GREEN} ✓ n8n Workflows${NC}" +fi +echo "" +sleep 2 + +# Logging function +log() { + echo -e "${GREEN}[$(date +'%Y-%m-%d %H:%M:%S')] $1${NC}" +} + +warn() { + echo -e "${YELLOW}[$(date +'%Y-%m-%d %H:%M:%S')] WARNING: $1${NC}" +} + +error() { + echo -e "${RED}[$(date +'%Y-%m-%d %H:%M:%S')] ERROR: $1${NC}" +} + +# Check if running as root +check_root() { + if [ "$EUID" -eq 0 ]; then + warn "Running as root. This is not recommended for production." + fi +} + +# Detect OS +detect_os() { + if [[ "$OSTYPE" == "linux-gnu"* ]]; then + OS="linux" + if [ -f /etc/debian_version ]; then + DISTRO="debian" + elif [ -f /etc/redhat-release ]; then + DISTRO="redhat" + else + DISTRO="unknown" + fi + elif [[ "$OSTYPE" == "darwin"* ]]; then + OS="macos" + else + error "Unsupported operating system: $OSTYPE" + exit 1 + fi + log "Detected OS: $OS ($DISTRO)" +} + +# Install Docker +install_docker() { + log "Installing Docker..." + + if command -v docker &> /dev/null; then + log "Docker is already installed" + docker --version + return + fi + + if [ "$OS" == "linux" ]; then + if [ "$DISTRO" == "debian" ]; then + # Ubuntu/Debian + sudo apt-get update + sudo apt-get install -y apt-transport-https ca-certificates curl gnupg lsb-release + curl -fsSL https://download.docker.com/linux/ubuntu/gpg | sudo gpg --dearmor -o /usr/share/keyrings/docker-archive-keyring.gpg + echo "deb [arch=$(dpkg --print-architecture) signed-by=/usr/share/keyrings/docker-archive-keyring.gpg] https://download.docker.com/linux/ubuntu $(lsb_release -cs) stable" | sudo tee /etc/apt/sources.list.d/docker.list > /dev/null + sudo apt-get update + sudo apt-get install -y docker-ce docker-ce-cli containerd.io docker-compose-plugin + elif [ "$DISTRO" == "redhat" ]; then + # CentOS/RHEL/Rocky + sudo yum install -y yum-utils + sudo yum-config-manager --add-repo https://download.docker.com/linux/centos/docker-ce.repo + sudo yum install -y docker-ce docker-ce-cli containerd.io docker-compose-plugin + fi + + # Start Docker service + sudo systemctl start docker + sudo systemctl enable docker + + # Add current user to docker group (requires logout/login) + sudo usermod -aG docker $USER + warn "Please logout and login again for Docker group membership to take effect" + + elif [ "$OS" == "macos" ]; then + error "Please install Docker Desktop for Mac manually from https://docs.docker.com/desktop/mac/install/" + exit 1 + fi + + log "Docker installation completed" +} + +# Install NVIDIA Container Toolkit +install_nvidia_support() { + log "Checking for NVIDIA GPU support..." + + # Check if nvidia-smi is available + if ! command -v nvidia-smi &> /dev/null; then + warn "nvidia-smi not found. Skipping GPU support setup." + warn "If you have an NVIDIA GPU, please install NVIDIA drivers first." + return + fi + + log "NVIDIA GPU detected. Installing NVIDIA Container Toolkit..." + + if [ "$OS" == "linux" ]; then + if [ "$DISTRO" == "debian" ]; then + # Ubuntu/Debian + distribution=$(. /etc/os-release;echo $ID$VERSION_ID) \ + && curl -fsSL https://nvidia.github.io/libnvidia-container/gpgkey | sudo gpg --dearmor -o /usr/share/keyrings/nvidia-container-toolkit-keyring.gpg \ + && curl -s -L https://nvidia.github.io/libnvidia-container/$distribution/libnvidia-container.list | \ + sed 's#deb https://#deb [signed-by=/usr/share/keyrings/nvidia-container-toolkit-keyring.gpg] https://#g' | \ + sudo tee /etc/apt/sources.list.d/nvidia-container-toolkit.list + sudo apt-get update + sudo apt-get install -y nvidia-container-toolkit + elif [ "$DISTRO" == "redhat" ]; then + # CentOS/RHEL/Rocky + distribution=$(. /etc/os-release;echo $ID$VERSION_ID) \ + && curl -s -L https://nvidia.github.io/libnvidia-container/$distribution/nvidia-container-toolkit.repo | \ + sudo tee /etc/yum.repos.d/nvidia-container-toolkit.repo + sudo yum install -y nvidia-container-toolkit + fi + + # Configure Docker daemon + sudo nvidia-ctk runtime configure --runtime=docker + sudo systemctl restart docker + + log "NVIDIA Container Toolkit installed successfully" + + # Test GPU access + if sudo docker run --rm --gpus all nvidia/cuda:11.0.3-base-ubuntu20.04 nvidia-smi; then + log "✅ GPU support is working correctly" + else + warn "GPU test failed. GPU support may not be working properly." + fi + fi +} + +# Create directories and set permissions +setup_directories() { + log "Setting up directories..." + + # Create data directories + mkdir -p ./data/{comfyui,python,n8n} + mkdir -p ./logs + + # Set proper permissions + chmod 755 ./data + chmod 755 ./logs + + log "Directories created successfully" +} + +# Generate GPU-aware docker-compose configuration +generate_compose_config() { + log "Generating Docker Compose configuration..." + + # Check if GPU support is available + GPU_AVAILABLE=false + if command -v nvidia-smi &> /dev/null && sudo docker run --rm --gpus all nvidia/cuda:11.0.3-base-ubuntu20.04 nvidia-smi &> /dev/null; then + GPU_AVAILABLE=true + log "GPU support enabled" + else + warn "No GPU support detected. Running in CPU mode." + fi + + # Create CPU-only version if no GPU + if [ "$GPU_AVAILABLE" = false ]; then + log "Creating CPU-only configuration..." + + # Remove GPU-specific configurations + sed -i.bak '/runtime: nvidia/d' docker-compose.yml + sed -i.bak '/NVIDIA_VISIBLE_DEVICES/d' docker-compose.yml + sed -i.bak '/CUDA_VISIBLE_DEVICES/d' docker-compose.yml + sed -i.bak '/PYTORCH_CUDA_ALLOC_CONF/d' docker-compose.yml + sed -i.bak '/CUDA_LAUNCH_BLOCKING/d' docker-compose.yml + sed -i.bak '/TORCH_CUDNN_V8_API_ENABLED/d' docker-compose.yml + sed -i.bak '/CUDA_MODULE_LOADING/d' docker-compose.yml + sed -i.bak '/CUDA_CACHE_DISABLE/d' docker-compose.yml + sed -i.bak '/WHISPER_CUDA/d' docker-compose.yml + sed -i.bak 's/FASTER_WHISPER_DEVICE=cuda/FASTER_WHISPER_DEVICE=cpu/' docker-compose.yml + sed -i.bak '/COMFYUI_.*=/d' docker-compose.yml + fi + + log "Docker Compose configuration ready" +} + +# Deploy ClaraVerse services +deploy_services() { + log "Deploying selected ClaraVerse services..." + + # Build service list based on selections + SERVICES="" + if [[ "$DEPLOY_COMFYUI" == true ]]; then + SERVICES="$SERVICES clara_comfyui" + fi + if [[ "$DEPLOY_PYTHON" == true ]]; then + SERVICES="$SERVICES clara_python" + fi + if [[ "$DEPLOY_N8N" == true ]]; then + SERVICES="$SERVICES clara_n8n" + fi + + log "Services to deploy:$SERVICES" + + # Pull latest images for selected services + log "Pulling Docker images for selected services..." + sudo docker compose pull$SERVICES + + # Start selected services + log "Starting selected services..." + sudo docker compose up -d$SERVICES + + log "Selected services deployed successfully" +} + +# Wait for services to be healthy +wait_for_services() { + log "Waiting for selected services to become healthy..." + + local max_attempts=60 + local attempt=0 + + while [ $attempt -lt $max_attempts ]; do + local healthy=true + + # Check health of selected services + if [[ "$DEPLOY_COMFYUI" == true ]]; then + if ! curl -f http://localhost:8188/ &> /dev/null; then + healthy=false + fi + fi + + if [[ "$DEPLOY_PYTHON" == true ]]; then + if ! curl -f http://localhost:5001/health &> /dev/null; then + healthy=false + fi + fi + + if [[ "$DEPLOY_N8N" == true ]]; then + if ! curl -f http://localhost:5678/healthz &> /dev/null; then + healthy=false + fi + fi + + if [[ "$healthy" == true ]]; then + log "✅ All selected services are healthy!" + break + fi + + echo -n "." + sleep 5 + attempt=$((attempt + 1)) + done + + if [ $attempt -eq $max_attempts ]; then + warn "Services are taking longer than expected to start. Check logs with: docker compose logs" + fi +} + +# Display success message +show_success() { + echo "" + echo -e "${GREEN}🎉 ClaraVerse Remote Server Installation Complete!${NC}" + echo "==============================================" + echo "" + echo -e "${BLUE}Access your selected services at:${NC}" + + local server_ip + server_ip=$(hostname -I | awk '{print $1}') + + if [[ "$DEPLOY_COMFYUI" == true ]]; then + echo " 🎨 ComfyUI: http://$server_ip:8188" + fi + if [[ "$DEPLOY_PYTHON" == true ]]; then + echo " � Python Backend: http://$server_ip:5001" + fi + if [[ "$DEPLOY_N8N" == true ]]; then + echo " � n8n: http://$server_ip:5678" + fi + echo "" + echo -e "${YELLOW}Useful commands:${NC}" + echo " 📊 Check status: docker compose ps" + echo " 📝 View logs: docker compose logs -f" + echo " 🔄 Restart: docker compose restart" + echo " ⏹️ Stop: docker compose down" + echo "" +} + +# Main installation flow +main() { + log "Starting ClaraVerse Remote Server installation..." + + check_root + detect_os + install_docker + install_nvidia_support + setup_directories + generate_compose_config + deploy_services + wait_for_services + show_success + + log "Installation completed successfully!" +} + +# Run main function +main "$@" \ No newline at end of file diff --git a/cli/.gitattributes b/cli/.gitattributes deleted file mode 100644 index 606d1e09..00000000 --- a/cli/.gitattributes +++ /dev/null @@ -1,4 +0,0 @@ -# Ensure proper line endings for scripts -claraverse text eol=lf -install.sh text eol=lf -install.ps1 text eol=crlf diff --git a/cli/claraverse b/cli/claraverse deleted file mode 100755 index b2a50916..00000000 --- a/cli/claraverse +++ /dev/null @@ -1,922 +0,0 @@ -#!/bin/bash -# ============================================ -# ClaraVerse CLI - Your Private AI Workspace -# ============================================ -# Install: curl -fsSL https://get.claraverse.ai | bash -# Usage: claraverse [options] -# ============================================ - -set -e - -# Version and URLs -VERSION="1.0.0" -GITHUB_REPO="ClaraVerseOSS/ClaraVerse" -DOCKER_IMAGE="claraverseoss/claraverse:latest" -SEARXNG_IMAGE="searxng/searxng:latest" -UPDATE_CHECK_URL="https://api.github.com/repos/${GITHUB_REPO}/releases/latest" -INSTALL_DIR="${CLARAVERSE_HOME:-$HOME/.claraverse}" -DATA_DIR="${CLARAVERSE_DATA:-$INSTALL_DIR/data}" -CONFIG_FILE="$INSTALL_DIR/config" -CONTAINER_NAME="claraverse" -SEARXNG_CONTAINER="claraverse-search" -NETWORK_NAME="claraverse-net" - -# Colors -RED='\033[0;31m' -GREEN='\033[0;32m' -YELLOW='\033[1;33m' -BLUE='\033[0;34m' -PURPLE='\033[0;35m' -CYAN='\033[0;36m' -NC='\033[0m' # No Color -BOLD='\033[1m' - -# ============================================ -# Helper Functions -# ============================================ - -print_logo() { - echo "" - echo -e "${PURPLE}█▀▀ █ █▀█ █▀█ █▀█ █ █ █▀▀ █▀█ █▀▀ █▀▀${NC}" - echo -e "${PURPLE}█ █ █▀█ █▀▄ █▀█ ▀▄▀ █▀▀ █▀▄ ▀▀█ █▀▀${NC}" - echo -e "${PURPLE}▀▀▀ ▀▀▀ ▀ ▀ ▀ ▀ ▀ ▀ ▀ ▀▀▀ ▀ ▀ ▀▀▀ ▀▀▀${NC}" - echo -e " ${CYAN}Your Private AI Workspace${NC}" - echo "" -} - -log_info() { - echo -e "${BLUE}ℹ${NC} $1" -} - -log_success() { - echo -e "${GREEN}✓${NC} $1" -} - -log_warning() { - echo -e "${YELLOW}⚠${NC} $1" -} - -log_error() { - echo -e "${RED}✗${NC} $1" -} - -log_step() { - echo -e "${PURPLE}→${NC} $1" -} - -# Check if command exists -command_exists() { - command -v "$1" >/dev/null 2>&1 -} - -# Check Docker is installed and running -check_docker() { - if ! command_exists docker; then - log_error "Docker is not installed" - echo "" - echo "Install Docker:" - echo " Linux: curl -fsSL https://get.docker.com | sh" - echo " macOS: brew install --cask docker" - echo " Windows: https://docs.docker.com/desktop/install/windows-install/" - exit 1 - fi - - if ! docker info >/dev/null 2>&1; then - log_error "Docker is not running" - echo "" - echo "Start Docker:" - echo " Linux: sudo systemctl start docker" - echo " macOS: Open Docker Desktop" - echo " Windows: Start Docker Desktop" - exit 1 - fi -} - -# Ensure directories exist -ensure_directories() { - mkdir -p "$INSTALL_DIR" "$DATA_DIR" -} - -# Load config -load_config() { - if [ -f "$CONFIG_FILE" ]; then - source "$CONFIG_FILE" - fi -} - -# Save config -save_config() { - cat > "$CONFIG_FILE" << EOF -# ClaraVerse Configuration -CLARAVERSE_PORT=${CLARAVERSE_PORT:-80} -CLARAVERSE_VERSION=${VERSION} -LAST_UPDATE_CHECK=$(date +%s) -EOF -} - -# Get container status -get_status() { - if docker ps --format '{{.Names}}' | grep -q "^${CONTAINER_NAME}$"; then - echo "running" - elif docker ps -a --format '{{.Names}}' | grep -q "^${CONTAINER_NAME}$"; then - echo "stopped" - else - echo "not_installed" - fi -} - -# Get SearXNG container status -get_searxng_status() { - if docker ps --format '{{.Names}}' | grep -q "^${SEARXNG_CONTAINER}$"; then - echo "running" - elif docker ps -a --format '{{.Names}}' | grep -q "^${SEARXNG_CONTAINER}$"; then - echo "stopped" - else - echo "not_installed" - fi -} - -# Ensure Docker network exists -ensure_network() { - if ! docker network ls --format '{{.Name}}' | grep -q "^${NETWORK_NAME}$"; then - docker network create "$NETWORK_NAME" >/dev/null 2>&1 || true - fi -} - -# Start SearXNG container -start_searxng() { - local searxng_status=$(get_searxng_status) - - if [ "$searxng_status" = "running" ]; then - return 0 - fi - - if [ "$searxng_status" = "stopped" ]; then - docker start "$SEARXNG_CONTAINER" >/dev/null - return 0 - fi - - # Pull and start SearXNG - log_step "Setting up web search (SearXNG)..." - docker pull "$SEARXNG_IMAGE" 2>&1 | grep -E "(Pull|Digest|Status)" || true - - docker run -d \ - --name "$SEARXNG_CONTAINER" \ - --network "$NETWORK_NAME" \ - -v claraverse-searxng:/etc/searxng \ - --restart unless-stopped \ - "$SEARXNG_IMAGE" >/dev/null - - # Wait for SearXNG to initialize and create settings.yml - sleep 5 - - # Enable JSON format for API access (required for ClaraVerse web search) - docker exec "$SEARXNG_CONTAINER" sh -c " - if grep -q 'formats:' /etc/searxng/settings.yml 2>/dev/null; then - if ! grep -q '- json' /etc/searxng/settings.yml; then - sed -i '/formats:/a\\ - json' /etc/searxng/settings.yml - fi - fi - " 2>/dev/null || true - - # Restart to apply config - docker restart "$SEARXNG_CONTAINER" >/dev/null 2>&1 || true - - log_success "Web search enabled" -} - -# Stop SearXNG container -stop_searxng() { - local searxng_status=$(get_searxng_status) - if [ "$searxng_status" = "running" ]; then - docker stop "$SEARXNG_CONTAINER" >/dev/null 2>&1 || true - fi -} - -# Get container health -get_health() { - docker inspect --format='{{.State.Health.Status}}' "$CONTAINER_NAME" 2>/dev/null || echo "unknown" -} - -# Wait for healthy status -wait_for_healthy() { - local max_wait=${1:-120} - local waited=0 - - echo -n "Waiting for ClaraVerse to be ready" - while [ $waited -lt $max_wait ]; do - local health=$(get_health) - if [ "$health" = "healthy" ]; then - echo "" - return 0 - fi - echo -n "." - sleep 2 - waited=$((waited + 2)) - done - echo "" - return 1 -} - -# Check for updates -check_for_updates() { - if ! command_exists curl; then - return 1 - fi - - local latest=$(curl -sf "$UPDATE_CHECK_URL" 2>/dev/null | grep -o '"tag_name": *"[^"]*"' | head -1 | cut -d'"' -f4) - if [ -n "$latest" ] && [ "$latest" != "v$VERSION" ]; then - echo "$latest" - return 0 - fi - return 1 -} - -# ============================================ -# Commands -# ============================================ - -cmd_init() { - print_logo - log_info "Initializing ClaraVerse..." - echo "" - - check_docker - ensure_directories - - local port="${1:-80}" - - # Check if already running - local status=$(get_status) - if [ "$status" = "running" ]; then - log_warning "ClaraVerse is already running" - echo "" - echo "Access: http://localhost:$port" - echo "" - echo "Commands:" - echo " claraverse status - Check status" - echo " claraverse stop - Stop ClaraVerse" - echo " claraverse restart - Restart ClaraVerse" - exit 0 - fi - - # Create Docker network - log_step "Setting up network..." - ensure_network - - # Start SearXNG for web search - start_searxng - - # Pull latest image - log_step "Pulling latest ClaraVerse image..." - docker pull "$DOCKER_IMAGE" 2>&1 | grep -E "(Pull|Digest|Status)" || true - log_success "Image pulled" - - # Remove old container if exists - if [ "$status" = "stopped" ]; then - log_step "Removing old container..." - docker rm "$CONTAINER_NAME" >/dev/null 2>&1 || true - fi - - # Start container with SearXNG connection - log_step "Starting ClaraVerse on port $port..." - docker run -d \ - --name "$CONTAINER_NAME" \ - --network "$NETWORK_NAME" \ - -p "${port}:80" \ - -v claraverse-data:/data \ - -e "SEARXNG_URL=http://${SEARXNG_CONTAINER}:8080" \ - -e "SEARXNG_URLS=http://${SEARXNG_CONTAINER}:8080" \ - --restart unless-stopped \ - "$DOCKER_IMAGE" >/dev/null - - # Wait for healthy - if wait_for_healthy 120; then - log_success "ClaraVerse is ready!" - echo "" - echo -e "${GREEN}============================================${NC}" - echo -e "${GREEN} ClaraVerse is running!${NC}" - echo -e "${GREEN}============================================${NC}" - echo "" - echo -e " ${BOLD}Access:${NC} http://localhost:${port}" - echo "" - echo -e " ${BOLD}Features:${NC}" - echo " - AI Chat & Assistants" - echo " - Web Search (SearXNG)" - echo " - Code Execution (E2B)" - echo " - Image Generation" - echo "" - echo -e " ${BOLD}First Steps:${NC}" - echo " 1. Open http://localhost:${port} in browser" - echo " 2. Register account (first user = admin)" - echo " 3. Add AI provider keys in Settings" - echo "" - echo -e " ${BOLD}Commands:${NC}" - echo " claraverse status - Check status" - echo " claraverse logs - View logs" - echo " claraverse stop - Stop ClaraVerse" - echo " claraverse update - Update to latest" - echo "" - - # Save config - CLARAVERSE_PORT=$port - save_config - else - log_error "ClaraVerse failed to start properly" - echo "" - echo "Check logs: claraverse logs" - exit 1 - fi -} - -cmd_start() { - check_docker - - local status=$(get_status) - case "$status" in - "running") - log_info "ClaraVerse is already running" - ;; - "stopped") - # Ensure network and SearXNG are running - ensure_network - start_searxng - - log_step "Starting ClaraVerse..." - docker start "$CONTAINER_NAME" >/dev/null - if wait_for_healthy 60; then - log_success "ClaraVerse started" - load_config - echo "Access: http://localhost:${CLARAVERSE_PORT:-80}" - else - log_error "Failed to start" - exit 1 - fi - ;; - "not_installed") - log_warning "ClaraVerse is not initialized" - echo "Run: claraverse init" - exit 1 - ;; - esac -} - -cmd_stop() { - check_docker - - local status=$(get_status) - if [ "$status" = "running" ]; then - log_step "Stopping ClaraVerse..." - docker stop "$CONTAINER_NAME" >/dev/null - stop_searxng - log_success "ClaraVerse stopped" - else - log_info "ClaraVerse is not running" - stop_searxng - fi -} - -cmd_restart() { - check_docker - - # Ensure network and SearXNG are running - ensure_network - start_searxng - - log_step "Restarting ClaraVerse..." - docker restart "$CONTAINER_NAME" >/dev/null 2>&1 || { - log_error "ClaraVerse is not running. Use: claraverse start" - exit 1 - } - - if wait_for_healthy 60; then - log_success "ClaraVerse restarted" - load_config - echo "Access: http://localhost:${CLARAVERSE_PORT:-80}" - else - log_error "Failed to restart properly" - exit 1 - fi -} - -cmd_status() { - check_docker - load_config - - echo "" - echo -e "${BOLD}ClaraVerse Status${NC}" - echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━" - - local status=$(get_status) - local health=$(get_health) - local searxng_status=$(get_searxng_status) - - case "$status" in - "running") - echo -e "Main: ${GREEN}● Running${NC}" - case "$health" in - "healthy") - echo -e "Health: ${GREEN}● Healthy${NC}" - ;; - "unhealthy") - echo -e "Health: ${RED}● Unhealthy${NC}" - ;; - *) - echo -e "Health: ${YELLOW}● Starting${NC}" - ;; - esac - - # SearXNG status - case "$searxng_status" in - "running") - echo -e "Search: ${GREEN}● Running${NC}" - ;; - "stopped") - echo -e "Search: ${YELLOW}● Stopped${NC}" - ;; - *) - echo -e "Search: ${RED}● Not installed${NC}" - ;; - esac - - echo -e "Port: ${CLARAVERSE_PORT:-80}" - echo -e "URL: http://localhost:${CLARAVERSE_PORT:-80}" - - # Show resource usage - local stats=$(docker stats --no-stream --format "{{.CPUPerc}}\t{{.MemUsage}}" "$CONTAINER_NAME" 2>/dev/null) - if [ -n "$stats" ]; then - local cpu=$(echo "$stats" | cut -f1) - local mem=$(echo "$stats" | cut -f2) - echo -e "CPU: $cpu" - echo -e "Memory: $mem" - fi - ;; - "stopped") - echo -e "Status: ${YELLOW}● Stopped${NC}" - echo "" - echo "Start with: claraverse start" - ;; - "not_installed") - echo -e "Status: ${RED}● Not Installed${NC}" - echo "" - echo "Initialize with: claraverse init" - ;; - esac - - echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━" - echo "" - - # Check for updates (async, don't block) - if [ -n "${LAST_UPDATE_CHECK:-}" ]; then - local now=$(date +%s) - local diff=$((now - LAST_UPDATE_CHECK)) - # Check every 24 hours - if [ $diff -gt 86400 ]; then - local latest=$(check_for_updates) - if [ -n "$latest" ]; then - echo -e "${YELLOW}Update available: $latest${NC}" - echo "Run: claraverse update" - echo "" - fi - fi - fi -} - -cmd_logs() { - check_docker - - local follow="" - local lines="100" - - while [[ $# -gt 0 ]]; do - case "$1" in - -f|--follow) - follow="-f" - shift - ;; - -n|--lines) - lines="$2" - shift 2 - ;; - *) - shift - ;; - esac - done - - docker logs $follow --tail "$lines" "$CONTAINER_NAME" 2>&1 -} - -cmd_update() { - check_docker - load_config - - log_info "Checking for updates..." - - # Ensure network and SearXNG - ensure_network - start_searxng - - # Pull latest images - log_step "Pulling latest images..." - local pull_output=$(docker pull "$DOCKER_IMAGE" 2>&1) - docker pull "$SEARXNG_IMAGE" >/dev/null 2>&1 || true - - if echo "$pull_output" | grep -q "Image is up to date"; then - log_success "Already running the latest version" - return 0 - fi - - log_success "New version downloaded" - - # Check if running - local status=$(get_status) - local was_running=false - - if [ "$status" = "running" ]; then - was_running=true - log_step "Stopping current instance..." - docker stop "$CONTAINER_NAME" >/dev/null - fi - - if [ "$status" != "not_installed" ]; then - log_step "Removing old container..." - docker rm "$CONTAINER_NAME" >/dev/null 2>&1 || true - fi - - # Start new container with SearXNG connection - log_step "Starting updated ClaraVerse..." - docker run -d \ - --name "$CONTAINER_NAME" \ - --network "$NETWORK_NAME" \ - -p "${CLARAVERSE_PORT:-80}:80" \ - -v claraverse-data:/data \ - -e "SEARXNG_URL=http://${SEARXNG_CONTAINER}:8080" \ - -e "SEARXNG_URLS=http://${SEARXNG_CONTAINER}:8080" \ - --restart unless-stopped \ - "$DOCKER_IMAGE" >/dev/null - - if wait_for_healthy 120; then - log_success "ClaraVerse updated successfully!" - echo "Access: http://localhost:${CLARAVERSE_PORT:-80}" - - # Update config - save_config - else - log_error "Update failed - ClaraVerse is not healthy" - echo "Check logs: claraverse logs" - exit 1 - fi -} - -cmd_uninstall() { - check_docker - - echo -e "${YELLOW}This will remove ClaraVerse and all its data.${NC}" - read -p "Are you sure? (y/N) " -n 1 -r - echo "" - - if [[ ! $REPLY =~ ^[Yy]$ ]]; then - log_info "Cancelled" - exit 0 - fi - - log_step "Stopping ClaraVerse..." - docker stop "$CONTAINER_NAME" >/dev/null 2>&1 || true - docker stop "$SEARXNG_CONTAINER" >/dev/null 2>&1 || true - - log_step "Removing containers..." - docker rm "$CONTAINER_NAME" >/dev/null 2>&1 || true - docker rm "$SEARXNG_CONTAINER" >/dev/null 2>&1 || true - - read -p "Remove all data? (y/N) " -n 1 -r - echo "" - if [[ $REPLY =~ ^[Yy]$ ]]; then - log_step "Removing data volumes..." - docker volume rm claraverse-data >/dev/null 2>&1 || true - docker volume rm claraverse-searxng >/dev/null 2>&1 || true - fi - - log_step "Removing network..." - docker network rm "$NETWORK_NAME" >/dev/null 2>&1 || true - - log_step "Removing config..." - rm -rf "$INSTALL_DIR" - - log_success "ClaraVerse uninstalled" - echo "" - echo "To remove the CLI tool:" - echo " sudo rm /usr/local/bin/claraverse" -} - -cmd_clean() { - check_docker - - echo -e "${YELLOW}⚠ This will remove ALL ClaraVerse data:${NC}" - echo " - All databases (MongoDB, MySQL)" - echo " - All uploaded files" - echo " - All user accounts" - echo " - All search data" - echo "" - echo -e "${RED}This action cannot be undone!${NC}" - echo "" - read -p "Are you sure you want to delete everything? (y/N) " -n 1 -r - echo "" - - if [[ ! $REPLY =~ ^[Yy]$ ]]; then - log_info "Cancelled" - exit 0 - fi - - # Double confirm - read -p "Type 'DELETE' to confirm: " confirm - if [ "$confirm" != "DELETE" ]; then - log_info "Cancelled" - exit 0 - fi - - log_step "Stopping containers..." - docker stop "$CONTAINER_NAME" >/dev/null 2>&1 || true - docker stop "$SEARXNG_CONTAINER" >/dev/null 2>&1 || true - - log_step "Removing containers..." - docker rm "$CONTAINER_NAME" >/dev/null 2>&1 || true - docker rm "$SEARXNG_CONTAINER" >/dev/null 2>&1 || true - - log_step "Removing all data volumes..." - docker volume rm claraverse-data >/dev/null 2>&1 || true - docker volume rm claraverse-searxng >/dev/null 2>&1 || true - - log_step "Removing network..." - docker network rm "$NETWORK_NAME" >/dev/null 2>&1 || true - - log_success "All ClaraVerse data has been removed" - echo "" - echo "To start fresh:" - echo " claraverse init" -} - -cmd_shell() { - check_docker - - local status=$(get_status) - if [ "$status" != "running" ]; then - log_error "ClaraVerse is not running" - exit 1 - fi - - docker exec -it "$CONTAINER_NAME" /bin/bash -} - -cmd_backup() { - check_docker - ensure_directories - - local backup_file="${1:-claraverse-backup-$(date +%Y%m%d-%H%M%S).tar.gz}" - - log_step "Creating backup..." - - # Stop if running for consistent backup - local was_running=false - if [ "$(get_status)" = "running" ]; then - was_running=true - log_step "Stopping ClaraVerse for backup..." - docker stop "$CONTAINER_NAME" >/dev/null - fi - - # Create backup from volume - docker run --rm \ - -v claraverse-data:/data \ - -v "$(pwd):/backup" \ - ubuntu:22.04 \ - tar czf "/backup/$backup_file" -C /data . - - # Restart if was running - if [ "$was_running" = true ]; then - log_step "Restarting ClaraVerse..." - docker start "$CONTAINER_NAME" >/dev/null - fi - - log_success "Backup created: $backup_file" -} - -cmd_restore() { - check_docker - - local backup_file="$1" - - if [ -z "$backup_file" ]; then - log_error "Usage: claraverse restore " - exit 1 - fi - - if [ ! -f "$backup_file" ]; then - log_error "Backup file not found: $backup_file" - exit 1 - fi - - echo -e "${YELLOW}This will replace all current data with the backup.${NC}" - read -p "Are you sure? (y/N) " -n 1 -r - echo "" - - if [[ ! $REPLY =~ ^[Yy]$ ]]; then - log_info "Cancelled" - exit 0 - fi - - # Stop if running - if [ "$(get_status)" = "running" ]; then - log_step "Stopping ClaraVerse..." - docker stop "$CONTAINER_NAME" >/dev/null - fi - - # Restore backup - log_step "Restoring from backup..." - docker run --rm \ - -v claraverse-data:/data \ - -v "$(cd "$(dirname "$backup_file")" && pwd):/backup" \ - ubuntu:22.04 \ - bash -c "rm -rf /data/* && tar xzf /backup/$(basename "$backup_file") -C /data" - - log_success "Backup restored" - - # Start - log_step "Starting ClaraVerse..." - docker start "$CONTAINER_NAME" >/dev/null 2>&1 || cmd_init -} - -cmd_config() { - load_config - - case "${1:-show}" in - show) - echo "" - echo -e "${BOLD}ClaraVerse Configuration${NC}" - echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━" - echo "Install Dir: $INSTALL_DIR" - echo "Data Dir: $DATA_DIR" - echo "Port: ${CLARAVERSE_PORT:-80}" - echo "Version: $VERSION" - echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━" - echo "" - ;; - set) - local key="$2" - local value="$3" - if [ -z "$key" ] || [ -z "$value" ]; then - log_error "Usage: claraverse config set " - exit 1 - fi - case "$key" in - port) - CLARAVERSE_PORT="$value" - save_config - log_success "Port set to $value" - log_info "Restart to apply: claraverse restart" - ;; - *) - log_error "Unknown config key: $key" - exit 1 - ;; - esac - ;; - *) - log_error "Unknown config command: $1" - echo "Usage: claraverse config [show|set ]" - exit 1 - ;; - esac -} - -cmd_version() { - echo "ClaraVerse CLI v$VERSION" - echo "" - - # Show Docker image info if running - local status=$(get_status) - local searxng_status=$(get_searxng_status) - - if [ "$status" = "running" ] || [ "$status" = "stopped" ]; then - local image_id=$(docker inspect --format='{{.Image}}' "$CONTAINER_NAME" 2>/dev/null | cut -c8-19) - echo "Main Image: $DOCKER_IMAGE" - [ -n "$image_id" ] && echo " ($image_id)" - fi - - if [ "$searxng_status" = "running" ] || [ "$searxng_status" = "stopped" ]; then - echo "Search Image: $SEARXNG_IMAGE" - fi - - echo "" - echo "Components:" - echo " - ClaraVerse (All-in-One)" - echo " Backend, Frontend, MongoDB, MySQL, Redis, E2B" - echo " - SearXNG (Web Search)" -} - -cmd_help() { - print_logo - echo -e "${BOLD}Usage:${NC} claraverse [options]" - echo "" - echo -e "${BOLD}Commands:${NC}" - echo " init [port] Initialize and start ClaraVerse (default port: 80)" - echo " start Start ClaraVerse" - echo " stop Stop ClaraVerse" - echo " restart Restart ClaraVerse" - echo " status Show status and health" - echo " logs [-f] View logs (-f to follow)" - echo " update Update to latest version" - echo " backup [file] Create backup of all data" - echo " restore Restore from backup" - echo " config View or set configuration" - echo " shell Open shell in container" - echo " clean Remove all data and start fresh" - echo " uninstall Remove ClaraVerse completely" - echo " version Show version information" - echo " help Show this help message" - echo "" - echo -e "${BOLD}What's Included:${NC}" - echo " - AI Chat & Assistants" - echo " - Web Search (SearXNG)" - echo " - Code Execution (E2B)" - echo " - Image Generation" - echo " - MongoDB, MySQL, Redis" - echo "" - echo -e "${BOLD}Examples:${NC}" - echo " claraverse init # Start on port 80" - echo " claraverse init 8080 # Start on port 8080" - echo " claraverse logs -f # Follow logs" - echo " claraverse backup # Create backup" - echo " claraverse config set port 8080" - echo "" - echo -e "${BOLD}Environment Variables:${NC}" - echo " CLARAVERSE_HOME Installation directory (default: ~/.claraverse)" - echo " CLARAVERSE_DATA Data directory (default: ~/.claraverse/data)" - echo "" - echo "Documentation: https://docs.claraverse.ai" - echo "Issues: https://github.com/${GITHUB_REPO}/issues" - echo "" -} - -# ============================================ -# Main -# ============================================ - -main() { - local command="${1:-help}" - shift || true - - case "$command" in - init) - cmd_init "$@" - ;; - start) - cmd_start "$@" - ;; - stop) - cmd_stop "$@" - ;; - restart) - cmd_restart "$@" - ;; - status) - cmd_status "$@" - ;; - logs) - cmd_logs "$@" - ;; - update) - cmd_update "$@" - ;; - backup) - cmd_backup "$@" - ;; - restore) - cmd_restore "$@" - ;; - config) - cmd_config "$@" - ;; - shell) - cmd_shell "$@" - ;; - clean) - cmd_clean "$@" - ;; - uninstall) - cmd_uninstall "$@" - ;; - version|-v|--version) - cmd_version "$@" - ;; - help|-h|--help) - cmd_help "$@" - ;; - *) - log_error "Unknown command: $command" - echo "Run 'claraverse help' for usage" - exit 1 - ;; - esac -} - -main "$@" diff --git a/cli/install.ps1 b/cli/install.ps1 deleted file mode 100644 index 5007d2c3..00000000 --- a/cli/install.ps1 +++ /dev/null @@ -1,283 +0,0 @@ -# ============================================ -# ClaraVerse CLI Installer for Windows -# ============================================ -# Usage: iwr -useb https://get.claraverse.ai/windows | iex -# ============================================ - -$ErrorActionPreference = "Stop" - -# Colors -function Write-Color($Color, $Message) { - Write-Host $Message -ForegroundColor $Color -} - -function Write-Info($Message) { Write-Color "Cyan" "[i] $Message" } -function Write-Success($Message) { Write-Color "Green" "[✓] $Message" } -function Write-Warning($Message) { Write-Color "Yellow" "[!] $Message" } -function Write-Error($Message) { Write-Color "Red" "[✗] $Message" } -function Write-Step($Message) { Write-Color "Magenta" "[→] $Message" } - -# Print logo -function Show-Logo { - Write-Host "" - Write-Color "Magenta" "[=] [=] [=] [=] [=] [=] [=] [=] [=] [=]" - Write-Color "Magenta" " C L A R A V E R S E " - Write-Color "Magenta" "[=] [=] [=] [=] [=] [=] [=] [=] [=] [=]" - Write-Host "" - Write-Color "Cyan" " Your Private AI Workspace - Windows Installer" - Write-Host "" -} - -# Check if Docker is installed -function Test-Docker { - try { - $null = docker version 2>&1 - return $true - } catch { - return $false - } -} - -# Main installation -function Install-ClaraVerse { - Show-Logo - - # Create install directory - $InstallDir = "$env:USERPROFILE\.claraverse" - $BinDir = "$InstallDir\bin" - - if (-not (Test-Path $InstallDir)) { - New-Item -ItemType Directory -Path $InstallDir -Force | Out-Null - } - if (-not (Test-Path $BinDir)) { - New-Item -ItemType Directory -Path $BinDir -Force | Out-Null - } - - # Download CLI wrapper script - Write-Step "Creating ClaraVerse CLI..." - - $CliScript = @' -@echo off -setlocal enabledelayedexpansion - -set "DOCKER_IMAGE=claraverseoss/claraverse:latest" -set "SEARXNG_IMAGE=searxng/searxng:latest" -set "CONTAINER_NAME=claraverse" -set "SEARXNG_CONTAINER=claraverse-search" -set "NETWORK_NAME=claraverse-net" -set "PORT=80" - -if "%1"=="" goto :help -if "%1"=="help" goto :help -if "%1"=="-h" goto :help -if "%1"=="--help" goto :help -if "%1"=="init" goto :init -if "%1"=="start" goto :start -if "%1"=="stop" goto :stop -if "%1"=="restart" goto :restart -if "%1"=="status" goto :status -if "%1"=="logs" goto :logs -if "%1"=="update" goto :update -if "%1"=="uninstall" goto :uninstall -if "%1"=="version" goto :version - -echo Unknown command: %1 -echo Run 'claraverse help' for usage -exit /b 1 - -:help -echo. -echo ClaraVerse CLI - Your Private AI Workspace -echo. -echo Usage: claraverse ^ [options] -echo. -echo Commands: -echo init [port] Initialize and start ClaraVerse (default port: 80) -echo start Start ClaraVerse -echo stop Stop ClaraVerse -echo restart Restart ClaraVerse -echo status Show status -echo logs View logs -echo update Update to latest version -echo uninstall Remove ClaraVerse -echo version Show version -echo help Show this help -echo. -echo What's Included: -echo - AI Chat and Assistants -echo - Web Search (SearXNG) -echo - Code Execution (E2B) -echo - Image Generation -echo - MongoDB, MySQL, Redis -echo. -echo Examples: -echo claraverse init -echo claraverse init 8080 -echo claraverse logs -echo. -goto :eof - -:init -if not "%2"=="" set "PORT=%2" -echo [i] Initializing ClaraVerse on port %PORT%... -echo [->] Creating network... -docker network create %NETWORK_NAME% 2>nul -echo [->] Setting up web search (SearXNG)... -docker pull %SEARXNG_IMAGE% -docker rm -f %SEARXNG_CONTAINER% 2>nul -docker run -d --name %SEARXNG_CONTAINER% --network %NETWORK_NAME% -v claraverse-searxng:/etc/searxng --restart unless-stopped %SEARXNG_IMAGE% -echo [->] Pulling latest ClaraVerse image... -docker pull %DOCKER_IMAGE% -echo [->] Starting ClaraVerse... -docker rm -f %CONTAINER_NAME% 2>nul -docker run -d --name %CONTAINER_NAME% --network %NETWORK_NAME% -p %PORT%:80 -v claraverse-data:/data -e "SEARXNG_URL=http://%SEARXNG_CONTAINER%:8080" -e "SEARXNG_URLS=http://%SEARXNG_CONTAINER%:8080" --restart unless-stopped %DOCKER_IMAGE% -echo. -echo [OK] ClaraVerse is starting! -echo. -echo Access: http://localhost:%PORT% -echo. -echo Features: -echo - AI Chat and Assistants -echo - Web Search (SearXNG) -echo - Code Execution (E2B) -echo - Image Generation -echo. -echo First Steps: -echo 1. Open http://localhost:%PORT% in browser -echo 2. Register account (first user = admin) -echo 3. Add AI provider keys in Settings -echo. -goto :eof - -:start -echo [->] Starting ClaraVerse... -docker start %SEARXNG_CONTAINER% 2>nul -docker start %CONTAINER_NAME% -echo [OK] Started -goto :eof - -:stop -echo [->] Stopping ClaraVerse... -docker stop %CONTAINER_NAME% -docker stop %SEARXNG_CONTAINER% 2>nul -echo [OK] Stopped -goto :eof - -:restart -echo [->] Restarting ClaraVerse... -docker restart %SEARXNG_CONTAINER% 2>nul -docker restart %CONTAINER_NAME% -echo [OK] Restarted -goto :eof - -:status -echo. -echo ClaraVerse Status -echo ================================ -for /f "tokens=*" %%i in ('docker ps -a --filter "name=%CONTAINER_NAME%" --format "{{.Status}}"') do ( - echo Main: %%i -) -for /f "tokens=*" %%i in ('docker inspect --format "{{.State.Health.Status}}" %CONTAINER_NAME% 2^>nul') do ( - echo Health: %%i -) -for /f "tokens=*" %%i in ('docker ps -a --filter "name=%SEARXNG_CONTAINER%" --format "{{.Status}}"') do ( - echo Search: %%i -) -echo ================================ -echo. -goto :eof - -:logs -docker logs --tail 100 %CONTAINER_NAME% -goto :eof - -:update -echo [->] Updating ClaraVerse... -docker network create %NETWORK_NAME% 2>nul -docker pull %DOCKER_IMAGE% -docker pull %SEARXNG_IMAGE% -docker stop %CONTAINER_NAME% 2>nul -docker rm %CONTAINER_NAME% 2>nul -docker run -d --name %CONTAINER_NAME% --network %NETWORK_NAME% -p %PORT%:80 -v claraverse-data:/data -e "SEARXNG_URL=http://%SEARXNG_CONTAINER%:8080" -e "SEARXNG_URLS=http://%SEARXNG_CONTAINER%:8080" --restart unless-stopped %DOCKER_IMAGE% -echo [OK] Updated! -goto :eof - -:uninstall -echo [!] This will stop and remove ClaraVerse. -set /p "CONFIRM=Are you sure? (y/N): " -if /i not "%CONFIRM%"=="y" goto :eof -docker stop %CONTAINER_NAME% 2>nul -docker stop %SEARXNG_CONTAINER% 2>nul -docker rm %CONTAINER_NAME% 2>nul -docker rm %SEARXNG_CONTAINER% 2>nul -docker network rm %NETWORK_NAME% 2>nul -echo [OK] ClaraVerse removed -echo. -echo To also remove data: docker volume rm claraverse-data claraverse-searxng -goto :eof - -:version -echo ClaraVerse CLI v1.0.0 (Windows) -echo. -echo Components: -echo - ClaraVerse (All-in-One) -echo Backend, Frontend, MongoDB, MySQL, Redis, E2B -echo - SearXNG (Web Search) -goto :eof -'@ - - $CliBatPath = "$BinDir\claraverse.bat" - $CliScript | Out-File -FilePath $CliBatPath -Encoding ASCII - - Write-Success "CLI script created" - - # Add to PATH - Write-Step "Adding to PATH..." - - $CurrentPath = [Environment]::GetEnvironmentVariable("Path", "User") - if ($CurrentPath -notlike "*$BinDir*") { - [Environment]::SetEnvironmentVariable("Path", "$CurrentPath;$BinDir", "User") - $env:Path = "$env:Path;$BinDir" - Write-Success "Added to PATH" - Write-Warning "You may need to restart your terminal for PATH changes to take effect" - } else { - Write-Info "Already in PATH" - } - - # Check Docker - Write-Host "" - if (Test-Docker) { - Write-Success "Docker is available" - } else { - Write-Warning "Docker Desktop is not running or not installed" - Write-Host "" - Write-Host "Install Docker Desktop from:" - Write-Host " https://docs.docker.com/desktop/install/windows-install/" - Write-Host "" - Write-Host "After installing Docker, run: claraverse init" - return - } - - Write-Host "" - Write-Host "============================================" -ForegroundColor Green - Write-Host " ClaraVerse CLI installed successfully!" -ForegroundColor Green - Write-Host "============================================" -ForegroundColor Green - Write-Host "" - Write-Host "Quick Start:" -ForegroundColor White - Write-Host " claraverse init # Start ClaraVerse" - Write-Host " claraverse status # Check status" - Write-Host " claraverse help # Show all commands" - Write-Host "" - - $StartNow = Read-Host "Start ClaraVerse now? (Y/n)" - if ($StartNow -ne "n" -and $StartNow -ne "N") { - Write-Host "" - & "$CliBatPath" init - } else { - Write-Host "" - Write-Host "Run 'claraverse init' when ready to start." - } -} - -# Run installation -Install-ClaraVerse diff --git a/cli/install.sh b/cli/install.sh deleted file mode 100755 index b22c53a6..00000000 --- a/cli/install.sh +++ /dev/null @@ -1,225 +0,0 @@ -#!/bin/bash -# ============================================ -# ClaraVerse CLI Installer -# ============================================ -# Usage: curl -fsSL https://get.claraverse.app | bash -# ============================================ - -set -e - -# Colors -RED='\033[0;31m' -GREEN='\033[0;32m' -YELLOW='\033[1;33m' -BLUE='\033[0;34m' -PURPLE='\033[0;35m' -CYAN='\033[0;36m' -NC='\033[0m' -BOLD='\033[1m' - -# URLs -GITHUB_REPO="claraverse-space/ClaraVerseAI" -CLI_URL="https://raw.githubusercontent.com/${GITHUB_REPO}/main/cli/claraverse" -INSTALL_PATH="/usr/local/bin/claraverse" - -print_logo() { - echo "" - echo -e "${PURPLE}█▀▀ █ █▀█ █▀█ █▀█ █ █ █▀▀ █▀█ █▀▀ █▀▀${NC}" - echo -e "${PURPLE}█ █ █▀█ █▀▄ █▀█ ▀▄▀ █▀▀ █▀▄ ▀▀█ █▀▀${NC}" - echo -e "${PURPLE}▀▀▀ ▀▀▀ ▀ ▀ ▀ ▀ ▀ ▀ ▀ ▀▀▀ ▀ ▀ ▀▀▀ ▀▀▀${NC}" - echo "" - echo -e "${CYAN}Installing ClaraVerse CLI${NC}" - echo "" -} - -log_info() { - echo -e "${BLUE}ℹ${NC} $1" -} - -log_success() { - echo -e "${GREEN}✓${NC} $1" -} - -log_warning() { - echo -e "${YELLOW}⚠${NC} $1" -} - -log_error() { - echo -e "${RED}✗${NC} $1" -} - -log_step() { - echo -e "${PURPLE}→${NC} $1" -} - -# Detect OS -detect_os() { - case "$(uname -s)" in - Linux*) echo "linux";; - Darwin*) echo "macos";; - CYGWIN*|MINGW*|MSYS*) echo "windows";; - *) echo "unknown";; - esac -} - -# Detect architecture -detect_arch() { - case "$(uname -m)" in - x86_64|amd64) echo "amd64";; - aarch64|arm64) echo "arm64";; - armv7l) echo "armv7";; - *) echo "unknown";; - esac -} - -# Check if command exists -command_exists() { - command -v "$1" >/dev/null 2>&1 -} - -# Check Docker -check_docker() { - if ! command_exists docker; then - log_warning "Docker is not installed" - echo "" - echo "ClaraVerse requires Docker. Install it first:" - echo "" - case "$(detect_os)" in - linux) - echo " curl -fsSL https://get.docker.com | sh" - echo " sudo usermod -aG docker \$USER" - echo " newgrp docker" - ;; - macos) - echo " brew install --cask docker" - echo " # Or download from: https://docs.docker.com/desktop/install/mac-install/" - ;; - windows) - echo " Download from: https://docs.docker.com/desktop/install/windows-install/" - ;; - esac - echo "" - return 1 - fi - return 0 -} - -# Install CLI -install_cli() { - local os=$(detect_os) - local arch=$(detect_arch) - - log_info "Detected: $os ($arch)" - - # Check for sudo/root - local use_sudo="" - if [ "$(id -u)" -ne 0 ]; then - if command_exists sudo; then - use_sudo="sudo" - else - log_error "This script requires root privileges" - log_info "Run with: sudo bash -c '\$(curl -fsSL https://get.claraverse.app)'" - exit 1 - fi - fi - - # Download CLI - log_step "Downloading ClaraVerse CLI..." - - local tmp_file=$(mktemp) - if command_exists curl; then - curl -fsSL "$CLI_URL" -o "$tmp_file" - elif command_exists wget; then - wget -q "$CLI_URL" -O "$tmp_file" - else - log_error "curl or wget is required" - exit 1 - fi - - # Install - log_step "Installing to $INSTALL_PATH..." - $use_sudo mv "$tmp_file" "$INSTALL_PATH" - $use_sudo chmod +x "$INSTALL_PATH" - - log_success "ClaraVerse CLI installed!" -} - -# Verify installation -verify_installation() { - if command_exists claraverse; then - log_success "Installation verified" - return 0 - else - log_error "Installation verification failed" - log_info "You may need to add /usr/local/bin to your PATH" - return 1 - fi -} - -# Main -main() { - print_logo - - # Check OS - local os=$(detect_os) - if [ "$os" = "unknown" ]; then - log_error "Unsupported operating system" - exit 1 - fi - - # Windows special handling - if [ "$os" = "windows" ]; then - log_warning "Windows detected" - echo "" - echo "For Windows, we recommend using WSL2 (Windows Subsystem for Linux):" - echo " 1. Install WSL2: wsl --install" - echo " 2. Open Ubuntu terminal" - echo " 3. Run: curl -fsSL https://get.claraverse.app | bash" - echo "" - echo "Alternatively, use Docker Desktop with the all-in-one image:" - echo " docker run -d -p 80:80 claraverseoss/claraverse" - echo "" - exit 0 - fi - - # Install CLI - install_cli - - # Verify - if ! verify_installation; then - exit 1 - fi - - # Check Docker - echo "" - if check_docker; then - log_success "Docker is available" - else - log_warning "Install Docker first, then run: claraverse init" - exit 0 - fi - - echo "" - echo -e "${GREEN}============================================${NC}" - echo -e "${GREEN} ClaraVerse CLI installed successfully!${NC}" - echo -e "${GREEN}============================================${NC}" - echo "" - echo -e "${BOLD}Quick Start:${NC}" - echo " claraverse init # Start ClaraVerse" - echo " claraverse status # Check status" - echo " claraverse help # Show all commands" - echo "" - echo -e "${BOLD}Start ClaraVerse now?${NC}" - read -p "Run 'claraverse init'? (Y/n) " -n 1 -r - echo "" - - if [[ ! $REPLY =~ ^[Nn]$ ]]; then - echo "" - exec claraverse init - else - echo "" - echo "Run 'claraverse init' when ready to start." - fi -} - -main "$@" diff --git a/dev-docker.sh b/dev-docker.sh deleted file mode 100755 index 79a1609c..00000000 --- a/dev-docker.sh +++ /dev/null @@ -1,129 +0,0 @@ -#!/bin/bash -# ============================================================================ -# ClaraVerse Docker Development Environment -# ============================================================================ -# This script starts the development environment with HMR for both -# frontend (Vite) and backend (Go with Air). -# -# Usage: -# ./dev-docker.sh # Start development environment -# ./dev-docker.sh build # Rebuild and start -# ./dev-docker.sh down # Stop all containers -# ./dev-docker.sh clean # Remove all resources except MySQL data -# ./dev-docker.sh logs # View logs -# ./dev-docker.sh logs -f # Follow logs -# ============================================================================ - -set -e - -# Colors for output -GREEN='\033[0;32m' -YELLOW='\033[1;33m' -CYAN='\033[0;36m' -RED='\033[0;31m' -NC='\033[0m' # No Color - -# Check for .env file -if [ ! -f .env ]; then - echo -e "${RED}❌ Error: .env file not found!${NC}" - echo "" - echo -e "${YELLOW}Please create a .env file with the required configuration.${NC}" - echo -e "${YELLOW}You can use .env.example as a template:${NC}" - echo "" - echo -e "${CYAN} cp .env.example .env${NC}" - echo "" - echo -e "${YELLOW}Then edit .env and add your configuration values.${NC}" - echo "" - exit 1 -fi - -# Use 'docker compose' (V2) or 'docker-compose' (V1) -if command -v docker &> /dev/null && docker compose version &> /dev/null; then - COMPOSE="docker compose" -elif command -v docker-compose &> /dev/null; then - COMPOSE="docker-compose" -else - echo "Error: Neither 'docker compose' nor 'docker-compose' found." - echo "Please install Docker with Compose plugin." - exit 1 -fi - -case "${1:-up}" in - up) - echo -e "${CYAN}🚀 Starting ClaraVerse Development Environment...${NC}" - echo -e "${YELLOW} Frontend: http://localhost:5173 (Vite HMR)${NC}" - echo -e "${YELLOW} Backend: http://localhost:3001 (Go Air)${NC}" - echo "" - $COMPOSE -f docker-compose.yml -f docker-compose.dev.yml up - ;; - build) - echo -e "${CYAN}🔨 Rebuilding and starting development environment...${NC}" - $COMPOSE -f docker-compose.yml -f docker-compose.dev.yml up --build - ;; - down) - echo -e "${CYAN}🛑 Stopping development environment...${NC}" - $COMPOSE -f docker-compose.yml -f docker-compose.dev.yml down - ;; - clean) - echo -e "${CYAN}🧹 Cleaning up Docker resources (preserving MySQL data)...${NC}" - echo -e "${YELLOW}⚠️ This will remove:${NC}" - echo -e "${YELLOW} - All containers${NC}" - echo -e "${YELLOW} - All volumes EXCEPT mysql-data-new${NC}" - echo -e "${YELLOW} - Network${NC}" - echo -e "${YELLOW} - Built images${NC}" - echo "" - read -p "Are you sure? (y/N) " -n 1 -r - echo - if [[ $REPLY =~ ^[Yy]$ ]]; then - # Stop and remove containers - $COMPOSE -f docker-compose.yml -f docker-compose.dev.yml down - - # Remove specific volumes (all except mysql-data-new) - echo -e "${CYAN}Removing volumes...${NC}" - docker volume rm claraverse-scarlet_backend-data 2>/dev/null || true - docker volume rm claraverse-scarlet_backend-uploads 2>/dev/null || true - docker volume rm claraverse-scarlet_backend-logs 2>/dev/null || true - docker volume rm claraverse-scarlet_mongodb-data 2>/dev/null || true - docker volume rm claraverse-scarlet_redis-data 2>/dev/null || true - - # Remove old mysql-data volume if it exists - echo -e "${CYAN}Removing old MySQL volume...${NC}" - docker volume rm claraverse-scarlet_mysql-data 2>/dev/null || true - - # Force remove network - echo -e "${CYAN}Removing network...${NC}" - docker network rm claraverse-scarlet_claraverse-network 2>/dev/null || true - - # Remove built images - echo -e "${CYAN}Removing built images...${NC}" - docker rmi claraverse-scarlet-backend 2>/dev/null || true - docker rmi claraverse-scarlet-frontend 2>/dev/null || true - docker rmi claraverse-scarlet-e2b-service 2>/dev/null || true - - echo -e "${GREEN}✅ Cleanup complete! MySQL data preserved in mysql-data-new volume.${NC}" - else - echo -e "${YELLOW}Cleanup cancelled.${NC}" - fi - ;; - logs) - shift - $COMPOSE -f docker-compose.yml -f docker-compose.dev.yml logs "$@" - ;; - restart) - echo -e "${CYAN}🔄 Restarting development environment...${NC}" - $COMPOSE -f docker-compose.yml -f docker-compose.dev.yml restart - ;; - *) - echo "Usage: $0 {up|build|down|clean|logs|restart}" - echo "" - echo "Commands:" - echo " up - Start development environment (default)" - echo " build - Rebuild containers and start" - echo " down - Stop all containers" - echo " clean - Remove all resources except MySQL data" - echo " logs - View container logs (add -f to follow)" - echo " restart - Restart all containers" - exit 1 - ;; -esac - diff --git a/diagnose.bat b/diagnose.bat deleted file mode 100644 index 77cf05bf..00000000 --- a/diagnose.bat +++ /dev/null @@ -1,215 +0,0 @@ -@echo off -REM ============================================ -REM ClaraVerse Diagnostic Tool for Windows -REM ============================================ -REM Run this script to diagnose common issues -REM ============================================ - -setlocal enabledelayedexpansion - -echo ============================================ -echo ClaraVerse Diagnostics -echo ============================================ -echo. - -REM ============================================ -REM System Information -REM ============================================ -echo System Information: -echo OS: Windows -ver -echo. - -REM ============================================ -REM Docker Installation Check -REM ============================================ -echo Docker Installation: - -where docker >nul 2>nul -if %ERRORLEVEL% EQU 0 ( - echo [OK] Docker installed - docker --version -) else ( - echo [ERROR] Docker not found - echo Please install: https://www.docker.com/products/docker-desktop -) - -docker compose version >nul 2>nul -if %ERRORLEVEL% EQU 0 ( - echo [OK] Docker Compose installed - docker compose version -) else ( - echo [ERROR] Docker Compose not found -) - -echo. - -REM ============================================ -REM Docker Service Status -REM ============================================ -echo Docker Service Status: - -docker info >nul 2>nul -if %ERRORLEVEL% EQU 0 ( - echo [OK] Docker daemon is running -) else ( - echo [ERROR] Docker daemon is not running - echo Start Docker Desktop -) - -echo. - -REM ============================================ -REM Environment Configuration -REM ============================================ -echo Environment Configuration: - -if exist .env ( - echo [OK] .env file exists - - REM Check ENCRYPTION_MASTER_KEY - findstr /C:"ENCRYPTION_MASTER_KEY=" .env | findstr /V /C:"ENCRYPTION_MASTER_KEY=$" | findstr /V /C:"auto-generated" >nul - if %ERRORLEVEL% EQU 0 ( - echo [OK] ENCRYPTION_MASTER_KEY is set - ) else ( - echo [ERROR] ENCRYPTION_MASTER_KEY is not set or invalid - ) - - REM Check JWT_SECRET - findstr /C:"JWT_SECRET=" .env | findstr /V /C:"JWT_SECRET=$" | findstr /V /C:"auto-generated" >nul - if %ERRORLEVEL% EQU 0 ( - echo [OK] JWT_SECRET is set - ) else ( - echo [ERROR] JWT_SECRET is not set or invalid - ) - - REM Check MYSQL_PASSWORD - findstr /C:"MYSQL_PASSWORD=" .env | findstr /V /C:"MYSQL_PASSWORD=$" >nul - if %ERRORLEVEL% EQU 0 ( - echo [OK] MYSQL_PASSWORD is set - ) else ( - echo [WARNING] MYSQL_PASSWORD is not set (will use default) - ) -) else ( - echo [ERROR] .env file not found - echo Run quickstart.bat to create it automatically -) - -echo. - -REM ============================================ -REM Container Status -REM ============================================ -echo Container Status: - -docker compose ps >nul 2>nul -if %ERRORLEVEL% EQU 0 ( - docker compose ps - echo. -) else ( - echo [WARNING] No containers running - echo Start with: docker compose up -d -) - -echo. - -REM ============================================ -REM Port Usage Check -REM ============================================ -echo Port Usage Check: -echo Checking if services are listening... - -netstat -an | findstr ":80 " >nul && echo [OK] Port 80 (Frontend) is in use || echo [WARNING] Port 80 (Frontend) is not in use -netstat -an | findstr ":3001 " >nul && echo [OK] Port 3001 (Backend) is in use || echo [WARNING] Port 3001 (Backend) is not in use -netstat -an | findstr ":3306 " >nul && echo [OK] Port 3306 (MySQL) is in use || echo [WARNING] Port 3306 (MySQL) is not in use -netstat -an | findstr ":27017 " >nul && echo [OK] Port 27017 (MongoDB) is in use || echo [WARNING] Port 27017 (MongoDB) is not in use -netstat -an | findstr ":6379 " >nul && echo [OK] Port 6379 (Redis) is in use || echo [WARNING] Port 6379 (Redis) is not in use - -echo. - -REM ============================================ -REM Connectivity Tests -REM ============================================ -echo Connectivity Tests: - -REM Test backend -curl -s http://localhost:3001/health >nul 2>nul -if %ERRORLEVEL% EQU 0 ( - echo [OK] Backend API is responding -) else ( - echo [ERROR] Backend API not responding - echo Check: docker compose logs backend -) - -REM Test frontend -curl -s http://localhost:80 >nul 2>nul -if %ERRORLEVEL% EQU 0 ( - echo [OK] Frontend is responding -) else ( - echo [ERROR] Frontend not responding - echo Check: docker compose logs frontend -) - -echo. - -REM ============================================ -REM Disk Space Check -REM ============================================ -echo Disk Space: -for /f "tokens=3" %%a in ('dir /-c ^| findstr /C:"bytes free"') do set FREESPACE=%%a -echo Free space: %FREESPACE% bytes -echo. - -REM ============================================ -REM Docker Volume Check -REM ============================================ -echo Docker Volumes: -docker volume ls | findstr claraverse -if %ERRORLEVEL% NEQ 0 ( - echo No ClaraVerse volumes found -) -echo. - -REM ============================================ -REM Recent Errors from Logs -REM ============================================ -echo Recent Errors (last 20 lines): - -docker compose ps >nul 2>nul -if %ERRORLEVEL% EQU 0 ( - echo Backend errors: - docker compose logs --tail=20 backend 2>&1 | findstr /I "error" || echo No recent errors - echo. - - echo Frontend errors: - docker compose logs --tail=20 frontend 2>&1 | findstr /I "error" || echo No recent errors - echo. -) else ( - echo [WARNING] No containers running -) - -REM ============================================ -REM Recommendations -REM ============================================ -echo ====================================== -echo Recommendations: -echo ====================================== - -if not exist .env ( - echo - Create .env file: quickstart.bat -) - -docker compose ps >nul 2>nul -if %ERRORLEVEL% NEQ 0 ( - echo - Start services: quickstart.bat -) - -echo. -echo For more help: -echo - View logs: docker compose logs -f -echo - Restart services: docker compose restart -echo - Full restart: docker compose down then quickstart.bat -echo - Report issues: https://github.com/yourusername/ClaraVerse-Scarlet-OSS/issues -echo. - -pause diff --git a/diagnose.sh b/diagnose.sh deleted file mode 100755 index f3fe6df8..00000000 --- a/diagnose.sh +++ /dev/null @@ -1,249 +0,0 @@ -#!/bin/bash -# ============================================ -# ClaraVerse Diagnostic Tool -# ============================================ -# Run this script to diagnose common issues -# ============================================ - -set -e - -echo "🔍 ClaraVerse Diagnostics" -echo "========================" -echo "" - -# Colors -GREEN='\033[0;32m' -YELLOW='\033[1;33m' -RED='\033[0;31m' -BLUE='\033[0;34m' -NC='\033[0m' # No Color - -# ============================================ -# System Information -# ============================================ -echo -e "${BLUE}System Information:${NC}" -echo "OS: $(uname -s) $(uname -r)" -echo "Architecture: $(uname -m)" -echo "" - -# ============================================ -# Docker Installation Check -# ============================================ -echo -e "${BLUE}Docker Installation:${NC}" - -if command -v docker &> /dev/null; then - echo -e "${GREEN}✅ Docker installed${NC}" - docker --version -else - echo -e "${RED}❌ Docker not found${NC}" - echo " Please install: https://www.docker.com/products/docker-desktop" -fi - -if docker compose version &> /dev/null 2>&1; then - echo -e "${GREEN}✅ Docker Compose installed${NC}" - docker compose version -else - echo -e "${RED}❌ Docker Compose not found${NC}" -fi - -echo "" - -# ============================================ -# Docker Service Status -# ============================================ -echo -e "${BLUE}Docker Service Status:${NC}" - -if docker info &> /dev/null; then - echo -e "${GREEN}✅ Docker daemon is running${NC}" -else - echo -e "${RED}❌ Docker daemon is not running${NC}" - echo " Start Docker Desktop or run: sudo systemctl start docker" -fi - -echo "" - -# ============================================ -# Environment Configuration -# ============================================ -echo -e "${BLUE}Environment Configuration:${NC}" - -if [ -f .env ]; then - echo -e "${GREEN}✅ .env file exists${NC}" - - # Check required variables - if grep -q "ENCRYPTION_MASTER_KEY=" .env && ! grep -q "ENCRYPTION_MASTER_KEY=$" .env && ! grep -q "ENCRYPTION_MASTER_KEY=auto-generated" .env; then - echo -e " ${GREEN}✅ ENCRYPTION_MASTER_KEY is set${NC}" - else - echo -e " ${RED}❌ ENCRYPTION_MASTER_KEY is not set or invalid${NC}" - fi - - if grep -q "JWT_SECRET=" .env && ! grep -q "JWT_SECRET=$" .env && ! grep -q "JWT_SECRET=auto-generated" .env; then - echo -e " ${GREEN}✅ JWT_SECRET is set${NC}" - else - echo -e " ${RED}❌ JWT_SECRET is not set or invalid${NC}" - fi - - if grep -q "MYSQL_PASSWORD=" .env && ! grep -q "MYSQL_PASSWORD=$" .env; then - echo -e " ${GREEN}✅ MYSQL_PASSWORD is set${NC}" - else - echo -e " ${YELLOW}⚠️ MYSQL_PASSWORD is not set (will use default)${NC}" - fi -else - echo -e "${RED}❌ .env file not found${NC}" - echo " Run ./quickstart.sh to create it automatically" -fi - -echo "" - -# ============================================ -# Container Status -# ============================================ -echo -e "${BLUE}Container Status:${NC}" - -if docker compose ps &> /dev/null; then - docker compose ps --format "table {{.Service}}\t{{.Status}}\t{{.Health}}" - echo "" - - # Count services - TOTAL=$(docker compose ps --format json 2>/dev/null | jq -s 'length' 2>/dev/null || echo 0) - - if command -v jq &> /dev/null; then - RUNNING=$(docker compose ps --format json 2>/dev/null | jq -s '[.[] | select(.State == "running")] | length' 2>/dev/null || echo 0) - HEALTHY=$(docker compose ps --format json 2>/dev/null | jq -s '[.[] | select(.Health == "healthy")] | length' 2>/dev/null || echo 0) - echo "Summary: $RUNNING/$TOTAL running, $HEALTHY healthy" - else - echo "Summary: (install jq for detailed status)" - fi -else - echo -e "${YELLOW}⚠️ No containers running${NC}" - echo " Start with: docker compose up -d" -fi - -echo "" - -# ============================================ -# Port Usage Check -# ============================================ -echo -e "${BLUE}Port Usage Check:${NC}" - -PORTS=(80 3001 3306 27017 6379 8080) -PORT_NAMES=("Frontend" "Backend" "MySQL" "MongoDB" "Redis" "SearXNG") - -for i in "${!PORTS[@]}"; do - PORT=${PORTS[$i]} - NAME=${PORT_NAMES[$i]} - - if command -v lsof &> /dev/null; then - if lsof -i :$PORT &> /dev/null; then - echo -e " ${GREEN}✅ Port $PORT ($NAME) is in use${NC}" - else - echo -e " ${YELLOW}⚠️ Port $PORT ($NAME) is not in use${NC}" - fi - elif command -v netstat &> /dev/null; then - if netstat -tulpn 2>/dev/null | grep -q ":$PORT "; then - echo -e " ${GREEN}✅ Port $PORT ($NAME) is in use${NC}" - else - echo -e " ${YELLOW}⚠️ Port $PORT ($NAME) is not in use${NC}" - fi - else - echo -e " ${YELLOW}⚠️ Cannot check port $PORT (install lsof or netstat)${NC}" - fi -done - -echo "" - -# ============================================ -# Connectivity Tests -# ============================================ -echo -e "${BLUE}Connectivity Tests:${NC}" - -# Test backend health endpoint -if curl -s http://localhost:3001/health > /dev/null 2>&1; then - HEALTH_RESPONSE=$(curl -s http://localhost:3001/health) - if echo "$HEALTH_RESPONSE" | grep -q "healthy"; then - echo -e " ${GREEN}✅ Backend API is responding (healthy)${NC}" - else - echo -e " ${YELLOW}⚠️ Backend API responding but not healthy${NC}" - echo " Response: $HEALTH_RESPONSE" - fi -else - echo -e " ${RED}❌ Backend API not responding${NC}" - echo " Check: docker compose logs backend" -fi - -# Test frontend -if curl -s http://localhost:80 > /dev/null 2>&1; then - echo -e " ${GREEN}✅ Frontend is responding${NC}" -else - echo -e " ${RED}❌ Frontend not responding${NC}" - echo " Check: docker compose logs frontend" -fi - -echo "" - -# ============================================ -# Disk Space Check -# ============================================ -echo -e "${BLUE}Disk Space:${NC}" -df -h . | tail -1 -echo "" - -# ============================================ -# Docker Volume Check -# ============================================ -echo -e "${BLUE}Docker Volumes:${NC}" -docker volume ls | grep claraverse || echo "No ClaraVerse volumes found" -echo "" - -# ============================================ -# Recent Errors from Logs -# ============================================ -echo -e "${BLUE}Recent Errors (last 50 lines):${NC}" - -if docker compose ps &> /dev/null; then - echo -e "${YELLOW}Backend errors:${NC}" - docker compose logs --tail=50 backend 2>&1 | grep -i "error" | tail -5 || echo " No recent errors" - echo "" - - echo -e "${YELLOW}Frontend errors:${NC}" - docker compose logs --tail=50 frontend 2>&1 | grep -i "error" | tail -5 || echo " No recent errors" - echo "" -else - echo -e "${YELLOW}⚠️ No containers running${NC}" -fi - -# ============================================ -# Recommendations -# ============================================ -echo "======================================" -echo -e "${BLUE}Recommendations:${NC}" -echo "======================================" - -# Check if services are not running -if ! docker compose ps &> /dev/null || [ "$(docker compose ps --format json | wc -l)" -eq 0 ]; then - echo "• Start services: ./quickstart.sh" -fi - -# Check if .env is missing -if [ ! -f .env ]; then - echo "• Create .env file: ./quickstart.sh" -fi - -# Check if not all services are healthy -if command -v jq &> /dev/null && docker compose ps &> /dev/null; then - TOTAL=$(docker compose ps --format json 2>/dev/null | wc -l | tr -d ' ') - HEALTHY=$(docker compose ps --format json 2>/dev/null | jq -r 'select(.Health == "healthy")' 2>/dev/null | wc -l | tr -d ' ') - - if [ "$HEALTHY" -lt 7 ] && [ "$TOTAL" -gt 0 ]; then - echo "• Wait for services to become healthy: docker compose ps" - echo "• Or view logs for unhealthy services: docker compose logs " - fi -fi - -echo "" -echo "For more help:" -echo "• View logs: docker compose logs -f" -echo "• Restart services: docker compose restart" -echo "• Full restart: docker compose down && ./quickstart.sh" -echo "• Report issues: https://github.com/yourusername/ClaraVerse-Scarlet-OSS/issues" -echo "" diff --git a/docker-compose.dev.yml b/docker-compose.dev.yml deleted file mode 100644 index 000662e2..00000000 --- a/docker-compose.dev.yml +++ /dev/null @@ -1,66 +0,0 @@ -services: - # Development frontend with HMR - frontend: - build: - context: ./frontend - dockerfile: Dockerfile - target: builder - volumes: - - ./frontend/src:/app/src - - ./frontend/public:/app/public - - frontend-node-modules:/app/node_modules - ports: - - "5173:5173" - environment: - - CHOKIDAR_USEPOLLING=true - command: npm run dev -- --host 0.0.0.0 - networks: - - claraverse-network - - # Development backend with air hot reload - backend: - build: - context: ./backend - dockerfile: Dockerfile.dev - volumes: - - ./backend:/app - - backend-go-cache:/go/pkg/mod - environment: - - ENVIRONMENT=development - - DATABASE_URL=mysql://claraverse_user:${MYSQL_PASSWORD:-claraverse_pass_2024}@mysql:3306/claraverse?parseTime=true - - MONGODB_URI=mongodb://mongodb:27017/claraverse - - REDIS_URL=redis://redis:6379 - - SEARXNG_URLS=http://searxng:8080 - - E2B_SERVICE_URL=http://e2b-service:8001 - - CGO_ENABLED=1 - depends_on: - mongodb: - condition: service_healthy - mysql: - condition: service_healthy - redis: - condition: service_healthy - searxng: - condition: service_healthy - e2b-service: - condition: service_healthy - networks: - - claraverse-network - ports: - - "3001:3001" - healthcheck: - test: ["CMD", "wget", "--no-verbose", "--tries=1", "--spider", "http://127.0.0.1:3001/health"] - interval: 10s - timeout: 5s - start_period: 60s # Give Air time for initial build - retries: 10 # More retries to handle Air restarts - -volumes: - frontend-node-modules: - driver: local - backend-go-cache: - driver: local - -networks: - claraverse-network: - driver: bridge diff --git a/docker-compose.override.yml b/docker-compose.override.yml deleted file mode 100644 index f91d2735..00000000 --- a/docker-compose.override.yml +++ /dev/null @@ -1,35 +0,0 @@ -# Docker Compose Override for Development -# This file is automatically loaded by docker-compose to override settings for local development -# To use production settings, rename this file or use: docker-compose -f docker-compose.yml up - -services: - backend: - # Mount source code for hot reload (requires a file watcher in the app) - volumes: - # Mount source code (read-only for safety) - - ./backend:/app/src:ro - # Seed file is now in config directory (mounted in main docker-compose.yml) - - environment: - # Development environment - - ENVIRONMENT=development - # Relaxed CORS for local development (includes both Vite ports and port 80) - - ALLOWED_ORIGINS=http://localhost,http://localhost:5173,http://localhost:5174,http://localhost:3000,http://localhost:8080 - - # Remove restart policy for development (easier to debug crashes) - restart: "no" - - # Add labels for easier identification - labels: - - "dev.claraverse.environment=development" - - "dev.claraverse.service=backend" - - # For development, it's recommended to run frontend with `npm run dev` locally - # This provides hot reload and better developer experience - # If you still want to use Docker for frontend in dev mode: - frontend: - # Use port 80 in development (same as production) - restart: "no" - labels: - - "dev.claraverse.environment=development" - - "dev.claraverse.service=frontend" \ No newline at end of file diff --git a/docker-compose.prod.yml b/docker-compose.prod.yml deleted file mode 100644 index 8e25b6c0..00000000 --- a/docker-compose.prod.yml +++ /dev/null @@ -1,186 +0,0 @@ -# ============================================ -# ClaraVerse Production Docker Compose -# ============================================ -# Uses pre-built images from GitHub Container Registry -# No git clone needed - just download this file and run! -# -# Quick start: -# curl -fsSL https://raw.githubusercontent.com/yourusername/ClaraVerse-Scarlet-OSS/main/install.sh | bash -# -# Or manually: -# curl -O https://raw.githubusercontent.com/yourusername/ClaraVerse-Scarlet-OSS/main/docker-compose.prod.yml -# docker compose -f docker-compose.prod.yml up -d -# ============================================ - -services: - mongodb: - image: mongo:7 - container_name: claraverse-mongodb - volumes: - - mongodb-data:/data/db - environment: - - MONGO_INITDB_DATABASE=claraverse - restart: unless-stopped - healthcheck: - test: ["CMD", "mongosh", "--eval", "db.adminCommand('ping')"] - interval: 30s - timeout: 10s - start_period: 30s - retries: 3 - networks: - - claraverse-network - - redis: - image: redis:7-alpine - container_name: claraverse-redis - command: redis-server --appendonly yes --maxmemory 100mb --maxmemory-policy allkeys-lru - volumes: - - redis-data:/data - restart: unless-stopped - healthcheck: - test: ["CMD", "redis-cli", "ping"] - interval: 10s - timeout: 5s - start_period: 5s - retries: 5 - networks: - - claraverse-network - - mysql: - image: mysql:8.0 - container_name: claraverse-mysql - command: --default-authentication-plugin=mysql_native_password --character-set-server=utf8mb4 --collation-server=utf8mb4_unicode_ci - volumes: - - mysql-data:/var/lib/mysql - environment: - - MYSQL_ROOT_PASSWORD=${MYSQL_ROOT_PASSWORD:-claraverse_root_2024} - - MYSQL_DATABASE=claraverse - - MYSQL_USER=claraverse_user - - MYSQL_PASSWORD=${MYSQL_PASSWORD:-claraverse_pass_2024} - restart: unless-stopped - healthcheck: - test: ["CMD", "mysql", "-h", "127.0.0.1", "-u", "claraverse_user", "-pclaraverse_pass_2024", "claraverse", "-e", "SELECT 1"] - interval: 5s - timeout: 5s - start_period: 60s - retries: 12 - networks: - - claraverse-network - - searxng: - image: searxng/searxng:latest - container_name: claraverse-searxng - environment: - - SEARXNG_BASE_URL=http://searxng:8080/ - - SEARXNG_HOSTNAME=searxng - restart: unless-stopped - networks: - - claraverse-network - healthcheck: - test: ["CMD", "wget", "--no-verbose", "--tries=1", "--spider", "http://localhost:8080/"] - interval: 30s - timeout: 3s - start_period: 15s - retries: 3 - - e2b-service: - image: ghcr.io/yourusername/claraverse-e2b:latest - container_name: claraverse-e2b - environment: - - E2B_MODE=${E2B_MODE:-local} - - E2B_LOCAL_USE_DOCKER=${E2B_LOCAL_USE_DOCKER:-true} - - E2B_SANDBOX_POOL_SIZE=${E2B_SANDBOX_POOL_SIZE:-3} - - E2B_EXECUTION_TIMEOUT=${E2B_EXECUTION_TIMEOUT:-30000} - - E2B_RATE_LIMIT_PER_MIN=${E2B_RATE_LIMIT_PER_MIN:-20} - volumes: - - /var/run/docker.sock:/var/run/docker.sock - restart: unless-stopped - healthcheck: - test: ["CMD", "python", "-c", "import requests; requests.get('http://localhost:8001/health')"] - interval: 30s - timeout: 10s - start_period: 5s - retries: 3 - networks: - - claraverse-network - - backend: - image: ghcr.io/yourusername/claraverse-backend:latest - container_name: claraverse-backend - ports: - - "3001:3001" - volumes: - - backend-data:/app/data - - backend-uploads:/app/uploads - - backend-logs:/app/logs - environment: - - ENVIRONMENT=${ENVIRONMENT:-development} - - DATABASE_URL=mysql://claraverse_user:${MYSQL_PASSWORD:-claraverse_pass_2024}@mysql:3306/claraverse?parseTime=true - - MYSQL_DATABASE=claraverse - - UPLOAD_DIR=/app/uploads - - SUPABASE_URL=${SUPABASE_URL} - - SUPABASE_KEY=${SUPABASE_KEY} - - ALLOWED_ORIGINS=${ALLOWED_ORIGINS:-http://localhost,http://localhost:80} - - SEARXNG_URLS=http://searxng:8080 - - SEARXNG_URL=http://searxng:8080 - - E2B_SERVICE_URL=http://e2b-service:8001 - - FRONTEND_URL=${FRONTEND_URL:-http://localhost:80} - - BACKEND_URL=${BACKEND_URL:-http://localhost:3001} - - MONGODB_URI=mongodb://mongodb:27017/claraverse - - REDIS_URL=redis://redis:6379 - - ENCRYPTION_MASTER_KEY=${ENCRYPTION_MASTER_KEY} - - JWT_SECRET=${JWT_SECRET} - - JWT_ACCESS_TOKEN_EXPIRY=${JWT_ACCESS_TOKEN_EXPIRY:-15m} - - JWT_REFRESH_TOKEN_EXPIRY=${JWT_REFRESH_TOKEN_EXPIRY:-168h} - - DEV_API_KEY=${DEV_API_KEY:-claraverse-dev-key-2024} - - SUPERADMIN_USER_IDS=${SUPERADMIN_USER_IDS:-} - depends_on: - mongodb: - condition: service_healthy - mysql: - condition: service_healthy - searxng: - condition: service_healthy - e2b-service: - condition: service_healthy - redis: - condition: service_healthy - restart: unless-stopped - healthcheck: - test: ["CMD", "wget", "--no-verbose", "--tries=1", "--spider", "http://localhost:3001/health"] - interval: 30s - timeout: 3s - start_period: 10s - retries: 3 - networks: - - claraverse-network - - frontend: - image: ghcr.io/yourusername/claraverse-frontend:latest - container_name: claraverse-frontend - ports: - - "80:80" - depends_on: - backend: - condition: service_healthy - restart: unless-stopped - healthcheck: - test: ["CMD", "wget", "--no-verbose", "--tries=1", "--spider", "http://127.0.0.1/health"] - interval: 30s - timeout: 3s - start_period: 5s - retries: 3 - networks: - - claraverse-network - -volumes: - backend-data: - backend-uploads: - backend-logs: - mongodb-data: - mysql-data: - redis-data: - -networks: - claraverse-network: - driver: bridge diff --git a/docker-compose.production.yml b/docker-compose.production.yml deleted file mode 100644 index 1f5d9760..00000000 --- a/docker-compose.production.yml +++ /dev/null @@ -1,36 +0,0 @@ -# Docker Compose Production Override -# ============================================================================ -# Usage: docker-compose -f docker-compose.yml -f docker-compose.production.yml up --build -d -# -# IMPORTANT: All secrets and configuration values come from the root .env file. -# Before deploying, update .env with production values: -# - ENVIRONMENT=production -# - VITE_API_BASE_URL=https://api.yourdomain.com -# - VITE_WS_URL=wss://api.yourdomain.com -# - ALLOWED_ORIGINS=https://yourdomain.com -# - FRONTEND_URL=https://yourdomain.com -# - etc. -# -# Architecture: -# claraverse.app → Frontend (port 80) -# api.claraverse.app → Backend (port 3001) -# ============================================================================ - -services: - mongodb: - restart: unless-stopped - - redis: - restart: unless-stopped - - searxng: - restart: unless-stopped - - e2b-service: - restart: unless-stopped - - backend: - restart: unless-stopped - - frontend: - restart: unless-stopped diff --git a/docker-compose.yml b/docker-compose.yml deleted file mode 100644 index a8a76fb9..00000000 --- a/docker-compose.yml +++ /dev/null @@ -1,243 +0,0 @@ -services: - # MongoDB (for conversations, workflows, and data persistence) - mongodb: - image: mongo:7 - container_name: claraverse-mongodb - ports: - - "27017:27017" - volumes: - - mongodb-data:/data/db - environment: - - MONGO_INITDB_DATABASE=claraverse - restart: unless-stopped - healthcheck: - test: ["CMD", "mongosh", "--eval", "db.adminCommand('ping')"] - interval: 30s - timeout: 10s - start_period: 30s - retries: 3 - networks: - - claraverse-network - - # Redis (for scheduled jobs + WebSocket pub/sub) - # Required for: scheduled agent triggers, cross-instance messaging - redis: - image: redis:7-alpine - container_name: claraverse-redis - command: redis-server --appendonly yes --maxmemory 100mb --maxmemory-policy allkeys-lru - ports: - - "6379:6379" - volumes: - - redis-data:/data - restart: unless-stopped - healthcheck: - test: ["CMD", "redis-cli", "ping"] - interval: 10s - timeout: 5s - start_period: 5s - retries: 5 - networks: - - claraverse-network - - # MySQL (for providers, models, and capabilities) - # Replaces SQLite for production-ready persistence - mysql: - image: mysql:8.0 - container_name: claraverse-mysql - command: --default-authentication-plugin=mysql_native_password --character-set-server=utf8mb4 --collation-server=utf8mb4_unicode_ci - ports: - - "3306:3306" - volumes: - - mysql-data-new:/var/lib/mysql - - ./backend/migrations:/docker-entrypoint-initdb.d:ro - environment: - - MYSQL_ROOT_PASSWORD=${MYSQL_ROOT_PASSWORD:-claraverse_root_2024} - - MYSQL_DATABASE=claraverse - - MYSQL_USER=claraverse_user - - MYSQL_PASSWORD=${MYSQL_PASSWORD:-claraverse_pass_2024} - restart: unless-stopped - healthcheck: - test: ["CMD", "mysql", "-h", "127.0.0.1", "-u", "claraverse_user", "-pclaraverse_pass_2024", "claraverse", "-e", "SELECT 1"] - interval: 5s - timeout: 5s - start_period: 60s - retries: 12 - networks: - - claraverse-network - - searxng: - image: searxng/searxng:latest - container_name: claraverse-searxng - volumes: - - ./searxng/settings.yml:/etc/searxng/settings.yml:ro - environment: - - SEARXNG_BASE_URL=http://searxng:8080/ - - SEARXNG_HOSTNAME=searxng - cap_add: - - SETUID - - SETGID - restart: unless-stopped - networks: - - claraverse-network - healthcheck: - test: ["CMD", "wget", "--no-verbose", "--tries=1", "--spider", "http://localhost:8080/"] - interval: 30s - timeout: 3s - start_period: 15s - retries: 3 - - e2b-service: - build: - context: ./backend/e2b-service - dockerfile: Dockerfile - container_name: claraverse-e2b - environment: - # E2B Local Mode Configuration (v2.0 - no API key needed!) - - E2B_MODE=${E2B_MODE:-local} - - E2B_LOCAL_USE_DOCKER=${E2B_LOCAL_USE_DOCKER:-true} - - E2B_SANDBOX_POOL_SIZE=${E2B_SANDBOX_POOL_SIZE:-3} - - E2B_EXECUTION_TIMEOUT=${E2B_EXECUTION_TIMEOUT:-30000} - - E2B_RATE_LIMIT_PER_MIN=${E2B_RATE_LIMIT_PER_MIN:-20} - - # Optional: E2B Cloud Mode (if user has API key) - # Uncomment these and comment out E2B_MODE=local above: - # - E2B_MODE=production - # - E2B_API_KEY=${E2B_API_KEY} - volumes: - - /var/run/docker.sock:/var/run/docker.sock # Docker-in-Docker for local mode - restart: unless-stopped - healthcheck: - test: ["CMD", "python", "-c", "import requests; requests.get('http://localhost:8001/health')"] - interval: 30s - timeout: 10s - start_period: 5s - retries: 3 - networks: - - claraverse-network - - backend: - build: - context: ./backend - dockerfile: Dockerfile - args: - SKIP_TESTS: ${SKIP_TESTS:-false} - container_name: claraverse-backend - ports: - - "3001:3001" - volumes: - # Persistent data - - backend-data:/app/data - - backend-uploads:/app/uploads - - backend-logs:/app/logs - environment: - # All values from root .env file (single source of truth) - - ENVIRONMENT=${ENVIRONMENT:-development} - # Override paths for Docker environment - - DATABASE_URL=mysql://claraverse_user:${MYSQL_PASSWORD:-claraverse_pass_2024}@mysql:3306/claraverse?parseTime=true - - MYSQL_DATABASE=claraverse - - UPLOAD_DIR=/app/uploads - # Supabase Authentication - - SUPABASE_URL=${SUPABASE_URL} - - SUPABASE_KEY=${SUPABASE_KEY} - # CORS - - ALLOWED_ORIGINS=${ALLOWED_ORIGINS:-http://localhost:5173} - # SearXNG integration (supports multiple URLs for load balancing) - - SEARXNG_URLS=${SEARXNG_URLS:-http://searxng:8080} - - SEARXNG_URL=${SEARXNG_URLS:-http://searxng:8080} - # E2B service integration - - E2B_SERVICE_URL=http://e2b-service:8001 - # Frontend URL (for payment redirects) - - FRONTEND_URL=${FRONTEND_URL:-http://localhost:5173} - # Backend public URL for generating absolute download URLs - - BACKEND_URL=${BACKEND_URL:-http://localhost:3001} - # MongoDB - - MONGODB_URI=${MONGODB_URI:-mongodb://mongodb:27017/claraverse} - # Redis (for scheduler + pub/sub) - - REDIS_URL=${REDIS_URL:-redis://redis:6379} - # Encryption key for user data - - ENCRYPTION_MASTER_KEY=${ENCRYPTION_MASTER_KEY} - # JWT Authentication (v2.0) - - JWT_SECRET=${JWT_SECRET} - - JWT_ACCESS_TOKEN_EXPIRY=${JWT_ACCESS_TOKEN_EXPIRY:-15m} - - JWT_REFRESH_TOKEN_EXPIRY=${JWT_REFRESH_TOKEN_EXPIRY:-168h} - # Development API key for testing (only works when ENVIRONMENT != production) - - DEV_API_KEY=${DEV_API_KEY:-claraverse-dev-key-2024} - # Admin - - SUPERADMIN_USER_IDS=${SUPERADMIN_USER_IDS:-} - - - COMPOSIO_API_KEY=${COMPOSIO_API_KEY} - - COMPOSIO_GOOGLESHEETS_AUTH_CONFIG_ID=${COMPOSIO_GOOGLESHEETS_AUTH_CONFIG_ID} - - COMPOSIO_GMAIL_AUTH_CONFIG_ID=${COMPOSIO_GMAIL_AUTH_CONFIG_ID} - # Rate Limiting - - RATE_LIMIT_GLOBAL_API=${RATE_LIMIT_GLOBAL_API:-200} - - RATE_LIMIT_PUBLIC_READ=${RATE_LIMIT_PUBLIC_READ:-120} - - RATE_LIMIT_AUTHENTICATED=${RATE_LIMIT_AUTHENTICATED:-60} - - RATE_LIMIT_WEBSOCKET=${RATE_LIMIT_WEBSOCKET:-20} - - RATE_LIMIT_IMAGE_PROXY=${RATE_LIMIT_IMAGE_PROXY:-60} - - VITE_TURNSTILE_SITE_KEY=${VITE_TURNSTILE_SITE_KEY} - depends_on: - mongodb: - condition: service_healthy - mysql: - condition: service_healthy - searxng: - condition: service_healthy - e2b-service: - condition: service_healthy - redis: - condition: service_healthy - restart: unless-stopped - healthcheck: - test: ["CMD", "wget", "--no-verbose", "--tries=1", "--spider", "http://localhost:3001/health"] - interval: 30s - timeout: 3s - start_period: 10s - retries: 3 - networks: - - claraverse-network - extra_hosts: - - "host.docker.internal:host-gateway" - - frontend: - build: - context: ./frontend - dockerfile: Dockerfile - args: - VITE_SUPABASE_URL: ${VITE_SUPABASE_URL} - VITE_SUPABASE_ANON_KEY: ${VITE_SUPABASE_ANON_KEY} - VITE_API_BASE_URL: ${VITE_API_BASE_URL:-http://localhost:3001} - VITE_WS_URL: ${VITE_WS_URL:-ws://localhost:3001} - VITE_TURNSTILE_SITE_KEY: ${VITE_TURNSTILE_SITE_KEY} - container_name: claraverse-frontend - ports: - - "80:80" - depends_on: - backend: - condition: service_healthy - restart: unless-stopped - healthcheck: - test: ["CMD", "wget", "--no-verbose", "--tries=1", "--spider", "http://127.0.0.1/health"] - interval: 30s - timeout: 3s - start_period: 5s - retries: 3 - networks: - - claraverse-network - -volumes: - backend-data: - driver: local - backend-uploads: - driver: local - backend-logs: - driver: local - mongodb-data: - driver: local - mysql-data-new: - driver: local - redis-data: - driver: local - -networks: - claraverse-network: - driver: bridge diff --git a/docker-publish.sh b/docker-publish.sh new file mode 100755 index 00000000..a9050505 --- /dev/null +++ b/docker-publish.sh @@ -0,0 +1,22 @@ +#!/bin/bash + +# Check if DOCKER_USERNAME is set +if [ -z "$DOCKER_USERNAME" ]; then + echo "Please set DOCKER_USERNAME environment variable" + echo "Usage: DOCKER_USERNAME=yourusername ./docker-publish.sh" + exit 1 +fi + +# Build the image +docker build -t $DOCKER_USERNAME/clara-ollama:latest . + +# Login to Docker Hub +docker login + +# Push the image +docker push $DOCKER_USERNAME/clara-ollama:latest + +echo "Successfully published to Docker Hub as $DOCKER_USERNAME/clara-ollama:latest" +echo "Users can now pull and run using:" +echo "docker pull $DOCKER_USERNAME/clara-ollama:latest" +echo "docker run -p 8069:8069 $DOCKER_USERNAME/clara-ollama:latest" diff --git a/docker/all-in-one/Dockerfile b/docker/all-in-one/Dockerfile deleted file mode 100644 index a6db21dd..00000000 --- a/docker/all-in-one/Dockerfile +++ /dev/null @@ -1,127 +0,0 @@ -# ============================================ -# ClaraVerse All-in-One Docker Image -# ============================================ -# Usage: docker run -d -p 80:80 -v claraverse-data:/data claraverseoss/claraverse -# ============================================ - -# ============================================ -# Stage 1: Build Backend (Go) -# ============================================ -FROM golang:1.25.5-alpine AS backend-builder - -RUN apk add --no-cache gcc musl-dev - -WORKDIR /build -COPY backend/go.mod backend/go.sum ./ -RUN go mod download - -COPY backend/ . -RUN CGO_ENABLED=1 GOOS=linux go build -ldflags "-s -w" -o claraverse ./cmd/server - -# ============================================ -# Stage 2: Build Frontend (React) -# ============================================ -FROM node:20-alpine AS frontend-builder - -WORKDIR /app -COPY frontend/package.json frontend/package-lock.json ./ -RUN npm ci --legacy-peer-deps - -COPY frontend/ . -ENV VITE_API_BASE_URL="" -ENV VITE_WS_URL="" -# Skip type checking - use vite directly like main Dockerfile -RUN npx vite build - -# ============================================ -# Stage 3: Final All-in-One Image -# ============================================ -FROM ubuntu:22.04 - -LABEL maintainer="ClaraVerse Team" -LABEL description="ClaraVerse All-in-One - Your Private AI Workspace" - -ENV DEBIAN_FRONTEND=noninteractive -ENV TZ=UTC - -# Install dependencies + musl for Alpine-built Go binary -# Includes Chromium and fonts for PDF generation (emoji, CJK, symbols) -RUN apt-get update && apt-get install -y \ - supervisor \ - nginx \ - gnupg curl wget \ - mysql-server \ - redis-server \ - python3 python3-pip \ - ca-certificates tzdata xxd \ - chromium-browser \ - fonts-liberation \ - fonts-noto \ - fonts-noto-color-emoji \ - fonts-noto-cjk \ - fontconfig \ - musl \ - && rm -rf /var/lib/apt/lists/* \ - && ln -s /usr/lib/x86_64-linux-musl/libc.so /lib/libc.musl-x86_64.so.1 \ - && fc-cache -f -v - -# Install MongoDB 7 -RUN curl -fsSL https://www.mongodb.org/static/pgp/server-7.0.asc | gpg --dearmor -o /usr/share/keyrings/mongodb-server-7.0.gpg \ - && echo "deb [ arch=amd64,arm64 signed-by=/usr/share/keyrings/mongodb-server-7.0.gpg ] https://repo.mongodb.org/apt/ubuntu jammy/mongodb-org/7.0 multiverse" > /etc/apt/sources.list.d/mongodb-org-7.0.list \ - && apt-get update \ - && apt-get install -y mongodb-org \ - && rm -rf /var/lib/apt/lists/* - -# Create directories -RUN mkdir -p \ - /data/mongodb /data/mysql /data/redis /data/uploads /data/logs /data/backend \ - /app/backend /app/frontend /app/e2b \ - /var/log/supervisor /run/mysqld \ - && chown -R mysql:mysql /data/mysql /run/mysqld - -# Copy backend binary -COPY --from=backend-builder /build/claraverse /app/backend/claraverse -COPY backend/providers.example.json /app/backend/ - -# Copy migrations for MySQL init -COPY backend/migrations/ /app/migrations/ - -# Copy frontend build -COPY --from=frontend-builder /app/dist /app/frontend - -# Copy E2B service (uses simple executor - no E2B API key required) -COPY backend/e2b-service/requirements.txt /app/e2b/ -RUN pip3 install --no-cache-dir -r /app/e2b/requirements.txt -COPY backend/e2b-service/main.py /app/e2b/ -COPY backend/e2b-service/main_simple.py /app/e2b/ - -# Copy configs -COPY docker/all-in-one/nginx.conf /etc/nginx/sites-available/default -COPY docker/all-in-one/supervisord.conf /etc/supervisor/conf.d/claraverse.conf -COPY docker/all-in-one/start.sh /app/start.sh -RUN chmod +x /app/start.sh - -# Chromium environment for PDF generation -ENV CHROME_BIN=/usr/bin/chromium-browser -ENV CHROME_PATH=/usr/lib/chromium-browser/ - -# Environment defaults -# Note: Search functionality requires external SearXNG. Set SEARXNG_URL to enable. -ENV ENVIRONMENT=production -ENV DATABASE_URL=mysql://claraverse_user:claraverse_pass@127.0.0.1:3306/claraverse?parseTime=true -ENV MONGODB_URI=mongodb://127.0.0.1:27017/claraverse -ENV REDIS_URL=redis://127.0.0.1:6379 -ENV FRONTEND_URL=http://localhost -ENV BACKEND_URL=http://localhost -ENV ALLOWED_ORIGINS=http://localhost,http://localhost:80 -ENV E2B_SERVICE_URL=http://localhost:8001 -ENV SEARXNG_URL="" -ENV SEARXNG_URLS="" - -VOLUME ["/data"] -EXPOSE 80 - -HEALTHCHECK --interval=30s --timeout=10s --start-period=90s --retries=3 \ - CMD curl -sf http://localhost/health || exit 1 - -CMD ["/app/start.sh"] diff --git a/docker/all-in-one/nginx.conf b/docker/all-in-one/nginx.conf deleted file mode 100644 index d109c4fb..00000000 --- a/docker/all-in-one/nginx.conf +++ /dev/null @@ -1,50 +0,0 @@ -server { - listen 80; - server_name _; - - root /app/frontend; - index index.html; - - # Frontend static files - location / { - try_files $uri $uri/ /index.html; - } - - # Backend API proxy - location /api/ { - proxy_pass http://127.0.0.1:3001; - proxy_http_version 1.1; - proxy_set_header Upgrade $http_upgrade; - proxy_set_header Connection "upgrade"; - proxy_set_header Host $host; - proxy_set_header X-Real-IP $remote_addr; - proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; - proxy_set_header X-Forwarded-Proto $scheme; - proxy_read_timeout 300s; - proxy_connect_timeout 75s; - } - - # WebSocket proxy - location /ws { - proxy_pass http://127.0.0.1:3001; - proxy_http_version 1.1; - proxy_set_header Upgrade $http_upgrade; - proxy_set_header Connection "upgrade"; - proxy_set_header Host $host; - proxy_read_timeout 86400; - } - - # Health check endpoint - location /health { - proxy_pass http://127.0.0.1:3001/health; - } - - # Uploads - location /uploads/ { - alias /data/uploads/; - } - - # Gzip - gzip on; - gzip_types text/plain text/css application/json application/javascript text/xml application/xml; -} diff --git a/docker/all-in-one/start.sh b/docker/all-in-one/start.sh deleted file mode 100644 index 80ca9acd..00000000 --- a/docker/all-in-one/start.sh +++ /dev/null @@ -1,113 +0,0 @@ -#!/bin/bash -# ============================================ -# ClaraVerse All-in-One Startup Script -# ============================================ - -set -e - -echo "============================================" -echo " ClaraVerse All-in-One Starting..." -echo "============================================" - -# ============================================ -# Generate encryption keys if not provided -# ============================================ -if [ -z "$ENCRYPTION_MASTER_KEY" ]; then - echo "Generating ENCRYPTION_MASTER_KEY..." - export ENCRYPTION_MASTER_KEY=$(head -c 32 /dev/urandom | xxd -p -c 32) - echo " Key generated (stored in container - use -e to persist)" -fi - -if [ -z "$JWT_SECRET" ]; then - echo "Generating JWT_SECRET..." - export JWT_SECRET=$(head -c 64 /dev/urandom | xxd -p -c 64) - echo " Key generated (stored in container - use -e to persist)" -fi - -# ============================================ -# Initialize data directories -# ============================================ -echo "Initializing data directories..." -mkdir -p /data/mongodb /data/mysql /data/redis /data/uploads /data/logs /data/backend -chown -R mysql:mysql /data/mysql /run/mysqld 2>/dev/null || true - -# ============================================ -# Initialize MySQL if needed -# ============================================ -if [ ! -d "/data/mysql/mysql" ]; then - echo "Initializing MySQL database..." - mysqld --initialize-insecure --user=mysql --datadir=/data/mysql - - # Start MySQL temporarily to create user/database - mysqld --user=mysql --datadir=/data/mysql & - MYSQL_PID=$! - - # Wait for MySQL to be ready - for i in {1..30}; do - if mysqladmin ping -h 127.0.0.1 --silent 2>/dev/null; then - break - fi - sleep 1 - done - - # Create database and user - mysql -h 127.0.0.1 </dev/null || true - wait $MYSQL_PID 2>/dev/null || true -fi - -# ============================================ -# Initialize MongoDB if needed -# ============================================ -if [ ! -f "/data/mongodb/WiredTiger" ]; then - echo "MongoDB will initialize on first start..." -fi - -# ============================================ -# Update MySQL config for proper operation -# ============================================ -cat > /etc/mysql/mysql.conf.d/docker.cnf < **Your Complete AI Development Suite - 100% Local, Privacy-First** + +Welcome to ClaraVerse, the most comprehensive local AI development environment that brings together all the tools you need for AI-powered development, automation, and creativity - without ever sending your data to the cloud. + +## 🚀 What is ClaraVerse? + +ClaraVerse is not just another AI chat interface. It's a complete AI superstack that runs entirely on your machine, providing: + +- **🤖 Advanced AI Assistant** - Clara, your intelligent coding companion +- **🎨 Web Development Suite** - LumaUI for full-stack development +- **🔧 Agent Builder** - Create and deploy custom AI agents +- **🖼️ Image Generation** - Local Stable Diffusion with ComfyUI +- **⚡ Workflow Automation** - N8N integration for complex automations +- **📊 Smart Dashboard** - Customizable workspace with intelligent widgets +- **📓 Jupyter Notebooks** - Interactive development environment +- **🖥️ Server Management** - Docker and service orchestration + +## 🎯 Core Features Overview + +### 🏠 Dashboard - Your Command Center +The Dashboard is your customizable home base featuring: + +- **📊 Dynamic Widgets** - Resizable, draggable widget system powered by react-grid-layout +- **🔗 Quick Actions** - Direct access to all major features +- **📈 System Status** - Real-time monitoring of Ollama, Python backend, and services +- **🎨 Personalization** - Custom wallpapers, themes, and layout preferences +- **📧 Email Widget** - Integrated communication tools +- **💬 Quick Chat** - Instant AI conversations without leaving the dashboard +- **🔄 Flow Widgets** - Visual workflow representations + +**Key Capabilities:** +```typescript +// Widget types available: +- Welcome Widget: Personalized greeting and quick start +- Privacy Widget: Data security and local processing info +- What's New: Latest features and updates +- Email Widget: Communication integration +- Quick Chat: Instant AI assistance +- Flow Widget: Workflow visualization +- Webhook Widget: External service integration +``` + +### 🤖 Clara Assistant - Your AI Coding Companion + +Clara is an advanced AI assistant with autonomous capabilities: + +**🧠 Intelligent Features:** +- **Multi-Provider Support** - OpenAI, Anthropic, Ollama, OpenRouter +- **Autonomous Execution** - Can work independently with tool calling +- **File Processing** - Handle documents, images, code files +- **Voice Integration** - Text-to-Speech and Speech-to-Text +- **Background Operation** - Continue working while you focus elsewhere +- **Artifact Generation** - Create interactive charts, diagrams, and visualizations + +**🛠️ Advanced Capabilities:** +```typescript +// Clara's toolkit includes: +- Code analysis and generation +- File system operations +- Web research and data processing +- Image analysis and description +- Document summarization +- Real-time collaboration +- MCP (Model Context Protocol) integration +- Custom system prompts and behavior modification +``` + +**🎨 Artifact System:** +Clara can automatically generate: +- Interactive charts and graphs +- Mermaid diagrams and flowcharts +- Data visualizations +- Code examples with syntax highlighting +- HTML previews and demos +- CSV data tables + +### 💻 LumaUI - Full-Stack Web Development + +LumaUI is a complete web development environment powered by WebContainer: + +**🌐 Development Environment:** +- **Monaco Editor** - VS Code-quality editing experience +- **Live Preview** - Real-time preview with hot reloading +- **Terminal Integration** - Full terminal access with npm, git, etc. +- **File Manager** - VS Code-like file explorer with context menus +- **Project Templates** - React, Vue, Vanilla JS, and more +- **WebContainer** - Browser-based Node.js runtime + +**🤖 AI-Powered Development:** +- **Smart Scaffolding** - AI generates complete project structures +- **Tool Integration** - AI can create, edit, and manage files +- **Chat Integration** - Get help while coding +- **Autonomous Building** - AI can build entire applications +- **Error Resolution** - Intelligent debugging and fixes + +**📁 Project Management:** +```typescript +// Supported project types: +- React + Vite + Tailwind CSS +- Vue.js applications +- Vanilla JavaScript projects +- Node.js backend services +- Static websites +- Custom AI-generated applications +``` + +### 🔧 Agent Studio - Build Custom AI Agents + +Create, manage, and deploy custom AI agents: + +**🎯 Agent Creation:** +- **Visual Flow Builder** - Drag-and-drop agent design +- **Custom Workflows** - Define complex automation sequences +- **Tool Integration** - Connect to external APIs and services +- **Conditional Logic** - Smart decision-making capabilities +- **State Management** - Persistent agent memory + +**🚀 Agent Management:** +- **Agent Library** - Store and organize your agents +- **Version Control** - Track agent evolution +- **Testing Environment** - Safe agent testing +- **Deployment Tools** - Deploy agents to production +- **Performance Monitoring** - Track agent performance + +### 🎨 Image Generation - Local Stable Diffusion + +Powerful image generation with complete privacy: + +**🖼️ Generation Features:** +- **ComfyUI Integration** - Advanced workflow-based generation +- **Multiple Models** - SDXL, SD 1.5, Flux, and custom models +- **LoRA Support** - Fine-tuned model variations +- **ControlNet** - Precise image control +- **Upscaling** - Enhance image resolution +- **Batch Generation** - Generate multiple images + +**🎨 Creative Tools:** +```typescript +// Available features: +- Text-to-Image generation +- Image-to-Image transformation +- Inpainting and outpainting +- Style transfer +- Custom model training +- Prompt enhancement with AI +- Automatic parameter optimization +``` + +### ⚡ N8N Automation - Workflow Powerhouse + +Integrated workflow automation platform: + +**🔄 Automation Features:** +- **1000+ Integrations** - Connect to any service +- **Visual Workflow Builder** - No-code automation +- **Webhook Support** - Real-time triggers +- **Data Processing** - Transform and manipulate data +- **Conditional Logic** - Smart workflow routing +- **Error Handling** - Robust error recovery + +**🔗 Integration Examples:** +- GitHub to Slack notifications +- Email processing automation +- Data synchronization +- API orchestration +- File processing pipelines +- Social media automation + +### 📓 Notebooks - Interactive Development + +Jupyter notebook integration for data science and exploration: + +**📊 Data Science Tools:** +- **Interactive Computing** - Live code execution +- **Data Visualization** - Charts, graphs, and plots +- **Library Support** - NumPy, Pandas, Matplotlib, etc. +- **Markdown Support** - Rich documentation +- **Export Options** - Multiple output formats + +## 🛡️ Privacy & Security + +ClaraVerse is built with privacy as a core principle: + +**🔒 Local-First Architecture:** +- **No Cloud Dependencies** - Everything runs on your machine +- **Data Sovereignty** - Your data never leaves your computer +- **Offline Capable** - Works without internet connection +- **Encrypted Storage** - Local data protection +- **Optional Cloud** - Choose when to use external APIs + +**🛡️ Security Features:** +- **Sandboxed Execution** - Safe code execution environments +- **Process Isolation** - Secure container separation +- **Permission Controls** - Granular access management +- **Audit Logging** - Track all system activities + +## 🚀 Getting Started + +### 📋 Prerequisites + +**System Requirements:** +- **OS**: Windows 10+, macOS 10.15+, or Linux +- **RAM**: 8GB minimum, 16GB recommended +- **Storage**: 5GB available space +- **GPU**: Optional but recommended for image generation + +**For AI Features:** +- **Local Models**: Ollama for local LLM inference +- **Cloud Providers**: Optional OpenAI, Anthropic, or OpenRouter API keys + +### ⚡ Quick Start + +1. **Download ClaraVerse** from the releases page +2. **Install** the application for your platform +3. **Launch** ClaraVerse and complete the onboarding +4. **Configure** your AI providers in Settings +5. **Start Creating** - Choose your first project or chat with Clara! + +### 🎯 First Steps + +**For Developers:** +1. Open **LumaUI** → Create a new React project +2. Use **Smart Scaffolding** to generate a complete app +3. Chat with **Clara** for coding assistance + +**For Automation:** +1. Open **N8N** → Create your first workflow +2. Use **Agent Studio** to build custom agents +3. Automate your daily tasks + +**For Creators:** +1. Open **Image Generation** → Generate your first AI image +2. Use **Gallery** to organize your creations +3. Experiment with different models and styles + +## 🎨 Advanced Features + +### 🧪 Alpha Features + +Experimental features for early adopters: + +- **Advanced Agent Orchestration** - Multi-agent systems +- **Custom Model Training** - Train your own AI models +- **Plugin Architecture** - Extend ClaraVerse with custom plugins +- **Team Collaboration** - Share projects and agents +- **Cloud Synchronization** - Optional cloud backup + +### 🔧 Customization + +**Theming:** +- Light and dark mode support +- Custom color schemes +- Wallpaper customization +- Widget personalization + +**Configuration:** +- Custom system prompts for Clara +- Provider-specific settings +- Keyboard shortcuts +- Startup preferences + +## 🤝 Community & Support + +### 📚 Resources + +- **Documentation**: Comprehensive guides and tutorials +- **Discord Community**: Join our active community +- **GitHub**: Contribute to the project +- **Reddit**: Discussion and support + +### 🐛 Getting Help + +- **Built-in Help**: Press F1 or visit the Help section +- **Debug Tools**: Advanced debugging and diagnostics +- **Community Support**: Discord and Reddit communities +- **Issue Tracking**: GitHub issues for bug reports + +### 🎯 Contributing + +ClaraVerse is open source and welcomes contributions: + +- **Code Contributions**: Features, bug fixes, improvements +- **Documentation**: Help improve our docs +- **Testing**: Beta testing and feedback +- **Community**: Help other users + +## 🔮 Roadmap + +### 🚀 Coming Soon + +- **Mobile Companion App** - Android and iOS apps for remote access +- **Cloud Deployment** - Optional cloud hosting for teams +- **Advanced Analytics** - Usage insights and optimization +- **Plugin Marketplace** - Community-contributed extensions +- **Team Features** - Collaboration and sharing tools + +### 🎯 Future Vision + +ClaraVerse aims to be the complete local AI development environment, providing everything you need to build, deploy, and manage AI-powered applications without compromising on privacy or requiring expensive cloud services. + +--- + +## 🌟 Why Choose ClaraVerse? + +- **🔒 Privacy-First** - Your data stays on your machine +- **💰 Cost-Effective** - No expensive cloud bills +- **🚀 Complete Solution** - Everything you need in one place +- **🛠️ Developer-Friendly** - Built by developers, for developers +- **🤖 AI-Native** - Designed for the age of AI +- **🌍 Open Source** - Transparent and extensible + +--- + +*Ready to transform your development workflow? [Download ClaraVerse](https://github.com/badboysm890/claraverse/releases) and join the future of local AI development!* \ No newline at end of file diff --git a/docs/config.json b/docs/config.json new file mode 100644 index 00000000..d5b9b974 --- /dev/null +++ b/docs/config.json @@ -0,0 +1,114 @@ +{ + "categories": [ + "getting-started", + "features", + "ideas" + ], + "files": { + "getting-started": { + "title": "Getting Started", + "description": "Quick start guide to set up ClaraVerse", + "category": "getting-started", + "order": 1, + "filePath": "getting-started/README.md" + }, + "clara-assistant": { + "title": "Clara Assistant", + "description": "Your AI coding companion", + "category": "features", + "order": 2, + "filePath": "features/clara-assistant.md" + }, + "agents": { + "title": "Agents", + "description": "AI workflow automation", + "category": "features", + "order": 3, + "filePath": "features/agents.md" + }, + "n8n": { + "title": "N8N", + "description": "Workflow automation platform", + "category": "features", + "order": 4, + "filePath": "features/n8n.md" + }, + "rag": { + "title": "RAG", + "description": "Retrieval Augmented Generation", + "category": "features", + "order": 5, + "filePath": "features/rag.md" + }, + "lumaui": { + "title": "LumaUI", + "description": "AI-powered web development environment", + "category": "features", + "order": 6, + "filePath": "features/lumaui.md" + }, + "imagegen": { + "title": "ImageGen", + "description": "AI image generation with ComfyUI", + "category": "features", + "order": 7, + "filePath": "features/imagegen.md" + }, + "settings": { + "title": "Settings", + "description": "Configuration and customization", + "category": "features", + "order": 8, + "filePath": "features/settings.md" + }, + "ideas": { + "title": "Ideas: How People Use ClaraVerse", + "description": "Real workflows and practical use cases", + "category": "ideas", + "order": 1, + "filePath": "ideas/README.md" + }, + "ai-development-workflow": { + "title": "AI-Powered Development Workflow", + "description": "From idea to deployed app using the full ecosystem", + "category": "ideas", + "order": 2, + "filePath": "ideas/ai-development-workflow.md" + }, + "content-factory": { + "title": "Content Factory", + "description": "Automated content creation and publishing pipeline", + "category": "ideas", + "order": 3, + "filePath": "ideas/content-factory.md" + }, + "research-assistant": { + "title": "Research Assistant", + "description": "Academic research with AI analysis and 3D knowledge mapping", + "category": "ideas", + "order": 4, + "filePath": "ideas/research-assistant.md" + }, + "customer-support": { + "title": "Customer Support Bot", + "description": "Intelligent support using your company knowledge", + "category": "ideas", + "order": 5, + "filePath": "ideas/customer-support.md" + }, + "personal-brand": { + "title": "Personal Brand Builder", + "description": "Automated brand development and content creation", + "category": "ideas", + "order": 6, + "filePath": "ideas/personal-brand.md" + }, + "local-code-review": { + "title": "Local AI Code Review", + "description": "Private code analysis without cloud dependencies", + "category": "ideas", + "order": 7, + "filePath": "ideas/local-code-review.md" + } + } +} \ No newline at end of file diff --git a/docs/features/agents.md b/docs/features/agents.md new file mode 100644 index 00000000..e4a1c0f9 --- /dev/null +++ b/docs/features/agents.md @@ -0,0 +1,205 @@ +--- +title: "Agents" +description: "Visual automation that runs 24/7 on your schedule" +category: "features" +order: 3 +lastUpdated: "2025-09-06" +contributors: ["badboysm890"] +--- + +Clara with LEGO blocks representing agent nodes + +# Agents + +Visual workflow automation using node-based programming. + +## What Agents Are + +Agents are automated workflows you build by connecting nodes (think visual programming). Each node does one specific task, and you chain them together to create complex automations. No coding required, but you need to understand basic logic flow. + +## System Requirements + +- Same as base ClaraVerse (8GB RAM minimum) +- Additional services may increase requirements +- ComfyUI node needs GPU with 4GB+ VRAM + +## How It Works + +### Node System +Agents use Directed Acyclic Graphs (DAGs) - fancy term meaning "no loops allowed". This prevents infinite loops and makes workflows predictable. + +**Why no loops?** +- Prevents system crashes +- Guarantees workflows complete +- Makes debugging possible +- Resource-efficient + +### Available Nodes + +**Input/Output** +- Input: Get data into your workflow +- Output: Send results somewhere +- Static Text: Store text templates +- File Upload: Process files + +**AI Nodes** +- LLM: Generate text using your chosen model +- Structured LLM: Get JSON responses +- Agent Executor: Run Clara Assistant within workflow +- Whisper: Speech-to-text (requires model download) + +**Processing** +- JSON Parse: Handle JSON data +- Combine Text: Merge multiple text sources +- If/Else: Conditional logic +- API Request: Call external services + +**Integration Nodes** +- Notebook Writer: Save to your knowledge base +- ComfyUI: Generate images +- Text-to-Speech: Create audio output +- PDF Input: Extract text from PDFs + +## Building Your First Agent + +### Example: Email Summary Agent +``` +1. API Request (fetch emails) + ↓ +2. JSON Parse (extract content) + ↓ +3. LLM Node (summarize) + ↓ +4. Notebook Writer (save summary) +``` + +### Step-by-Step: +1. Open Agent Studio +2. Drag nodes from sidebar +3. Connect outputs to inputs +4. Configure each node (click to edit) +5. Test with "Run Agent" +6. Save when it works + +## Scheduling (v0.1.45+) + +Agents can run automatically on schedules: +1. Create agent in Agent Studio +2. Set schedule (hourly, daily, weekly) +3. Manage in Tasks tab (above Settings) +4. ClaraVerse must be running for scheduled tasks + +**Common Schedules:** +- Daily email summary at 9 AM +- Weekly report generation +- Hourly data backups + +## Real Use Cases + +### Document Processor +``` +PDF Input → Extract Text → LLM (analyze) → Structured Output → API (save to database) +``` + +### Content Pipeline +``` +Static Prompt → LLM (generate) → ComfyUI (create image) → Combine → File Output +``` + +### Research Automation +``` +Input Query → API (search) → LLM (summarize) → Notebook Writer +``` + +## Node Configuration Tips + +### LLM Nodes +- **Model**: Use smaller models for simple tasks +- **Temperature**: 0.7 for creative, 0.1 for factual +- **Max Tokens**: Limit to what you need (saves time) + +### API Request Node +- **Method**: GET for reading, POST for sending data +- **Headers**: Include authentication if needed +- **Timeout**: Set reasonable limits (default 30s) + +### Notebook Writer +- **Notebook**: Must exist before using +- **Append vs Replace**: Choose wisely +- **Format**: Markdown recommended + +## Performance Expectations + +- Simple workflow (3-4 nodes): 5-10 seconds +- Complex workflow (10+ nodes): 30-60 seconds +- Image generation: 30-120 seconds depending on GPU +- API calls: Depends on external service + +## Common Issues & Solutions + +**Agent Won't Run** +- Check all nodes are connected +- Verify required services running (ComfyUI for image nodes) +- Look for red error indicators on nodes + +**Slow Performance** +- Use smaller models where possible +- Reduce parallel API calls +- Check system resources + +**Output Not As Expected** +- Test each node individually +- Check data formatting between nodes +- Verify LLM prompts are clear + +## Integration with Other Features + +### With Clara Assistant +Clara can trigger agents: "Run my daily summary agent" + +### With N8N +- Use N8N for external integrations +- Trigger agents from N8N webhooks +- Combine for complex automations + +### With Notebooks +- Read from notebooks for context +- Write results back to notebooks +- Build knowledge over time + +## Limitations + +1. **No Loops**: Can't do "for each" or "while" operations +2. **Memory**: Large workflows may hit memory limits +3. **Error Handling**: Limited - workflow stops on error +4. **Debugging**: No step-through debugging yet +5. **Version Control**: No built-in versioning + +## Pro Tips + +1. Start simple - test with 2-3 nodes first +2. Name your nodes clearly +3. Test after adding each node +4. Save working versions before major changes +5. Use Static Text nodes for prompt templates +6. Keep LLM prompts concise and specific + +## Export Options + +Agents can be exported as JavaScript code: +1. Settings → Export as Code +2. Select your agent +3. Get standalone JS module +4. Note: Contains API keys - backend use only + +## Getting Started + +1. Open Agent Studio +2. Create "Hello World" agent: + - Static Text ("Hello") + - Combine Text (add "World") + - Output node +3. Run and verify output +4. Build from there + +Remember: Agents are powerful but require logical thinking. Start simple, test often, and gradually increase complexity. \ No newline at end of file diff --git a/docs/features/clara-assistant.md b/docs/features/clara-assistant.md new file mode 100644 index 00000000..3a5d992c --- /dev/null +++ b/docs/features/clara-assistant.md @@ -0,0 +1,194 @@ +--- +title: "Clara Assistant" +description: "Your AI-powered command center that connects everything" +category: "features" +order: 2 +lastUpdated: "2025-09-06" +contributors: ["badboysm890"] +--- + +Assistant + +# Clara Assistant + +Your local AI assistant that orchestrates your entire workspace. + +## What Clara Actually Does + +Clara is an AI assistant that runs on your computer and connects to everything else in ClaraVerse. Think of it as your command center - you can chat, research, generate content, and automate tasks all from one interface. + +## System Requirements + +- **Minimum**: 8GB RAM, any GPU helps +- **Recommended**: 16GB RAM, 4GB+ VRAM GPU +- **Best Experience**: 32GB RAM, 8GB+ VRAM GPU + +## Two Operating Modes + +### Chat Mode (Default) +Fast, lightweight conversations without system access. + +**Use for:** +- Code explanations +- Planning and brainstorming +- General questions +- Learning concepts + +**Example:** +``` +You: How do React hooks work? +Clara: [Explains hooks without accessing any tools] +``` + +### Agent Mode +Full system access with MCP tools. Slower but can actually do things. + +**Use for:** +- Web research +- File operations +- Code generation and execution +- Document processing + +**Example:** +``` +You: Research the latest AI papers and create a summary PDF +Clara: [Searches web, analyzes papers, generates PDF] +``` + +## Core Features + +### Web Research (Agent Mode + Docker) +If you have Docker installed, Clara gets unlimited web search through SearXNG: +- No API keys needed +- Searches Google, Bing, DuckDuckGo simultaneously +- Completely private + +**Setup:** Just have Docker running. Clara handles the rest. + +### File Processing +Drag and drop any file into chat: +- PDFs, Word docs, Excel sheets +- Code files (any language) +- Images for analysis +- CSVs for data work + +**Limitation:** Large files (>10MB) may be slow to process. + +### Custom Tools via N8N +Create webhooks in N8N, convert them to Clara tools: +1. Build workflow in N8N +2. Create webhook trigger +3. Go to Settings → Tools +4. Add webhook URL as new tool + +**Example tools users have built:** +- Email checker +- Database queries +- Slack notifications +- Calendar management + +### Notebooks Integration +Attach notebooks for Clara to reference: +``` +You: [Attach company_guidelines notebook] +You: Write code following our standards +Clara: [Uses notebook context for accurate responses] +``` + +### Memory System +Clara remembers facts about you between conversations: +- Your tech stack preferences +- Project patterns +- Common workflows + +**Note:** Memory is stored in browser IndexedDB. Clearing browser data loses memories. + +## Model Recommendations + +### Starter Model +**JanNano128K-4B**: Fast, runs on most hardware, good for basic tasks + +### Power User Models +**GPT-OSS-20B**: Balanced performance and quality +**SeedOSS-ByteDance-36B**: Best quality, needs beefy hardware + +### Vision Model +**InterVL-14B**: For image understanding tasks + +## Common Workflows + +### Daily Email Summary +``` +1. Switch to Agent Mode +2. "Check my email and create a summary" +3. Clara uses email tool to fetch messages +4. Generates summary and saves to notebook +``` + +### Research Task +``` +1. Agent Mode +2. "Research [topic] and create a report" +3. Clara searches web, analyzes sources +4. Creates formatted report +5. Optional: Save to notebook for future reference +``` + +### Code Project +``` +1. Upload project files for context +2. "Help me add authentication to this app" +3. Clara analyzes code, suggests implementation +4. Agent Mode: Can write and test code directly +``` + +## Performance Expectations + +- **Chat Mode**: Instant to 5 seconds depending on model +- **Agent Mode with search**: 10-30 seconds per search +- **File processing**: 5-20 seconds depending on size +- **Code execution**: 2-10 seconds + +## Limitations & Known Issues + +1. **Model Quality**: Clara is only as good as your chosen model +2. **Local Processing**: Slower than cloud services like ChatGPT +3. **Memory Limits**: Long conversations may hit context limits +4. **MCP Tools**: Currently only local MCP tools work (HTTP support coming) +5. **Voice**: Requires RAG container (additional 8GB download) + +## Troubleshooting + +**Clara not responding?** +- Check if model is loaded in Settings +- Verify Clara Core service is running +- Try refreshing the page + +**Agent Mode not working?** +- Ensure Docker is running for web search +- Check MCP server status in logs +- Switch to Chat Mode and back + +**Memory issues?** +- Use smaller models (4B or 7B) +- Close other applications +- Restart Clara Core service + +## Pro Tips + +1. Start with Chat Mode to plan, switch to Agent for execution +2. Use notebooks for any reference material you'll need repeatedly +3. Create N8N tools for repetitive tasks +4. Upload files rather than pasting long content +5. Use specific models for specific tasks (small for chat, large for complex reasoning) + +## Getting Started + +1. Install ClaraVerse and Docker +2. Download JanNano128K-4B model +3. Start with Chat Mode - ask anything +4. Try Agent Mode - "search for Python tutorials" +5. Create your first N8N tool +6. Build a notebook for your project + +Remember: Clara runs entirely on your machine. No data leaves your computer unless you explicitly configure external services. \ No newline at end of file diff --git a/docs/features/imagegen.md b/docs/features/imagegen.md new file mode 100644 index 00000000..6509ed62 --- /dev/null +++ b/docs/features/imagegen.md @@ -0,0 +1,296 @@ +--- +title: "ImageGen" +description: "Professional AI image generation without the subscription trap" +category: "features" +order: 7 +lastUpdated: "2025-09-06" +contributors: ["badboysm890"] +--- + +Clara creating AI-generated images + +# ImageGen + +Local AI image generation using ComfyUI. + +## What ImageGen Is + +ImageGen wraps ComfyUI (the standard for local Stable Diffusion) into ClaraVerse. Generate unlimited images on your hardware with no credits or subscriptions. + +## System Requirements + +- **Minimum**: 4GB VRAM GPU +- **Recommended**: 8GB+ VRAM GPU +- **Storage**: 10-50GB for models +- **RAM**: 16GB recommended + +Without GPU: Possible but extremely slow (2-10 minutes per image) + +## Setup Options + +### Option 1: Windows CUDA Container +Easiest setup for Windows users with NVIDIA GPUs: +1. Download container from ClaraVerse +2. Auto-installs with CUDA support +3. Includes basic models +4. Ready to use + +### Option 2: Bring Your Own ComfyUI +Already have ComfyUI? Just point ClaraVerse to it: +``` +Settings → ImageGen → ComfyUI URL +Default: http://localhost:8188 +``` + +### Option 3: Manual Setup +Install ComfyUI yourself and configure. + +## Two Interfaces + +### Simple UI (Clara's Interface) +Best for quick generation: +- Model selection dropdown +- Basic LoRA support +- Prompt enhancement via Clara +- Built-in gallery +- Batch generation + +### ComfyUI Interface +Full ComfyUI power: +- Complex workflows +- All parameters exposed +- Custom nodes +- Advanced techniques + +## Models + +### Getting Models + +**From CivitAI:** +1. Browse models in Model Manager +2. Click download +3. Auto-installs to correct folder + +**NSFW Models:** +- Need CivitAI API key +- Add in Settings → ImageGen +- Respects age verification + +**Manual Installation:** +``` +Place models in: +ComfyUI/models/checkpoints/ +``` + +### Recommended Models + +**For Beginners:** +- SD 1.5 base (fast, reliable) +- DreamShaper (good all-around) + +**For Quality:** +- SDXL models (need 8GB+ VRAM) +- Juggernaut XL + +**For Speed:** +- LCM models (5-10 steps) +- Turbo models + +## Common Workflows + +### Basic Generation +``` +1. Select model +2. Enter prompt +3. Click generate +4. Wait 20-60 seconds +5. Image appears in gallery +``` + +### With LoRA +``` +1. Download LoRA model +2. Select in LoRA dropdown +3. Adjust weight (0.5-1.0) +4. Generate as normal +``` + +### Batch Generation +``` +Simple UI: +- Set batch count +- Generates variations + +ComfyUI: +- Use batch nodes +- More control +``` + +## Clara Integration + +### Natural Language +``` +You: "Generate a sunset over mountains" +Clara: [Enhances prompt, triggers generation] +``` + +### In Agent Workflows +``` +Text Input → LLM (create prompt) → +ComfyUI Node → Image Output +``` + +## Performance + +### Generation Times (512x512) +- **4GB VRAM**: 30-60 seconds +- **8GB VRAM**: 15-30 seconds +- **12GB+ VRAM**: 10-20 seconds + +### VRAM Usage +- **SD 1.5**: 3-4GB +- **SDXL**: 6-8GB +- **With LoRAs**: +0.5-1GB each + +## Common Issues + +### Out of Memory (VRAM) +``` +Solutions: +1. Use smaller models (SD 1.5 instead of SDXL) +2. Reduce batch size to 1 +3. Lower resolution +4. Close other GPU applications +``` + +### LoRA Problems +``` +Issues: +- Too many LoRAs = crashes +- Incompatible LoRAs = artifacts +- High weights = distorted images + +Fix: +- Use 1-2 LoRAs max +- Keep weights under 1.0 +- Match LoRA to base model version +``` + +### Flux Models +``` +Known issue: LoRAs don't work with Flux +Workaround: Use Flux without LoRAs +Status: Fix in development +``` + +### Slow Generation +``` +Check: +1. GPU being used (not CPU) +2. Model size appropriate for VRAM +3. No other GPU tasks running +4. Sampling steps (20-30 is enough) +``` + +## Quality Tips + +### Better Prompts +``` +Good: "photograph of mountain landscape at golden hour, + professional photography, high detail" + +Bad: "mountain" +``` + +### Negative Prompts +``` +Always include: "low quality, blurry, distorted, + deformed, ugly, bad anatomy" +``` + +### Settings +- **Steps**: 20-30 (more isn't always better) +- **CFG Scale**: 6-8 (too high = fried) +- **Sampler**: DPM++ 2M Karras (reliable) + +## Advanced Features + +### Custom Workflows +In ComfyUI interface: +- Build complex pipelines +- Save/load workflows +- Share with community + +### Upscaling +``` +Generate at 512x512 → Upscale to 2048x2048 +Saves VRAM, improves quality +``` + +### Img2Img +``` +Upload image → Modify with prompt +Great for variations +``` + +## Storage Management + +Models take space: +- **SD 1.5 models**: 2-4GB each +- **SDXL models**: 6-7GB each +- **LoRAs**: 10-200MB each + +Clean up: +``` +Settings → ImageGen → Manage Models +Delete unused models +``` + +## Integration Examples + +### Content Pipeline +``` +Notebook (article) → Clara (extract key points) → +ImageGen (create illustrations) → Save to project +``` + +### Social Media Automation +``` +Agent scheduled daily: +Generate prompt → Create image → +Post to social platforms +``` + +### Design Variations +``` +Upload logo → Generate variations → +Save best to gallery +``` + +## Limitations + +1. **Hardware Dependent**: Quality/speed depends on GPU +2. **Model Size**: Good models are large (2-7GB) +3. **Learning Curve**: ComfyUI can be complex +4. **VRAM Limits**: Bigger isn't always possible +5. **Consistency**: Hard to get exact same image twice + +## Getting Started + +1. Ensure GPU drivers updated +2. Download SD 1.5 model (start small) +3. Try simple prompt: "cat sitting on desk" +4. Experiment with settings +5. Download LoRAs for style +6. Try ComfyUI interface for advanced work + +## Pro Tips + +1. **Start with SD 1.5** - Faster and easier +2. **Learn prompting** - Makes huge difference +3. **Save good prompts** - Reuse what works +4. **Batch generate** - Then pick best +5. **Use Clara** - For prompt enhancement +6. **Monitor VRAM** - Stay within limits + +Remember: Local generation means no censorship, no credits, no limits - but requires patience and good hardware. \ No newline at end of file diff --git a/docs/features/lumaui.md b/docs/features/lumaui.md new file mode 100644 index 00000000..d0dac650 --- /dev/null +++ b/docs/features/lumaui.md @@ -0,0 +1,296 @@ +--- +title: "LumaUI" +description: "AI-powered web development that adapts to your skill level" +category: "features" +order: 6 +lastUpdated: "2025-09-06" +contributors: ["badboysm890"] +--- + +Clara developing web applications with LumaUI + +# LumaUI + +Web development environment with AI assistance, running locally. + +## What LumaUI Is + +LumaUI provides two ways to build web applications: +1. **LumaUI-lite**: Simple HTML/CSS/JS editor with live preview +2. **Full LumaUI**: Complete development environment with WebContainer (Node.js in browser) + +Both integrate with Clara for AI-assisted development. + +## System Requirements + +- Base ClaraVerse requirements +- Chrome/Edge browser (WebContainer needs Chromium) +- 4GB+ free RAM for Full LumaUI +- Internet for CDN libraries (lite) or npm packages (full) + +## LumaUI-lite + +### What It Is +Basic web editor for single-page applications. No build tools, no complexity. + +### Features +- HTML, CSS, JavaScript files +- Live preview +- CDN library access +- AI code assistance + +### Built-in Libraries +```html + +- Tailwind CSS +- Font Awesome +- Google Fonts +- Animate.css +- Alpine.js +``` + +### Use Cases +- Landing pages +- Prototypes +- Learning web dev +- Simple tools + +### Workflow +``` +1. Create project +2. Edit HTML/CSS/JS +3. See live preview +4. Ask Clara for help +5. Export when done +``` + +## Full LumaUI + +### What It Is +Full development environment using WebContainer technology. Think VS Code + Node.js in your browser. + +### Features +- Full Node.js environment +- NPM package manager +- Terminal access +- Git support +- Hot reloading +- Framework support (React, Vue, Svelte) + +### Capabilities +```bash +# In the terminal +npm install any-package +npm run dev +git init +node server.js +``` + +### Use Cases +- React/Vue applications +- Full-stack development +- Complex projects +- Learning modern frameworks + +## AI Integration + +### How Clara Helps +``` +You: "Add a contact form" +Clara: [Generates form code, adds to project] + +You: "Make this responsive" +Clara: [Updates CSS with media queries] + +You: "Debug this error" +Clara: [Analyzes code, suggests fix] +``` + +### Best Models for Coding +- **Cloud**: GPT-4, Claude (best results) +- **Local**: 20B+ models with tool calling +- **Minimum**: 7B models (basic assistance only) + +## Choosing Which Version + +### Use LumaUI-lite When: +- Building simple sites +- Learning HTML/CSS/JS +- Need quick prototypes +- Limited system resources +- No build process needed + +### Use Full LumaUI When: +- Building React/Vue apps +- Need npm packages +- Want real development environment +- Building complex applications +- Need build tools + +## Common Workflows + +### Creating a Portfolio (Lite) +``` +1. New project → Choose template +2. Edit content in HTML +3. Style with Tailwind classes +4. Add interactions with Alpine.js +5. Export and deploy +``` + +### Building React App (Full) +``` +1. Select React template +2. npm install dependencies +3. Edit components +4. Clara helps with logic +5. npm run build +6. Deploy build folder +``` + +## Performance Expectations + +### LumaUI-lite +- Instant preview updates +- No build time +- Lightweight (~10MB memory) + +### Full LumaUI +- Initial load: 30-60 seconds +- npm install: Depends on packages +- Build times: Similar to local development +- Memory usage: 500MB-2GB + +## Integration with ClaraVerse + +### With Clara Assistant +``` +"Create a dashboard for my data" +Clara uses LumaUI to build it +``` + +### With ImageGen +``` +Generate images → Use in web project +``` + +### With Notebooks +``` +Build documentation site from notebook content +``` + +### With Agents +``` +Agent generates content → LumaUI displays it +``` + +### With N8N +``` +LumaUI frontend → N8N webhook backend +``` + +## Limitations + +### LumaUI-lite +1. No backend capabilities +2. No build tools +3. Limited to browser APIs +4. Single page only +5. No npm packages + +### Full LumaUI +1. WebContainer browser limitations +2. No native modules +3. Chrome/Edge only +4. Memory intensive +5. No direct file system access + +## Common Issues + +**WebContainer Won't Start** +- Use Chrome or Edge (not Firefox/Safari) +- Check available RAM +- Disable browser extensions +- Clear browser cache + +**npm Install Fails** +- Check internet connection +- Try different package registry +- Some packages incompatible with WebContainer + +**Preview Not Updating** +- Check for JavaScript errors +- Verify dev server running +- Try manual refresh + +## File Management + +### LumaUI-lite +- Files stored in IndexedDB +- Export as ZIP +- Import existing projects + +### Full LumaUI +- Virtual file system +- Git for version control +- Export entire project + +## Deployment + +### From LumaUI-lite +``` +1. Export project +2. Upload to any static host +3. Works immediately +``` + +### From Full LumaUI +``` +1. Run build command +2. Export dist/build folder +3. Deploy to static host +4. Or use CI/CD pipeline +``` + +## Pro Tips + +1. **Start with Lite** for simple projects +2. **Use Clara** for boilerplate code +3. **Test mobile view** in preview +4. **Save regularly** (browser storage can be cleared) +5. **Use templates** to start faster +6. **Keep projects small** in Full version + +## Code Quality with Clara + +### Good Prompts +``` +"Add form validation using native HTML5" +"Create responsive grid with Tailwind" +"Add error handling to this function" +``` + +### Avoid +``` +"Make it better" (too vague) +"Copy this entire codebase" (copyright) +"Add every possible feature" (scope creep) +``` + +## Getting Started + +### LumaUI-lite +1. Click "New Project" +2. Choose "Blank" or template +3. Edit index.html +4. See instant preview +5. Ask Clara for enhancements + +### Full LumaUI +1. Select framework (React/Vue/Vanilla) +2. Wait for environment setup +3. Open terminal: `npm install` +4. Start dev server: `npm run dev` +5. Edit and see hot reload + +Remember: LumaUI is for building real projects, not just demos. The AI assistance makes it accessible for beginners while powerful enough for professionals. \ No newline at end of file diff --git a/docs/features/n8n.md b/docs/features/n8n.md new file mode 100644 index 00000000..54d96085 --- /dev/null +++ b/docs/features/n8n.md @@ -0,0 +1,223 @@ +--- +title: "N8N" +description: "Enterprise workflow automation without enterprise pricing" +category: "features" +order: 4 +lastUpdated: "2025-09-06" +contributors: ["badboysm890"] +--- + + +Clara automating workflows with n8n + +# N8N + +Workflow automation through a visual interface, integrated into ClaraVerse. + +## What N8N Is + +N8N is an open-source workflow automation tool (like Zapier but self-hosted). We embedded it in ClaraVerse to handle external integrations - things that need authentication, API keys, OAuth, etc. It runs as a Docker container or you can bring your own instance. + +## Why N8N? + +Honest answer: Building secure integrations for every service (Gmail, Slack, Discord, databases) would take forever and handling everyone's credentials is a security nightmare. N8N already does this well, so we integrated it instead of reinventing the wheel. + +## Setup Requirements + +### Option 1: Docker (Recommended) +```bash +# N8N runs automatically when you start it from ClaraVerse +# Just need Docker installed and running +``` + +### Option 2: External N8N +```bash +# Run your own N8N instance +npm install n8n -g +n8n start +# Then point ClaraVerse to http://localhost:5678 +``` + +## How It Works in ClaraVerse + +1. **Embedded View**: N8N UI appears inside ClaraVerse +2. **Service Detection**: Auto-finds N8N instance +3. **Tool Creation**: Convert workflows to Clara tools +4. **Webhook Integration**: Expose workflows as endpoints + +## Creating Clara Tools from N8N + +This is the killer feature - any N8N workflow becomes a Clara Assistant tool: + +### Step-by-Step: +1. Create workflow in N8N +2. Add Webhook trigger node +3. Build your automation +4. Test the webhook +5. In ClaraVerse sidebar: "Create Tool" +6. Name it (lowercase_with_underscores) +7. Clara can now use this tool + +### Example: Email Checker Tool +``` +N8N Workflow: +Webhook → Gmail Node → Format Data → Response + +In Clara: +"Check my email for urgent messages" +[Clara automatically uses your email tool] +``` + +## Common Workflows + +### Email to Notebook +``` +Trigger: Schedule (daily) +Gmail → Get Messages → Format → HTTP Request to ClaraVerse +Result: Daily email summaries in your notebook +``` + +### Slack Notifications +``` +Trigger: Webhook from Clara +Format Message → Slack Node → Send Message +Usage: Clara can send Slack messages +``` + +### Database Queries +``` +Trigger: Webhook +Parse Query → Database Node → Format Results +Usage: Clara queries your database +``` + +## N8N Nodes You'll Use Most + +- **Webhook**: Entry point for Clara tools +- **HTTP Request**: Call APIs and services +- **Gmail/Outlook**: Email operations +- **Slack/Discord**: Messaging +- **Google Sheets**: Spreadsheet operations +- **Database**: PostgreSQL, MySQL, MongoDB +- **Schedule**: Cron-based triggers + +## Authentication in N8N + +For OAuth and authenticated services: +1. Click "Open in Browser" button +2. N8N opens in your default browser +3. Authenticate services there +4. Return to ClaraVerse - connections persist + +## Performance & Limits + +- Webhook response time: 1-5 seconds typical +- Maximum workflow time: 5 minutes (configurable) +- Concurrent workflows: Depends on system resources +- Memory per workflow: ~100MB average + +## Integration Points + +### With Clara Assistant +```javascript +// Your N8N webhook becomes: +Tool: check_emails +Description: "Checks Gmail for new messages" +// Clara uses it automatically when relevant +``` + +### With Agents +- Trigger N8N workflows from Agent nodes +- Use N8N results in Agent workflows +- Combine for complex automations + +### With Notebooks +- Auto-import data to notebooks +- Trigger workflows when notebooks update +- Build knowledge management pipelines + +## Common Issues & Solutions + +**N8N Won't Start** +```bash +# Check if port 5678 is in use +lsof -i :5678 # Mac/Linux +netstat -ano | findstr :5678 # Windows + +# Kill existing process if needed +``` + +**Webhook Not Working** +- Ensure N8N is running +- Check webhook URL is correct +- Test in N8N interface first +- Verify firewall allows connections + +**Authentication Issues** +- Use browser mode for OAuth +- Check credentials are saved +- Refresh tokens may expire + +## Real Use Cases + +### Customer Support Automation +``` +Email arrives → Extract content → Check knowledge base → +Generate response → Send reply → Log to database +``` + +### Social Media Pipeline +``` +Schedule trigger → Generate content with Clara → +Create image → Post to platforms → Track engagement +``` + +### Data Sync +``` +Database change → Transform data → Update Google Sheets → +Notify team → Update notebook +``` + +## Limitations + +1. **Complexity**: N8N learning curve for advanced workflows +2. **External Dependencies**: Relies on external services being available +3. **Debugging**: Complex workflows can be hard to debug +4. **Rate Limits**: Subject to external API limits +5. **Authentication**: Some services require re-authentication periodically + +## Pro Tips + +1. **Start Simple**: Basic webhook → action → response +2. **Test in N8N First**: Before creating Clara tools +3. **Use Descriptive Names**: "fetch_customer_data" not "tool1" +4. **Handle Errors**: Add error handling nodes +5. **Log Everything**: Use N8N's logging for debugging +6. **Version Control**: Export workflows as JSON for backup + +## Security Considerations + +- Credentials stored in N8N's database (encrypted) +- Webhooks are public endpoints - use authentication if needed +- API keys in workflows - never share workflow exports +- Run N8N locally for maximum security + +## Getting Started + +1. Start N8N from ClaraVerse Settings +2. Create simple webhook workflow: + - Webhook trigger + - Set node (return "Hello World") +3. Test webhook with curl or browser +4. Create Clara tool from webhook +5. Ask Clara to use your tool + +## Advanced: Custom N8N Nodes + +You can add custom nodes for ClaraVerse-specific operations: +```javascript +// Place in n8n/custom folder +// Adds ClaraVerse-specific functionality +``` + +Remember: N8N handles the messy parts (auth, APIs, scheduling) so Clara can focus on intelligence. It's not pretty, but it works and saves months of development. \ No newline at end of file diff --git a/docs/features/rag.md b/docs/features/rag.md new file mode 100644 index 00000000..5eec3c2a --- /dev/null +++ b/docs/features/rag.md @@ -0,0 +1,251 @@ +--- +title: "RAG" +description: "Your intelligent knowledge canvas with 3D visualization" +category: "features" +order: 5 +lastUpdated: "2025-09-06" +contributors: ["badboysm890"] +--- + + +Clara with notebooks and knowledge visualization + +# Notebooks (RAG) + +Document storage with AI-powered search and conversation capabilities. + +## What Notebooks Are + +Notebooks are your knowledge management system. Upload documents, and then chat with them using RAG (Retrieval Augmented Generation). It's like having a research assistant who's read everything you've uploaded and can answer questions about it. + +## Technical Stack + +- **LightRAG**: The RAG engine (runs in Docker container) +- **Embedding Model**: mxbai-embed-large +- **Storage**: Browser IndexedDB + mounted Docker volumes +- **Visualization**: 3D knowledge graph using Three.js + +## System Requirements + +- Base ClaraVerse requirements +- Additional 8GB storage for RAG container +- Docker required for backend +- 2GB+ RAM for embedding operations + +## Setup + +1. Start Docker +2. Go to Notebooks tab +3. First launch downloads RAG container (~8GB) +4. Wait for health check to pass +5. Create your first notebook + +## How It Works + +### Document Processing Pipeline +``` +Upload Document → Extract Text → Generate Embeddings → +Store in Vector DB → Build Knowledge Graph → Ready for Chat +``` + +### Supported File Types +- PDF (recommended) +- Word documents (.docx) +- Text files (.txt, .md) +- Web clippings +- Email exports + +### Processing Times +- Small document (1-10 pages): 10-30 seconds +- Medium document (10-50 pages): 1-2 minutes +- Large document (50+ pages): 3-5 minutes +- Batch upload: Processes sequentially + +## Using Notebooks + +### Creating & Organizing +``` +1. Click "New Notebook" +2. Name it descriptively (e.g., "Project_Alpha_Docs") +3. Set category/tags for organization +4. Optional: Set custom wallpaper +``` + +### Uploading Documents +- Drag and drop files +- Or click upload button +- Monitor status indicator +- Green = processed, Yellow = processing, Red = error + +### Chatting with Documents +``` +You: What does the contract say about payment terms? +Notebook: [Searches documents, provides relevant answer with sources] +``` + +## Integration with Clara + +### As Clara's Memory +``` +1. Attach notebook to Clara Assistant +2. Clara references it automatically +Example: "According to your project docs, the deadline is..." +``` + +### Automatic Updates via Agents +``` +Agent Workflow: +Email → Extract → Convert to PDF → Save to Notebook +Result: Self-updating knowledge base +``` + +## The 3D Visualization + +Shows how your documents connect: +- **Nodes**: Documents or concepts +- **Edges**: Relationships/references +- **Clusters**: Related topics +- **Size**: Importance/frequency + +**Navigation**: +- Click and drag to rotate +- Scroll to zoom +- Click nodes for details + +## Real Use Cases + +### Research Project +``` +Structure: +- Papers_Notebook: Academic papers +- Notes_Notebook: Your observations +- Data_Notebook: Raw data files + +Workflow: +1. Upload papers as you find them +2. Chat to find connections +3. Clara references for writing +``` + +### Company Knowledge Base +``` +Structure: +- Policies_Notebook: HR and procedures +- Projects_Notebook: Active project docs +- Archives_Notebook: Historical data + +Usage: +- New employees query policies +- Project teams find decisions +- Audit trail of information +``` + +### Personal Assistant +``` +Daily Workflow: +1. Agent fetches emails +2. Converts to PDFs +3. Stores in Daily_Notebook +4. Clara can answer: "What did John say about the budget?" +``` + +## Performance Optimization + +### For Better RAG Results +1. **Quality Documents**: Clean PDFs work better than scanned images +2. **Logical Organization**: One topic per notebook +3. **Clear Naming**: Descriptive filenames help retrieval +4. **Regular Cleanup**: Remove outdated documents + +### For Faster Processing +1. Upload during off-hours +2. Process large batches overnight +3. Split huge documents if possible +4. Restart container if sluggish + +## Limitations + +1. **OCR Quality**: Scanned documents may have issues +2. **Language**: Best with English (other languages experimental) +3. **Context Window**: Very long documents may be truncated +4. **Graph Complexity**: 100+ documents can make visualization cluttered +5. **Storage**: Everything in browser storage (clear cache = lose data) + +## Troubleshooting + +**Backend Not Starting** +```bash +# Check Docker is running +docker ps + +# Restart container +docker restart claraverse-rag + +# Check logs +docker logs claraverse-rag +``` + +**Documents Not Processing** +- Check file format is supported +- Verify file size < 50MB +- Look for special characters in filename +- Try re-uploading + +**Chat Not Working** +- Ensure documents are fully processed (green status) +- Check backend health in settings +- Refresh the page +- Restart RAG container + +## Advanced Features + +### Custom Embeddings +```python +# In settings, configure embedding model +{ + "model": "mxbai-embed-large", + "dimensions": 1024, + "batch_size": 32 +} +``` + +### Knowledge Graph Tuning +- Adjust similarity threshold +- Configure clustering algorithm +- Set visualization parameters + +## Data Privacy + +- All processing happens locally +- Documents never leave your machine +- Embeddings stored in Docker volume +- No external API calls for RAG + +## Backup Strategy + +```bash +# Backup notebook data +docker cp claraverse-rag:/data ./backup + +# Backup browser storage +# Export from browser DevTools +``` + +## Pro Tips + +1. **One notebook per project** - Keeps context focused +2. **PDF over everything** - Most reliable format +3. **Update regularly** - Use agents for automatic updates +4. **Name consistently** - "YYYY-MM-DD_description" works well +5. **Test retrieval** - Ask questions to verify RAG is working + +## Getting Started + +1. Create notebook: "Test_Knowledge" +2. Upload a PDF you know well +3. Wait for processing to complete +4. Ask: "Summarize this document" +5. Try: "What are the key points about [specific topic]?" +6. Attach to Clara for enhanced conversations + +Remember: Notebooks are only as good as the documents you feed them. Quality in, quality out. \ No newline at end of file diff --git a/docs/features/settings.md b/docs/features/settings.md new file mode 100644 index 00000000..96df3f49 --- /dev/null +++ b/docs/features/settings.md @@ -0,0 +1,320 @@ +--- +title: "Settings" +description: "Complete control center for your ClaraVerse workspace" +category: "features" +order: 8 +lastUpdated: "2025-09-06" +contributors: ["badboysm890"] +--- + +Clara managing settings and configurations + +# Settings + +Central configuration for all ClaraVerse components. + +## Overview + +Settings controls everything: AI providers, local models, services, and system configuration. Most issues can be fixed here. + +## Main Sections + +### AI Services + +Configure AI providers (local and cloud): + +**Adding Providers:** +``` +1. Click "Add Provider" +2. Choose type: + - OpenAI (needs API key) + - Anthropic/Claude (needs API key) + - Ollama (local, no key) + - Custom OpenAI-compatible +3. Test connection +4. Set as primary if desired +``` + +**Provider Priority:** +- Primary: Used by default +- Secondary: Fallback options +- Specific: Some features need specific providers + +**Cost Management:** +- Local (Ollama, Clara Core): Free +- Cloud: Pay per token +- Track usage in provider dashboards + +### Local Models + +Manage models for Clara Core and Ollama: + +**Clara Core Models:** +``` +Recommended: +- JanNano128K-4B (starter, 3GB) +- GPT-OSS-20B (balanced, 12GB) +- SeedOSS-ByteDance-36B (best, 20GB) +``` + +**Download Process:** +1. Select model +2. Check storage space +3. Download (can take 10-60 minutes) +4. Auto-configures when complete + +**GPU Diagnostics:** +Shows: +- GPU model and VRAM +- Current usage +- Recommended models for your hardware + +### Services + +Control Docker containers and background services: + +**Core Services:** +- **Clara Core**: LLM engine (required) +- **N8N**: Workflow automation +- **Python Backend**: For notebooks/RAG +- **ComfyUI**: Image generation +- **LlamaSwap**: Model management + +**Service Management:** +```bash +# Each service shows: +Status: Running/Stopped +Port: Default port +Actions: Start/Stop/Restart +Logs: View output +``` + +**Common Issues:** +- Port conflicts: Change ports in advanced settings +- Memory issues: Stop unused services +- Docker not running: Start Docker first + +### Preferences + +Personalization options: + +**Appearance:** +- Theme: Light/Dark/System +- Wallpapers: Custom backgrounds +- Font size: Accessibility + +**Behavior:** +- Auto-save: Interval in seconds +- Notifications: Enable/disable +- Timezone: For scheduling + +**Privacy:** +- Telemetry: Always off (local-first) +- Data retention: How long to keep logs +- Export data: Backup everything + +### Profile + +Your information for Clara's context: + +**Personal Info:** +``` +Name: Used in conversations +Bio: Background for context +Preferences: Coding style, tools +``` + +**Usage Stats:** +- Messages sent +- Images generated +- Workflows created +- Storage used + +### Updates + +Keep ClaraVerse current: + +**Auto-Update Check:** +- Checks daily for new versions +- Shows changelog +- One-click update + +**Component Updates:** +- ClaraVerse main app +- Llama.cpp engine +- Docker containers +- Model updates + +**Alpha Features:** +Enable experimental features (may be unstable) + +### Export as Code + +Convert agent workflows to JavaScript: + +**Process:** +1. Select agent workflow +2. Choose export format: + - JavaScript module + - TypeScript + - Clara Flow SDK +3. Configure options +4. Generate code + +**Warning:** Exported code contains API keys - backend use only! + +**Use Cases:** +- Deploy workflows as microservices +- Integrate with existing systems +- Share workflows (remove keys first) + +## Quick Setup Guide + +### First Time Setup +``` +1. Start Docker +2. Go to Services → Start Clara Core +3. Go to Local Models → Download JanNano128K-4B +4. Go to AI Services → Set Clara Core as primary +5. Test in Clara Assistant +``` + +### For Power Users +``` +1. Add cloud provider (OpenAI/Claude) +2. Download larger models (20B+) +3. Start all services +4. Enable alpha features +``` + +### For Privacy-First Users +``` +1. Only use local models +2. Disable all telemetry +3. Run everything offline +4. Regular data exports for backup +``` + +## Performance Tuning + +### Memory Management +``` +Light use (8GB RAM): +- Run only Clara Core +- Use 4B models +- Stop unused services + +Heavy use (32GB+ RAM): +- Run all services +- Use large models +- Multiple containers +``` + +### GPU Optimization +``` +Check GPU diagnostics for: +- VRAM available +- Recommended model size +- Current utilization +``` + +## Troubleshooting + +### Service Won't Start +```bash +# Check Docker +docker ps + +# Check ports +netstat -an | grep PORT_NUMBER + +# View logs +docker logs SERVICE_NAME +``` + +### Model Download Fails +- Check disk space (need 2x model size) +- Verify internet connection +- Try different download source +- Clear cache and retry + +### API Provider Issues +- Test connection button +- Verify API key is correct +- Check billing/credits +- Try different endpoint + +## Data Storage + +### Where Everything Lives +``` +Browser Storage: +- Settings +- Chat history +- Notebooks metadata + +Docker Volumes: +- Models +- RAG embeddings +- Service data + +Local Filesystem: +- Downloaded models +- Exported data +- Backups +``` + +### Backup Strategy +```bash +# Export from Settings +Settings → Preferences → Export Data + +# Backup Docker volumes +docker cp claraverse:/data ./backup + +# Save browser data +Browser DevTools → Application → Storage +``` + +## Security Considerations + +- **API Keys**: Stored encrypted in browser +- **Local Models**: No data leaves machine +- **Docker**: Isolated containers +- **Network**: Services bind to localhost only + +## Common Configurations + +### Offline Setup +``` +1. Download all models while online +2. Disable update checks +3. Use only local providers +4. Works completely offline +``` + +### Hybrid Setup +``` +1. Local models for privacy +2. Cloud for complex tasks +3. Switch as needed +4. Best of both worlds +``` + +### Team Setup +``` +1. Shared N8N instance +2. Common model repository +3. Exported workflows +4. Standardized configuration +``` + +## Pro Tips + +1. **Start minimal** - Add services as needed +2. **Monitor resources** - Use task manager +3. **Regular backups** - Export weekly +4. **Test providers** - Before important work +5. **Update regularly** - But backup first + +Remember: Settings is where you fix problems and optimize performance. When something doesn't work, check here first. \ No newline at end of file diff --git a/docs/features/templates-button-added.md b/docs/features/templates-button-added.md new file mode 100644 index 00000000..5e2dc2e3 --- /dev/null +++ b/docs/features/templates-button-added.md @@ -0,0 +1,105 @@ +# ✨ Templates Button Added! + +## 📍 Location + +The **Templates** button has been added to the main toolbar in Agent Studio, right between **Workflows** and **Create Node** buttons. + +## 🎨 Button Details + +**Visual Design:** +- 🌸 Gradient background (sakura to pink) +- ✨ Sparkles icon +- 🔆 Shadow effect with hover animation +- 📝 "Templates" label + +**Button Position in Toolbar:** +``` +[New] [Workflows] [✨ Templates] [Create Node] [Import] [Export] [Save] [Execute] + ↑ + NEW! +``` + +## 🎯 Features + +### Click the Templates Button to: +- Browse all available templates +- Search and filter by category +- See template ratings and details +- Create new flows from templates instantly + +### The Button: +- ✅ Always visible in the toolbar +- ✅ Eye-catching gradient design +- ✅ Opens template browser modal +- ✅ Works from any screen (with or without active flow) + +## 🚀 How It Works + +1. **Click "Templates" button** in toolbar +2. **Browse templates** - See all 6+ templates with: + - Name and icon + - Difficulty level + - Category + - Description + - Tags + - Ratings +3. **Click any template** to create a flow +4. **Flow is created** with all nodes and connections! + +## 📊 Template Already Working! + +The `simple-chat-assistant.json` template has been updated with your actual flow: +- ✅ Input node ("How can i help you") +- ✅ Static Text node (System prompt for Clara chatbot) +- ✅ LLM Chat node (GPT configuration) +- ✅ Output node +- ✅ All connections between nodes + +**When you click this template, it will create a complete working chat assistant!** + +## 🎨 Visual Hierarchy + +The button stands out in the toolbar with: +- **Gradient color** - Different from other buttons +- **Sparkles icon** - Catches attention +- **Shadow effect** - Adds depth +- **Prominent position** - Easy to find + +## 🔧 Technical Details + +**File Modified:** +- `src/components/AgentStudio.tsx` + - Added Sparkles icon import + - Added Templates button in toolbar + - Button opens `isTemplateBrowserOpen` modal + +**Button Styling:** +```tsx +className="px-3 py-2 bg-gradient-to-r from-sakura-500 to-pink-500 +hover:from-sakura-600 hover:to-pink-600 text-white rounded-lg +flex items-center gap-2 text-sm font-medium transition-colors +shadow-md hover:shadow-lg" +``` + +## 💡 Usage Tips + +### For Users: +- Click **Templates** button anytime +- Browse and search templates +- Quick-start your agent development + +### For You (Adding More Templates): +1. Export flows from Agent Studio +2. Update JSON files in `templates/` folder +3. They automatically appear in template browser +4. Users can access via **Templates** button + +## 🎯 Next Steps + +The system is ready! You can now: +1. ✅ Click Templates button +2. ✅ See your chat assistant template +3. ✅ Create flows from it +4. ✅ Add more templates as needed + +**The Templates button is now prominently featured in the toolbar!** 🎉 diff --git a/docs/getting-started/README.md b/docs/getting-started/README.md new file mode 100644 index 00000000..316417fb --- /dev/null +++ b/docs/getting-started/README.md @@ -0,0 +1,162 @@ +--- +title: "Getting Started with ClaraVerse" +description: "Requirements and installation only" +category: "getting-started" +order: 1 +lastUpdated: "2025-09-05" +contributors: ["badboysm890"] +--- + + +Hello Welcome Clara + +# Getting Started + +Quick setup guide to get ClaraVerse running. + +## System Requirements + +### Minimum +- **OS**: Windows 10+, macOS 11+, or Linux (Ubuntu 20.04+) +- **RAM**: 8GB (will be slow) +- **Storage**: 5GB for app + first model +- **CPU**: Any x64 processor + +### Recommended +- **RAM**: 16GB or more +- **GPU**: 4GB+ VRAM for image generation +- **Storage**: 50GB+ for multiple models +- **Docker**: Required for N8N, RAG, and bundled services + +## Installation + +### Step 1: Download +Get the latest release from: + +The website: [https://claraverse.space](https://claraverse.space) +``` +https://github.com/claraverse-space/ClaraVerse/releases +``` + +Choose your platform: +- **Windows**: `ClaraVerse-win-x64.exe` +- **macOS**: `ClaraVerse-mac-universal.dmg` +- **Linux**: `ClaraVerse-linux-x64.AppImage` + +### Step 2: Install + +**Windows:** +1. Run the `.exe` installer +2. Windows Defender may warn - click "More info" → "Run anyway" +3. Launch from Start Menu + +**macOS:** +1. Open the `.dmg` file +2. Drag ClaraVerse to Applications +3. First launch: Right-click → Open (bypass Gatekeeper) +4. If blocked: System Preferences → Security → "Open Anyway" + +**Linux:** +```bash +chmod +x ClaraVerse-linux-x64.AppImage +./ClaraVerse-linux-x64.AppImage +``` + +### Step 3: First Launch +1. ClaraVerse opens in your browser +2. Download a starter model: + - Go to Settings → Local Models + - Download `JanNano128K-4B` (3GB) +3. Start chatting with Clara! + +## Optional Services + +### Docker (Highly Recommended) +Needed for N8N, RAG, and some features: + +**Windows/Mac**: Download Docker Desktop from docker.com +**Linux**: +```bash +sudo apt install docker.io docker-compose +sudo usermod -aG docker $USER +``` + +### N8N (Automation) +- **With Docker**: Auto-installs when you first open N8N tab +- **Without Docker**: Point to external N8N instance in Settings + +### ComfyUI (Image Generation) +- **Windows**: Bundled CUDA version available (one-click install) +- **Mac/Linux**: Bring your own ComfyUI instance +- **Setup**: Settings → ImageGen → Configure endpoint + +### RAG (Document Chat) +- **Storage needed**: + - macOS: +2GB + - Windows: +8GB (includes CUDA) +- **Install**: Notebooks tab → Follow setup prompts + +## Quick Test + +1. **Test Clara Chat**: + ``` + "Hello Clara, what can you do?" + ``` + +2. **Test Agent Mode** (if Docker installed): + ``` + Switch to Agent Mode + "Search for Python tutorials" + ``` + +3. **Test Image Generation** (if ComfyUI configured): + ``` + "Generate an image of a sunset" + ``` + +## Common First-Time Issues + +### "Model not found" +→ Download a model in Settings → Local Models + +### "Clara not responding" +→ Check Settings → Services → Clara Core is running + +### "Docker not found" +→ Install Docker Desktop and ensure it's running + +### "Permission denied" (Linux) +→ Add user to docker group and reboot + +### "Windows protected your PC" +→ Click "More info" → "Run anyway" (it's not signed) + +## Storage Locations + +- **App config**: `~/.claraverse/` +- **Models**: `~/.claraverse/models/` +- **Docker volumes**: Managed by Docker +- **Browser data**: IndexedDB (don't clear!) + +## What's Next? + +1. **Explore Clara Assistant** - Your main interface +2. **Try Notebooks** - Upload PDFs and chat with them +3. **Build an Agent** - Visual automation workflows +4. **Connect N8N** - External integrations +5. **Generate Images** - If you have a GPU + +## Need Help? + +- **Discord**: Most active community support +- **GitHub Issues**: Bug reports and features +- **Email**: praveensm890@gmail.com (expect delays - solo dev!) + +## Quick Reality Check + +- This is v0.1.x software - expect rough edges +- Local AI is slower than cloud services +- You'll need patience and some technical knowledge +- But you'll own everything and pay nothing + +Ready? Open ClaraVerse and let's build something cool! diff --git a/docs/ideas/README.md b/docs/ideas/README.md new file mode 100644 index 00000000..60f61210 --- /dev/null +++ b/docs/ideas/README.md @@ -0,0 +1,78 @@ +--- +title: "Ideas: How People Use ClaraVerse" +description: "Real workflows, practical use cases, and creative applications" +category: "ideas" +order: 1 +lastUpdated: "2025-09-06" +contributors: ["badboysm890"] +--- + +# 💡 Ideas: How People Use ClaraVerse + +**Real workflows from real users. See what's possible when AI tools actually work together.** + +This collection showcases the strongest workflows and most creative use cases for ClaraVerse. These aren't theoretical examples—they're battle-tested patterns that solve real problems. + +## 🎯 **Featured Workflows** + +### 🚀 **For Developers** +- **[AI-Powered Development Workflow](./ai-development-workflow.md)** - From idea to deployed app in hours, not days +- **[Local AI Code Review](./local-code-review.md)** - Private code analysis without sending to cloud APIs +- **[Documentation Generator](./auto-documentation.md)** - Turn your code into beautiful docs automatically + +### 🎨 **For Creators** +- **[Content Factory](./content-factory.md)** - Automated blog writing with custom images +- **[Social Media Automation](./social-media-pipeline.md)** - Generate posts, images, and schedule everything +- **[Personal Brand Builder](./personal-brand.md)** - Portfolio websites that update themselves + +### 📊 **For Researchers** +- **[Research Assistant](./research-assistant.md)** - Academic paper analysis with 3D knowledge mapping +- **[Meeting Intelligence](./meeting-intelligence.md)** - Turn recordings into actionable insights +- **[Knowledge Garden](./knowledge-garden.md)** - Build your second brain with AI curation + +### 🏢 **For Small Teams** +- **[Customer Support Bot](./customer-support.md)** - Handle inquiries with your company knowledge +- **[Project Management Hub](./project-management.md)** - Automate status updates and reporting +- **[Sales Pipeline](./sales-automation.md)** - Lead qualification and follow-up automation + +## 🌟 **Why These Work** + +**Integration Over Isolation** +Each workflow uses multiple ClaraVerse features working together. This isn't about individual tools—it's about orchestrated intelligence. + +**Privacy-First** +Everything runs locally. Your code, documents, and data never leave your machine unless you explicitly choose to share them. + +**Zero Subscription Costs** +While competitors charge $20-200+ monthly for similar capabilities, these workflows run on your hardware with optional cloud APIs only when you want them. + +## 🎭 **Choose Your Adventure** + +**New to ClaraVerse?** Start with [AI-Powered Development Workflow](./ai-development-workflow.md) to see the full ecosystem in action. + +**Content Creator?** Jump to [Content Factory](./content-factory.md) for automated publishing pipelines. + +**Researcher or Student?** Try [Research Assistant](./research-assistant.md) for academic workflow automation. + +**Running a Business?** Check out [Customer Support Bot](./customer-support.md) for practical automation. + +## 🔄 **Community Contributions** + +Got a workflow that's working great for you? We'd love to feature it! Each use case includes: + +- **Problem statement**: What real problem this solves +- **Setup guide**: Step-by-step implementation +- **Customization tips**: How to adapt for your needs +- **Troubleshooting**: Common issues and solutions + +## 📈 **Success Metrics** + +These workflows are proven to: +- **Save 10+ hours/week** on repetitive tasks +- **Improve output quality** through AI assistance +- **Reduce subscription costs** by $50-300/month +- **Increase creativity** by handling the mundane work + +--- + +**Ready to build something amazing? Pick a workflow and let's get started!** diff --git a/docs/ideas/ai-development-workflow.md b/docs/ideas/ai-development-workflow.md new file mode 100644 index 00000000..aabfbaec --- /dev/null +++ b/docs/ideas/ai-development-workflow.md @@ -0,0 +1,254 @@ +--- +title: "AI-Powered Development Workflow" +description: "From idea to deployed app using the full ClaraVerse ecosystem" +category: "ideas" +order: 2 +lastUpdated: "2025-09-06" +contributors: ["badboysm890"] +--- + +# 🚀 AI-Powered Development Workflow + +**Transform ideas into production applications using ClaraVerse's integrated ecosystem** + +This is ClaraVerse's flagship workflow—what happens when AI chat, web development, automation, and knowledge management work together seamlessly. + +## 💡 **The Problem** + +Building web applications typically involves: +- Hours of boilerplate setup +- Context switching between multiple tools +- Manual deployment and testing +- Disconnected documentation and assets +- Expensive cloud services for basic AI features + +## 🎯 **The ClaraVerse Solution** + +A single, integrated workflow that handles everything from ideation to deployment: + +``` +Idea → Research → Design → Code → Test → Deploy → Document → Maintain + ↑ ↓ + └──────────────── AI Feedback Loop ──────────────────────────┘ +``` + +## 🛠️ **Step-by-Step Implementation** + +### **Phase 1: Research & Planning (5-10 minutes)** + +**1. Brainstorm with Clara Assistant** +``` +You: "I want to build a task management app for small teams" + +Clara: Analyzes market, suggests features, identifies technical requirements +``` + +**2. Research Competition** +``` +Agent Mode: "Research existing task management tools and identify gaps" +→ Clara uses web search to gather competitor info +→ Results saved to Notebooks for reference +``` + +**3. Generate Visual Concepts** +``` +ImageGen: Create mockups and UI concepts +→ "Clean task management interface, modern design" +→ Generate multiple variations for inspiration +``` + +### **Phase 2: Development (30-60 minutes)** + +**4. Smart Scaffolding in LumaUI** +``` +LumaUI Full: Create React + TypeScript + Tailwind project +Clara Integration: "Build a kanban board component with drag/drop" +→ Clara generates complete component structure +→ Includes proper TypeScript types and responsive design +``` + +**5. AI-Assisted Development** +``` +Clara helps with: +- Component architecture decisions +- State management setup +- API design and implementation +- Error handling patterns +- Accessibility considerations +``` + +**6. Asset Generation** +``` +ImageGen: Generate custom icons and illustrations +→ Consistent style matching your design system +→ Export directly into project assets folder +``` + +### **Phase 3: Automation & Intelligence (15-30 minutes)** + +**7. Build Smart Features with Agents** +``` +Agent Workflow: Email notification system +→ API Request node: Check for new tasks +→ LLM node: Generate smart notifications +→ Webhook node: Send to notification service +``` + +**8. Documentation Automation** +``` +N8N Workflow: Auto-generate documentation +→ Monitor code changes in repo +→ Extract component info and APIs +→ Generate markdown docs +→ Update project wiki automatically +``` + +### **Phase 4: Knowledge & Memory (10-15 minutes)** + +**9. Project Knowledge Base** +``` +Notebooks: Store all project information +→ Requirements and decisions +→ Code patterns and conventions +→ User feedback and iterations +→ 3D visualization of project knowledge +``` + +**10. Clara Memory Integration** +``` +Clara learns your project: +→ Coding style preferences +→ Architecture decisions +→ Common patterns you use +→ Gets better at helping with each interaction +``` + +## 🔄 **The Complete Ecosystem in Action** + +**Real Example: E-commerce Dashboard** + +1. **Research**: Clara finds that most dashboards lack real-time inventory alerts +2. **Design**: ImageGen creates custom dashboard mockups +3. **Development**: LumaUI builds the interface with Clara's help +4. **Intelligence**: Agents monitor inventory and send smart alerts +5. **Documentation**: N8N auto-generates API docs and user guides +6. **Knowledge**: Notebooks store business rules and customer feedback +7. **Iteration**: Clara suggests improvements based on stored knowledge + +## 🎯 **Why This Works Better Than Alternatives** + +**vs. Traditional Development:** +- 10x faster initial setup +- Built-in AI assistance throughout +- Automated documentation and testing +- Integrated asset generation + +**vs. No-Code Platforms:** +- Full code control and customization +- No vendor lock-in +- Professional development practices +- Real programming language flexibility + +**vs. Cloud AI Tools:** +- Complete privacy - code never leaves your machine +- No API costs for basic AI features +- Unlimited usage without quotas +- Works offline after initial setup + +## 💰 **Cost Comparison** + +**Traditional Stack:** +- GitHub Copilot: $10/month +- Figma: $12/month +- Vercel Pro: $20/month +- OpenAI API: $20-100/month +- **Total: $62-142/month** + +**ClaraVerse:** +- One-time setup +- Optional cloud APIs only when needed +- **Total: $0-20/month** + +## 🎨 **Customization Examples** + +**For SaaS Applications:** +- Add Stripe integration via N8N +- User analytics dashboard in LumaUI +- Automated customer onboarding emails + +**For Portfolio Sites:** +- Auto-update with latest projects +- Generate case study content +- Social media cross-posting + +**For Internal Tools:** +- Connect to company databases +- Automated reporting workflows +- Team collaboration features + +## 🚨 **Common Gotchas & Solutions** + +**Issue**: WebContainer limitations with certain npm packages +**Solution**: Use LumaUI-lite for simpler projects, or develop locally and sync + +**Issue**: Local models slower than cloud APIs +**Solution**: Use cloud providers for complex tasks, local for privacy-sensitive work + +**Issue**: Complex agent workflows can be fragile +**Solution**: Start simple, add complexity gradually, test each node individually + +## 🔧 **Setup Requirements** + +**Minimum:** +- 8GB RAM +- Modern web browser +- Basic Docker knowledge + +**Recommended:** +- 16GB RAM +- GPU for local image generation +- SSD storage for better performance + +**Optional Cloud Services:** +- OpenAI/Anthropic for advanced AI features +- GitHub for version control +- Cloud hosting for deployment + +## 📈 **What You Can Realistically Build** + +**Weekend Project Ideas:** +- Personal portfolio site with auto-updating content +- Simple CRUD app with AI-generated forms +- Landing page with dynamic content generation +- Basic dashboard with automated data visualization + +**Week-Long Projects:** +- Client portal with document management +- E-commerce site with AI product descriptions +- Team collaboration tool with smart notifications +- Content management system with AI assistance + +**Real Benefits You'll Experience:** +- Faster initial setup compared to starting from scratch +- AI assistance when you're stuck on implementation +- Integrated workflow from idea to deployment +- Learning acceleration through AI explanations + +## 🔄 **Next Steps** + +1. **Try the basic workflow** with a simple project +2. **Add one new integration** each week +3. **Share your results** with the community +4. **Customize for your specific needs** + +## 🤝 **Community Extensions** + +**Popular Add-ons:** +- Database integration patterns +- Authentication workflows +- Payment processing automation +- Analytics and monitoring setup + +--- + +**Ready to transform your development process? Start with a simple project and watch the magic happen.** diff --git a/docs/ideas/content-factory.md b/docs/ideas/content-factory.md new file mode 100644 index 00000000..29c8c781 --- /dev/null +++ b/docs/ideas/content-factory.md @@ -0,0 +1,342 @@ +--- +title: "Content Factory" +description: "Automated content creation pipeline for blogs, social media, and marketing" +category: "ideas" +order: 3 +lastUpdated: "2025-09-06" +contributors: ["badboysm890"] +--- + +# 🎨 Content Factory + +**Turn one idea into a complete content ecosystem across all platforms** + +This workflow shows how content creators use ClaraVerse to automate their entire content pipeline—from research to publication, with custom images and cross-platform distribution. + +## 💡 **The Problem** + +Content creation is time-consuming: +- Research takes hours +- Writing requires constant inspiration +- Custom images cost money or time +- Publishing across platforms is manual +- SEO optimization is complex +- Analytics tracking is fragmented + +**Most creators spend 80% of time on logistics, 20% on creativity.** + +## 🎯 **The ClaraVerse Solution** + +Flip the ratio: Spend 80% of time being creative, let AI handle the logistics. + +``` +Topic Idea → Research → Writing → Images → SEO → Publish → Analytics + ↑ ↓ + └─────────── Continuous Learning & Optimization ───────────┘ +``` + +## 🛠️ **The Complete Pipeline** + +### **Phase 1: Research & Ideation (10 minutes)** + +**1. Topic Research with Clara** +``` +Clara Agent Mode: +"Research trending topics in [your niche] and identify content gaps" + +→ Web search across multiple sources +→ Analyze competitor content +→ Identify trending keywords +→ Save findings to research Notebook +``` + +**2. Content Strategy Planning** +``` +Clara analyzes your research: +→ Suggests content angles +→ Identifies target audience +→ Recommends content formats +→ Plans content calendar +``` + +### **Phase 2: Content Creation (20-30 minutes)** + +**3. AI-Assisted Writing** +``` +Clara in Chat Mode: +"Write a comprehensive blog post about [topic] targeting [audience]" + +→ Creates detailed outline +→ Writes engaging introduction +→ Develops main content sections +→ Adds compelling conclusion +→ Optimizes for readability +``` + +**4. Custom Image Generation** +``` +ImageGen workflow: +→ Generate hero images matching your brand +→ Create infographics and diagrams +→ Design social media graphics +→ Generate thumbnail variations +→ All consistent with your visual style +``` + +**5. Content Optimization** +``` +Agent workflow for SEO: +→ LLM node: Analyze content for keywords +→ Structured LLM: Generate meta descriptions +→ Text processing: Create social media snippets +→ Output formatted content for each platform +``` + +### **Phase 3: Multi-Platform Publishing (15 minutes)** + +**6. Platform-Specific Formatting** +``` +N8N automation: +→ Transform content for each platform +→ Resize images for platform specs +→ Generate platform-specific hashtags +→ Schedule optimal posting times +``` + +**7. Cross-Platform Distribution** +``` +Automated publishing to: +→ Your blog/website (via LumaUI) +→ Medium, Dev.to, Hashnode +→ Twitter, LinkedIn, Instagram +→ YouTube (if video content) +→ Newsletter platforms +``` + +### **Phase 4: Analytics & Learning (5 minutes setup, automatic thereafter)** + +**8. Performance Tracking** +``` +N8N workflow monitors: +→ Website analytics +→ Social media engagement +→ Email open rates +→ SEO rankings +→ Stores data in Notebooks for analysis +``` + +**9. Continuous Improvement** +``` +Clara analyzes performance data: +→ Identifies top-performing content types +→ Suggests optimization strategies +→ Recommends topics for future content +→ Updates content strategy based on data +``` + +## 🎯 **Real Example: Tech Blog Workflow** + +**Input**: "AI productivity tools" + +**Research Phase** (10 minutes): +- Clara finds 50+ AI tools trending +- Identifies gap: "Local AI tools comparison" +- Suggests 5 different content angles + +**Creation Phase** (25 minutes): +- 2,500-word comprehensive guide +- 8 custom comparison infographics +- Social media carousel posts +- Email newsletter version + +**Distribution Phase** (15 minutes): +- Published to personal blog +- Cross-posted to 4 platforms +- Scheduled 20 social media posts +- Newsletter sent to subscribers + +**Results**: 50,000+ views, 500+ new subscribers, 20+ business inquiries + +## 🔄 **Advanced Workflows** + +### **Video Content Pipeline** +``` +Research → Script (Clara) → Visuals (ImageGen) → Audio (TTS) → Editing → Publishing +``` + +### **Newsletter Automation** +``` +Weekly trigger → Curate best content → Generate newsletter → Personalize for segments → Send +``` + +### **Seasonal Content Planning** +``` +Calendar triggers → Research seasonal trends → Plan content series → Schedule creation +``` + +## 💰 **ROI Breakdown** + +**Traditional Content Creation:** +- Research: 3 hours +- Writing: 4 hours +- Design: 2 hours +- SEO: 1 hour +- Publishing: 2 hours +- **Total: 12 hours per piece** + +**ClaraVerse Content Factory:** +- Setup: 1 hour +- Creation: 1 hour +- Review & publish: 30 minutes +- **Total: 1.5 hours per piece** + +**8x time savings + unlimited custom images + automated distribution** + +## 🎨 **Customization Examples** + +### **For Course Creators:** +``` +Course outline → Lesson scripts → Slide graphics → Marketing materials → Student resources +``` + +### **For E-commerce:** +``` +Product features → Blog posts → Product images → Social proof → Email sequences +``` + +### **For Consultants:** +``` +Industry insights → Thought leadership → Case studies → Lead magnets → Client materials +``` + +### **For Agencies:** +``` +Client briefs → Campaign content → Visual assets → Performance reports → Strategy updates +``` + +## 🛠️ **Setup Guide** + +### **Essential Components:** +1. **Clara Assistant**: Content writing and strategy +2. **ImageGen**: Custom visual creation +3. **Notebooks**: Research and analytics storage +4. **N8N**: Publishing automation +5. **LumaUI**: Website/blog management + +### **Optional Integrations:** +- Social media APIs (Twitter, LinkedIn, etc.) +- Email marketing platforms (Mailchimp, ConvertKit) +- Analytics tools (Google Analytics, Plausible) +- SEO tools (SEMrush, Ahrefs APIs) + +### **Initial Setup (30 minutes):** +1. Configure social media connections in N8N +2. Set up content calendar in Notebooks +3. Create brand style guide for ImageGen +4. Test publishing workflow with sample content + +## 📊 **Content Types This Handles** + +**Written Content:** +- Blog posts and articles +- Social media posts +- Email newsletters +- Product descriptions +- Case studies and whitepapers + +**Visual Content:** +- Blog header images +- Social media graphics +- Infographics and charts +- Product mockups +- Brand illustrations + +**Interactive Content:** +- Polls and surveys +- Interactive infographics +- Calculators and tools +- Quizzes and assessments + +## 🚨 **Pro Tips & Common Mistakes** + +### **What Works:** +- Start with one platform, expand gradually +- Build templates for consistent style +- Monitor analytics to refine approach +- Batch create content when inspired + +### **Common Pitfalls:** +- Over-automating (maintain human touch) +- Ignoring platform-specific best practices +- Not customizing AI output enough +- Forgetting to update brand guidelines + +### **Quality Control:** +- Always review AI-generated content +- Test images across all platforms +- Check links and formatting +- Monitor brand voice consistency + +## 📈 **Success Metrics** + +Track these KPIs to measure success: +- **Time saved**: Hours per content piece +- **Content volume**: Pieces published per week +- **Engagement**: Average likes, shares, comments +- **Reach**: Total impressions across platforms +- **Conversions**: Newsletter signups, sales, inquiries +- **Cost savings**: Design and tool subscriptions eliminated + +## 🔄 **Scaling Strategies** + +### **Individual Creator → Small Team:** +- Share templates and workflows +- Assign platform specialists +- Create approval processes +- Scale image generation + +### **Small Team → Content Agency:** +- Multi-client workflows +- Brand-specific templates +- Performance dashboards +- Client reporting automation + +## 🤝 **What Content Creators Are Actually Doing** + +**Realistic Content Production Goals:** +- **Beginner**: 1-2 high-quality posts per week with AI assistance +- **Intermediate**: Daily content across 2-3 platforms with automation +- **Advanced**: Multi-platform content with custom visuals and scheduling + +**Time Investment You Can Expect:** +- **Setup**: 2-4 hours to configure your content pipeline +- **Weekly maintenance**: 3-5 hours for content review and strategy +- **Content creation**: 1-2 hours per piece vs 4-6 hours manual +- **Platform management**: Mostly automated after initial setup + +**Realistic Quality Improvements:** +- More consistent posting schedule +- Better visual consistency across platforms +- Faster iteration on content ideas +- More time for strategic thinking and engagement + +## 🔧 **Advanced Features** + +### **AI Content Personalization:** +``` +Reader data → Content preferences → Personalized versions → A/B testing +``` + +### **Trend Prediction:** +``` +Historical data → Pattern analysis → Trend forecasting → Content calendar +``` + +### **Competitive Intelligence:** +``` +Competitor monitoring → Gap analysis → Content opportunities → Strategic positioning +``` + +--- + +**Ready to build your content empire? Start with one piece and watch the pipeline work its magic.** diff --git a/docs/ideas/customer-support.md b/docs/ideas/customer-support.md new file mode 100644 index 00000000..dc4c99e4 --- /dev/null +++ b/docs/ideas/customer-support.md @@ -0,0 +1,430 @@ +--- +title: "Customer Support Bot" +description: "Intelligent customer support using your company knowledge base" +category: "ideas" +order: 5 +lastUpdated: "2025-09-06" +contributors: ["badboysm890"] +--- + +# 🤖 Customer Support Bot + +**Transform customer support with AI that actually knows your business** + +This workflow shows how small businesses and teams use ClaraVerse to create intelligent customer support that works 24/7, understands context, and gets smarter over time—all while keeping customer data private. + +## 💡 **The Problem** + +Customer support is expensive and inconsistent: +- Hiring support staff costs $30-50K+ per person +- Response times vary wildly +- Knowledge is scattered across team members +- Customers repeat the same questions +- Support quality depends on who answers +- Escalation processes are slow +- Documentation gets outdated quickly + +**Most small businesses can't afford quality support, most customers get frustrated waiting.** + +## 🎯 **The ClaraVerse Solution** + +Create an AI support system that knows your business better than most employees: + +``` +Knowledge Base → AI Training → Customer Query → Intelligent Response → Learning + ↑ ↓ + └────────── Continuous Improvement & Human Oversight ──────────┘ +``` + +## 🛠️ **The Complete Support System** + +### **Phase 1: Knowledge Foundation (2-3 hours setup)** + +**1. Company Knowledge Collection** +``` +Gather all support materials: +→ FAQ documents +→ Product documentation +→ Previous support tickets +→ Internal procedures +→ Common troubleshooting steps +→ Product manuals and guides +``` + +**2. Knowledge Base Creation** +``` +Upload to Notebooks: +→ All support documents (PDFs, docs, etc.) +→ Company policies and procedures +→ Product information and specs +→ Common problem solutions +→ Escalation procedures +``` + +**3. AI Training & Testing** +``` +Clara learns your business: +→ Understands product features +→ Knows company policies +→ Recognizes common issues +→ Learns your support tone/style +→ Tests responses against real scenarios +``` + +### **Phase 2: Support Bot Creation (1-2 hours)** + +**4. Intelligent Response System** +``` +Agent workflow for support: +→ Input: Customer query +→ LLM node: Analyze query intent +→ Notebook search: Find relevant information +→ Structured LLM: Generate helpful response +→ Quality check: Ensure accuracy +→ Output: Customer-ready answer +``` + +**5. Multi-Channel Integration** +``` +N8N connects to: +→ Website chat widget +→ Email support inbox +→ Social media messages +→ Help desk platforms +→ Slack for internal notifications +``` + +### **Phase 3: Smart Routing & Escalation (30 minutes)** + +**6. Complexity Assessment** +``` +AI determines if query needs: +→ Automated response (90% of cases) +→ Human review (complex issues) +→ Immediate escalation (urgent problems) +→ Technical specialist (product bugs) +``` + +**7. Human Handoff Process** +``` +When escalation needed: +→ Full context provided to human agent +→ Suggested response drafts +→ Previous customer history +→ Relevant documentation linked +→ Priority level assigned +``` + +### **Phase 4: Continuous Learning (Automated)** + +**8. Performance Monitoring** +``` +Track key metrics: +→ Response accuracy +→ Customer satisfaction +→ Resolution time +→ Escalation rate +→ Common query patterns +``` + +**9. Knowledge Updates** +``` +Automated improvement: +→ New support tickets analyzed +→ Knowledge gaps identified +→ Documentation updates suggested +→ AI training refined +→ Response quality improved +``` + +## 🎯 **Realistic Implementation Example** + +**Scenario**: Small SaaS company with basic support needs + +**What you'd actually set up**: +- Upload your existing FAQ and documentation to Notebooks +- Train Clara on your product's common issues +- Integrate with your existing chat or email system +- Test with internal team before going live + +**Realistic expectations after 3 months**: +- **Basic queries**: 70-80% handled automatically (password resets, billing questions) +- **Response time**: Instant for documented issues +- **Human escalation**: 20-30% for complex or sensitive issues +- **Time saved**: 10-15 hours per week on routine support + +**What this means**: +- Your team focuses on complex problems and customer success +- Customers get faster responses to common questions +- You have more time for product development +- Support quality becomes more consistent + +## 🔄 **Industry-Specific Adaptations** + +### **E-commerce Support:** +``` +Product catalog integration → Order status queries → Shipping information → Return policies +``` + +### **SaaS Support:** +``` +Feature documentation → Billing questions → Technical troubleshooting → Integration help +``` + +### **Service Business:** +``` +Appointment scheduling → Service descriptions → Pricing information → Location details +``` + +### **Consulting Support:** +``` +Service offerings → Case studies → Process explanations → Proposal information +``` + +## 💰 **ROI Analysis** + +**Traditional Support Costs:** +- Full-time support agent: $45,000/year +- Support software: $2,400/year +- Training and management: $8,000/year +- **Total: $55,400/year per agent** + +**ClaraVerse Support Bot:** +- Initial setup: 8 hours × $50/hour = $400 +- Monthly maintenance: 2 hours × $50/hour = $100 +- Optional cloud AI: $50/month +- **Total: $1,600/year** + +**97% cost reduction + 24/7 availability + consistent quality** + +## 🛠️ **Technical Implementation** + +### **Core Components:** +1. **Notebooks**: Company knowledge storage with RAG +2. **Clara Assistant**: Natural language understanding +3. **Agents**: Automated response generation +4. **N8N**: Multi-channel integration +5. **LumaUI**: Customer portal (optional) + +### **Integration Options:** +- **Website chat**: JavaScript widget +- **Email**: IMAP/SMTP integration +- **Social media**: Twitter, Facebook APIs +- **Help desk**: Zendesk, Freshdesk, Intercom +- **CRM**: Salesforce, HubSpot connections + +### **Response Quality Controls:** +- Confidence scoring for all responses +- Human review for low-confidence answers +- A/B testing for response variations +- Customer feedback collection +- Regular accuracy audits + +## 📊 **Support Categories Handled** + +### **Automatically Resolved (85-90%):** +- Product information requests +- Pricing and billing questions +- Account access issues +- Feature explanations +- Policy clarifications +- Status updates +- Simple troubleshooting + +### **Human Review Required (8-12%):** +- Complex technical issues +- Account modifications +- Refund requests +- Escalated complaints +- Custom integration help + +### **Immediate Escalation (2-3%):** +- Security concerns +- Legal issues +- Executive complaints +- System outages +- Data loss reports + +## 🎨 **Customer Experience Features** + +### **Intelligent Conversations:** +``` +Customer: "I can't log in" +Bot: "I can help with login issues. Are you seeing an error message, or is the page not loading?" + +Customer: "It says invalid password" +Bot: "Let's reset your password. I'll send you a secure reset link to [email]. Check your inbox in 2-3 minutes." +``` + +### **Proactive Support:** +``` +System detects: +→ User struggling with feature +→ Unusual error patterns +→ Service disruptions +→ Account anomalies + +Auto-reaches out with: +→ Help documentation +→ Tutorial videos +→ Direct assistance offers +→ Service status updates +``` + +### **Contextual Assistance:** +``` +Bot knows: +→ Customer's product plan +→ Previous conversations +→ Account history +→ Current session activity +→ Technical environment +``` + +## 🚨 **Common Implementation Challenges** + +### **Knowledge Quality Issues:** +**Problem**: AI gives incorrect information +**Solution**: Regular knowledge audits, human verification workflows + +### **Integration Complexity:** +**Problem**: Multiple support channels hard to manage +**Solution**: Start with one channel, expand gradually + +### **Customer Resistance:** +**Problem**: Customers prefer human agents +**Solution**: Transparent bot identification, easy human handoff + +### **Scope Creep:** +**Problem**: Bot expected to handle everything +**Solution**: Clear capability communication, proper escalation + +## 📈 **Success Metrics to Track** + +### **Operational Metrics:** +- **First Contact Resolution Rate**: % queries solved without escalation +- **Average Response Time**: Speed of initial response +- **Customer Satisfaction Score**: Feedback on bot interactions +- **Escalation Rate**: % requiring human intervention +- **Knowledge Base Hit Rate**: How often docs provide answers + +### **Business Metrics:** +- **Support Cost per Ticket**: Total support costs ÷ tickets handled +- **Agent Productivity**: Complex tickets resolved per human agent +- **Customer Retention**: Impact on churn rates +- **Upsell Opportunities**: Bot-identified sales opportunities + +## 🔧 **Setup Phases** + +### **Phase 1: Foundation (Week 1)** +- Collect and organize support documentation +- Set up Notebooks with company knowledge +- Train Clara on your support style +- Create basic response templates + +### **Phase 2: Integration (Week 2)** +- Connect to primary support channel +- Build escalation workflows +- Test with internal team +- Refine response quality + +### **Phase 3: Launch (Week 3)** +- Deploy to subset of customers +- Monitor performance closely +- Collect feedback and iterate +- Document best practices + +### **Phase 4: Optimization (Ongoing)** +- Expand to additional channels +- Add proactive support features +- Build analytics dashboards +- Scale to handle growth + +## 🤝 **Team Collaboration Features** + +### **Internal Support Dashboard:** +``` +LumaUI creates: +→ Real-time support metrics +→ Query pattern analysis +→ Knowledge gap reports +→ Team performance tracking +→ Customer satisfaction trends +``` + +### **Agent Assistance Tools:** +``` +When humans handle tickets: +→ Suggested responses from AI +→ Relevant documentation links +→ Customer history summary +→ Similar resolved cases +→ Escalation recommendations +``` + +### **Knowledge Management:** +``` +Team collaboration on: +→ Documentation updates +→ FAQ improvements +→ Policy clarifications +→ Training material creation +→ Best practice sharing +``` + +## 🔄 **Advanced Features** + +### **Multilingual Support:** +``` +N8N translation pipeline: +→ Detect customer language +→ Translate to English for processing +→ Generate response in English +→ Translate back to customer language +→ Maintain context throughout +``` + +### **Sentiment Analysis:** +``` +Emotion detection: +→ Frustrated customers get priority +→ Happy customers get upsell opportunities +→ Angry customers get immediate escalation +→ Confused customers get extra help +``` + +### **Predictive Support:** +``` +Pattern recognition: +→ Predict likely customer issues +→ Proactive documentation updates +→ Feature improvement suggestions +→ Product roadmap insights +``` + +## 🎯 **Getting Started Checklist** + +### **Before You Begin:** +- [ ] Collect all support documentation +- [ ] Define escalation procedures +- [ ] Choose primary support channel +- [ ] Set quality standards +- [ ] Plan success metrics + +### **Week 1 Setup:** +- [ ] Upload documents to Notebooks +- [ ] Train Clara on support scenarios +- [ ] Create basic agent workflows +- [ ] Test with team members +- [ ] Refine responses + +### **Launch Preparation:** +- [ ] Integrate with support platform +- [ ] Set up monitoring dashboards +- [ ] Train team on new process +- [ ] Create customer communication +- [ ] Plan feedback collection + +--- + +**Ready to transform your customer support? Start with your most common questions and watch AI take over the routine work.** diff --git a/docs/ideas/local-code-review.md b/docs/ideas/local-code-review.md new file mode 100644 index 00000000..bdb31042 --- /dev/null +++ b/docs/ideas/local-code-review.md @@ -0,0 +1,472 @@ +--- +title: "Local AI Code Review" +description: "Private code analysis and review using local AI models" +category: "ideas" +order: 7 +lastUpdated: "2025-09-06" +contributors: ["badboysm890"] +--- + +# 🔍 Local AI Code Review + +**Enterprise-grade code review that never leaves your machine** + +This workflow shows how development teams use ClaraVerse to get intelligent code analysis, security scanning, and improvement suggestions without ever sending code to external services. + +## 💡 **The Problem** + +Code review is crucial but challenging: +- Manual reviews miss subtle bugs and security issues +- GitHub Copilot and similar tools send code to cloud +- Security-sensitive code can't use cloud AI services +- Code review quality varies by reviewer expertise +- Large codebases overwhelm human reviewers +- Best practices enforcement is inconsistent +- Junior developers need more guidance + +**Teams need AI-powered code review that respects privacy and security.** + +## 🎯 **The ClaraVerse Solution** + +Comprehensive code analysis using entirely local AI: + +``` +Code Commit → Local AI Analysis → Security Scan → Quality Check → Report → Learning + ↑ ↓ + └────────── Team Standards & Continuous Improvement ──────────┘ +``` + +## 🛠️ **The Complete Code Review System** + +### **Phase 1: Setup & Standards (1-2 hours)** + +**1. Coding Standards Documentation** +``` +Upload to Notebooks: +→ Team coding style guides +→ Security requirements +→ Architecture patterns +→ Best practices documentation +→ Previous code review examples +→ Industry standards (OWASP, etc.) +``` + +**2. Repository Integration** +``` +Agent workflow monitors: +→ Git repositories for new commits +→ Pull request creation +→ Branch protection rules +→ Code change patterns +→ Developer activity +``` + +**3. Local Model Configuration** +``` +Clara Core optimized for: +→ Code analysis and understanding +→ Security vulnerability detection +→ Performance optimization suggestions +→ Documentation generation +→ Test case recommendations +``` + +### **Phase 2: Automated Analysis (2-5 minutes per review)** + +**4. Multi-Layer Code Analysis** +``` +Agent workflow performs: +→ Syntax and style checking +→ Logic flow analysis +→ Security vulnerability scanning +→ Performance bottleneck identification +→ Documentation completeness check +→ Test coverage assessment +``` + +**5. Intelligent Issue Detection** +``` +Clara identifies: +→ Potential bugs and edge cases +→ Security vulnerabilities +→ Performance issues +→ Code smells and anti-patterns +→ Maintainability concerns +→ Accessibility problems +``` + +**6. Context-Aware Suggestions** +``` +AI considers: +→ Project architecture +→ Existing code patterns +→ Team coding standards +→ Performance requirements +→ Security constraints +→ Business logic context +``` + +### **Phase 3: Human-AI Collaboration (10-15 minutes)** + +**7. Structured Review Reports** +``` +Generated reports include: +→ Executive summary of changes +→ Critical issues requiring immediate attention +→ Improvement suggestions with explanations +→ Code quality metrics +→ Security assessment +→ Performance impact analysis +``` + +**8. Interactive Review Process** +``` +Reviewers can: +→ Ask Clara questions about specific code +→ Get explanations for suggested changes +→ Discuss alternative implementations +→ Validate security concerns +→ Understand performance implications +``` + +### **Phase 4: Learning & Improvement (Automated)** + +**9. Team Learning System** +``` +Continuous improvement through: +→ Tracking common issue patterns +→ Updating team standards +→ Sharing best practices +→ Building institutional knowledge +→ Training junior developers +``` + +**10. Quality Metrics Tracking** +``` +Monitor progress on: +→ Bug detection rate +→ Security vulnerability trends +→ Code quality improvements +→ Review efficiency gains +→ Developer skill advancement +``` + +## 🎯 **Real Example: Fintech Startup** + +**Company**: Payment processing platform (regulatory compliance critical) + +**Challenge**: +- Can't use cloud AI due to PCI DSS requirements +- 5-person dev team, varying experience levels +- Need thorough security review for every change +- Manual reviews taking 2+ hours per PR + +**ClaraVerse Implementation**: +``` +Setup: +→ Uploaded PCI DSS compliance documentation +→ Added company security standards +→ Configured for financial services patterns +→ Integrated with GitLab workflows + +Analysis Pipeline: +→ Automatic PR analysis on creation +→ Security-focused scanning +→ Compliance checking +→ Performance impact assessment +``` + +**Results after 3 months**: +- **Review time**: Reduced from 2+ hours to 30 minutes +- **Bug catch rate**: 40% improvement in pre-production bug detection +- **Security issues**: 0 security vulnerabilities in production +- **Developer growth**: Junior devs producing senior-quality code +- **Compliance**: 100% PCI DSS compliance maintained + +## 🔄 **Language & Framework Specializations** + +### **JavaScript/TypeScript:** +``` +Specialized analysis for: +→ React component patterns +→ Node.js security issues +→ TypeScript type safety +→ Bundle size optimization +→ Async/await best practices +``` + +### **Python:** +``` +Focus areas: +→ Django/Flask security patterns +→ Data pipeline optimization +→ ML model validation +→ Package dependency analysis +→ PEP 8 compliance +``` + +### **Java/Kotlin:** +``` +Enterprise patterns: +→ Spring framework best practices +→ Memory management optimization +→ Security vulnerability scanning +→ Performance bottleneck detection +→ Architecture pattern validation +``` + +### **Go:** +``` +Cloud-native focus: +→ Concurrency pattern analysis +→ Error handling best practices +→ Performance optimization +→ Security scanning +→ Microservice patterns +``` + +## 🛡️ **Security Analysis Features** + +### **Vulnerability Detection:** +- **Input validation**: SQL injection, XSS prevention +- **Authentication**: Secure login patterns +- **Authorization**: Access control validation +- **Data encryption**: At-rest and in-transit protection +- **API security**: Rate limiting, input sanitization +- **Dependency scanning**: Known vulnerability checking + +### **Compliance Checking:** +- **GDPR**: Data privacy compliance +- **HIPAA**: Healthcare data protection +- **PCI DSS**: Payment card industry standards +- **SOX**: Financial reporting controls +- **ISO 27001**: Information security management + +### **Custom Security Rules:** +``` +Company-specific policies: +→ Internal API usage patterns +→ Database access restrictions +→ Third-party integration rules +→ Logging and monitoring requirements +→ Incident response procedures +``` + +## 💰 **Cost & Security Comparison** + +**Cloud-Based Code Review:** +- **SonarCloud**: $10+ per developer/month +- **GitHub Advanced Security**: $49 per committer/month +- **Veracode**: $500+ per application/month +- **Security risk**: Code uploaded to external servers +- **Compliance issues**: May violate regulatory requirements + +**ClaraVerse Local Review:** +- **Setup time**: 4-8 hours initial configuration +- **Ongoing cost**: $0 monthly (local processing) +- **Security**: Code never leaves your environment +- **Compliance**: Meets strictest requirements +- **Customization**: Unlimited rule customization + +## 🎨 **Review Report Examples** + +### **Security-Focused Report:** +```markdown +# Security Review Summary +**Risk Level: MEDIUM** + +## Critical Issues (0) +No critical security vulnerabilities detected. + +## High Priority (2) +1. **SQL Injection Risk** (Line 45) + - Unsanitized user input in database query + - Recommendation: Use parameterized queries + +2. **Sensitive Data Logging** (Line 78) + - PII potentially logged in error messages + - Recommendation: Sanitize error logs + +## Recommendations (5) +- Implement rate limiting on API endpoints +- Add input validation middleware +- Update dependency with security patch +... +``` + +### **Performance-Focused Report:** +```markdown +# Performance Review Summary +**Performance Impact: LOW** + +## Optimizations Identified (3) +1. **Database Query Optimization** (Line 23) + - N+1 query pattern detected + - Estimated impact: 200ms reduction per request + +2. **Memory Usage** (Line 67) + - Large object creation in loop + - Recommendation: Object pooling pattern +... +``` + +## 🛠️ **Setup for Different Team Sizes** + +### **Solo Developer:** +``` +Minimal setup: +→ Basic Clara Core configuration +→ Personal coding standards in Notebooks +→ Simple commit analysis workflow +→ Focus on learning and improvement +``` + +### **Small Team (2-10 developers):** +``` +Team collaboration: +→ Shared coding standards +→ Code review assignment automation +→ Team metrics dashboard +→ Knowledge sharing workflows +``` + +### **Medium Team (10-50 developers):** +``` +Scaled processes: +→ Multiple repository monitoring +→ Team-specific rule sets +→ Advanced metrics and reporting +→ Integration with project management +``` + +### **Enterprise (50+ developers):** +``` +Enterprise features: +→ Department-level customization +→ Compliance reporting automation +→ Advanced analytics and trends +→ Integration with enterprise tools +``` + +## 📊 **Quality Metrics Dashboard** + +### **Code Quality Trends:** +``` +LumaUI dashboard shows: +→ Bug detection rate over time +→ Security vulnerability trends +→ Code complexity metrics +→ Test coverage progression +→ Technical debt accumulation +``` + +### **Team Performance:** +``` +Developer insights: +→ Individual improvement tracking +→ Review turnaround times +→ Common mistake patterns +→ Learning progress metrics +→ Knowledge sharing contributions +``` + +### **Project Health:** +``` +Repository analytics: +→ Overall code quality score +→ Security posture assessment +→ Performance trend analysis +→ Maintainability index +→ Documentation completeness +``` + +## 🚨 **Best Practices & Common Issues** + +### **What Works:** +- Start with existing team standards +- Focus on security-critical areas first +- Use AI suggestions as starting points, not final decisions +- Regular model retraining with team feedback +- Gradual introduction to avoid overwhelming developers + +### **Common Pitfalls:** +- Over-relying on AI without human judgment +- Not customizing rules for project context +- Ignoring false positives instead of refining +- Forgetting to update standards as project evolves +- Not involving team in rule creation + +### **Quality Assurance:** +- Regular validation of AI suggestions +- Team feedback loops for improvement +- False positive tracking and reduction +- Rule effectiveness measurement +- Continuous model optimization + +## 🔧 **Integration Examples** + +### **GitLab Integration:** +``` +N8N workflow: +→ Monitor GitLab webhooks +→ Trigger analysis on PR creation +→ Post review comments automatically +→ Update PR status based on results +→ Generate compliance reports +``` + +### **GitHub Integration:** +``` +GitHub Actions + ClaraVerse: +→ Automated PR analysis +→ Security gate enforcement +→ Quality metrics collection +→ Team notification system +→ Deployment blocking for critical issues +``` + +### **Jenkins Integration:** +``` +CI/CD pipeline enhancement: +→ Pre-merge quality gates +→ Automated testing suggestions +→ Deployment risk assessment +→ Rollback decision support +→ Production monitoring alerts +``` + +## 📈 **Advanced Features** + +### **Architectural Analysis:** +``` +System-level insights: +→ Design pattern compliance +→ Dependency management +→ Modularity assessment +→ Scalability considerations +→ Maintainability predictions +``` + +### **Technical Debt Management:** +``` +Debt tracking: +→ Code complexity trends +→ Refactoring opportunities +→ Legacy code identification +→ Modernization planning +→ ROI analysis for improvements +``` + +### **Learning & Development:** +``` +Developer growth: +→ Skill gap identification +→ Personalized learning recommendations +→ Code review training +→ Best practice examples +→ Mentoring suggestions +``` + +--- + +**Ready to revolutionize your code review process? Start with a single repository and experience the power of local AI analysis.** diff --git a/docs/ideas/personal-brand.md b/docs/ideas/personal-brand.md new file mode 100644 index 00000000..3e796538 --- /dev/null +++ b/docs/ideas/personal-brand.md @@ -0,0 +1,423 @@ +--- +title: "Personal Brand Builder" +description: "Automated personal brand development with consistent content and visual identity" +category: "ideas" +order: 6 +lastUpdated: "2025-09-06" +contributors: ["badboysm890"] +--- + +# 🎯 Personal Brand Builder + +**Build a compelling personal brand that works for you 24/7** + +This workflow shows how professionals, freelancers, and entrepreneurs use ClaraVerse to develop and maintain a strong personal brand across all platforms—automatically generating consistent content, visuals, and messaging. + +## 💡 **The Problem** + +Building a personal brand is overwhelming: +- Consistent content creation takes hours daily +- Visual design requires expensive tools or skills +- Message consistency across platforms is hard +- Personal story development feels unclear +- Social proof generation is slow +- Network building is time-intensive +- Opportunity tracking is fragmented + +**Most people either burn out trying or never start building their brand.** + +## 🎯 **The ClaraVerse Solution** + +Automate the mechanics, amplify your authentic voice: + +``` +Personal Story → Content Strategy → Visual Identity → Content Creation → Distribution → Engagement + ↑ ↓ + └─────────────── Brand Evolution & Opportunity Capture ──────────────────────┘ +``` + +## 🛠️ **The Complete Brand Building System** + +### **Phase 1: Brand Foundation (2-3 hours setup)** + +**1. Personal Brand Discovery** +``` +Clara-guided brand workshop: +→ "What are your core values and expertise?" +→ "Who is your target audience?" +→ "What transformation do you help people achieve?" +→ "What's your unique perspective or approach?" +→ "What's your professional origin story?" +``` + +**2. Brand Strategy Development** +``` +Clara analyzes your responses: +→ Identifies key brand pillars +→ Develops core messaging framework +→ Creates audience personas +→ Suggests content themes +→ Plans brand positioning strategy +``` + +**3. Visual Identity Creation** +``` +ImageGen generates: +→ Professional headshots (AI-enhanced) +→ Brand color palette variations +→ Logo concepts and variations +→ Social media templates +→ Presentation slide templates +``` + +### **Phase 2: Content Engine Setup (1-2 hours)** + +**4. Content Calendar Planning** +``` +Agent workflow creates: +→ Weekly content themes +→ Platform-specific posting schedules +→ Content type rotation (articles, posts, videos) +→ Seasonal campaign planning +→ Event-based content opportunities +``` + +**5. Automated Content Creation** +``` +Daily content generation: +→ LinkedIn thought leadership posts +→ Twitter insights and commentary +→ Blog article ideas and outlines +→ Newsletter content suggestions +→ Social media graphics and quotes +``` + +**6. Personal Story Mining** +``` +Clara helps extract: +→ Career milestone stories +→ Lessons learned narratives +→ Client success stories +→ Industry insight pieces +→ Behind-the-scenes content +``` + +### **Phase 3: Multi-Platform Presence (30 minutes daily)** + +**7. Platform Optimization** +``` +Customized content for each platform: +→ LinkedIn: Professional insights +→ Twitter: Quick thoughts and engagement +→ Instagram: Visual storytelling +→ YouTube: Educational content +→ Personal website: Comprehensive showcase +``` + +**8. Automated Distribution** +``` +N8N workflow handles: +→ Cross-platform posting +→ Optimal timing for each platform +→ Platform-specific formatting +→ Hashtag optimization +→ Engagement tracking +``` + +### **Phase 4: Relationship Building (Automated + Strategic)** + +**9. Network Growth Strategy** +``` +Automated relationship building: +→ Identify industry influencers +→ Engage with target audience content +→ Comment on relevant discussions +→ Share valuable resources +→ Follow up on connections +``` + +**10. Opportunity Detection** +``` +AI monitors for: +→ Speaking opportunities +→ Collaboration requests +→ Media mentions +→ Industry trends to comment on +→ Networking events to attend +``` + +## 🎯 **Real Example: Freelance Designer Brand** + +**Starting Point**: Talented designer with no online presence + +**Brand Foundation** (3 hours): +- Defined niche: "Design systems for growing SaaS companies" +- Target audience: CTOs and product managers +- Unique angle: "Design that scales with your engineering team" +- Core values: Simplicity, scalability, user-centricity + +**Content Strategy**: +- Monday: Design system tips +- Wednesday: Case study breakdowns +- Friday: Industry trend analysis +- Weekly: Client transformation stories + +**Results after 6 months:** +- **LinkedIn**: 15,000 followers (from 200) +- **Portfolio inquiries**: 3-5 per week +- **Speaking engagements**: 8 conferences +- **Rate increase**: 300% (from $50/hr to $200/hr) +- **Client quality**: Now works with Series B+ companies + +## 🔄 **Brand Archetype Adaptations** + +### **The Expert/Educator:** +``` +Content focus: Educational posts, tutorials, industry insights, research sharing +Visual style: Clean, professional, data-focused graphics +Platform priority: LinkedIn, YouTube, industry publications +``` + +### **The Innovator/Thought Leader:** +``` +Content focus: Trend predictions, contrarian viewpoints, future vision +Visual style: Bold, forward-thinking, concept-driven imagery +Platform priority: Twitter, LinkedIn, speaking circuits +``` + +### **The Helper/Coach:** +``` +Content focus: Tips, transformations, success stories, motivation +Visual style: Warm, approachable, people-focused imagery +Platform priority: Instagram, LinkedIn, newsletter, podcasts +``` + +### **The Storyteller/Creative:** +``` +Content focus: Behind-scenes, personal journey, creative process +Visual style: Authentic, artistic, personality-driven content +Platform priority: Instagram, YouTube, TikTok, personal blog +``` + +## 💰 **Investment vs. Return** + +**Traditional Brand Building Costs:** +- Brand strategist: $5,000-15,000 +- Visual designer: $3,000-8,000 +- Content creator: $2,000-5,000/month +- Social media manager: $1,500-3,000/month +- PR agency: $3,000-10,000/month +- **Total: $20,000+ setup + $6,500-18,000/month** + +**ClaraVerse Personal Brand Builder:** +- Initial setup: 8 hours of your time +- Monthly maintenance: 4 hours + $50 cloud AI +- **Total: Your time + $600/year** + +**97% cost reduction + complete control + authentic voice** + +## 🎨 **Content Categories & Templates** + +### **Thought Leadership Posts:** +``` +Template structure: +→ Hook: Contrarian or surprising statement +→ Context: Industry situation or trend +→ Insight: Your unique perspective +→ Evidence: Data, example, or story +→ Call to action: Engagement question +``` + +### **Case Study Stories:** +``` +Template structure: +→ Challenge: Client's initial problem +→ Approach: Your methodology +→ Solution: What you implemented +→ Results: Measurable outcomes +→ Lesson: Key takeaway for audience +``` + +### **Industry Commentary:** +``` +Template structure: +→ Trend: What's happening in your field +→ Analysis: Why it matters +→ Prediction: Where it's heading +→ Action: What people should do +→ Discussion: Invite different perspectives +``` + +## 🛠️ **Technical Implementation** + +### **Core Components:** +1. **Clara Assistant**: Brand strategy and content creation +2. **ImageGen**: Visual content and brand assets +3. **Notebooks**: Brand guidelines and content library +4. **Agents**: Automated content generation +5. **N8N**: Cross-platform distribution +6. **LumaUI**: Personal website/portfolio + +### **Integration Setup:** +- **Social platforms**: LinkedIn, Twitter, Instagram APIs +- **Content management**: WordPress, Ghost, or custom blog +- **Email marketing**: ConvertKit, Mailchimp, Beehiiv +- **Analytics**: Google Analytics, social platform insights +- **Design tools**: Canva Pro API for template variations + +### **Automation Workflows:** +- **Daily content**: Generate and schedule posts +- **Weekly newsletter**: Curate best content and insights +- **Monthly review**: Analyze performance and adjust strategy +- **Quarterly planning**: Update brand strategy and goals + +## 📊 **Brand Building Metrics** + +### **Awareness Metrics:** +- **Follower growth rate**: New followers per month +- **Reach and impressions**: Content visibility +- **Brand mention tracking**: How often you're referenced +- **Search visibility**: Rankings for your name + expertise + +### **Engagement Metrics:** +- **Engagement rate**: Likes, comments, shares per post +- **Comment quality**: Depth of conversations sparked +- **DM/connection requests**: Direct outreach volume +- **Event invitations**: Speaking and collaboration requests + +### **Business Impact:** +- **Inquiry volume**: Leads generated through brand +- **Rate increases**: Premium pricing due to reputation +- **Opportunity quality**: Better clients and projects +- **Partnership offers**: Collaboration and joint venture requests + +## 🚨 **Common Brand Building Mistakes** + +### **What Doesn't Work:** +- Posting random content without strategy +- Copying others' voices instead of developing your own +- Focusing on vanity metrics over meaningful engagement +- Neglecting one platform while spreading too thin +- Being overly promotional instead of providing value + +### **What Works:** +- Consistent voice and messaging across all platforms +- 80% value-driven content, 20% promotional +- Engaging genuinely with your community +- Sharing personal insights and experiences +- Building relationships before asking for anything + +### **Quality Control:** +- Regular brand audit against core values +- Audience feedback collection and integration +- Content performance analysis and optimization +- Visual consistency checking across platforms +- Message alignment verification + +## 📈 **Growth Acceleration Strategies** + +### **Content Amplification:** +``` +Repurpose single ideas across formats: +→ LinkedIn article → Twitter thread → Instagram carousel → YouTube video → Newsletter section +``` + +### **Collaboration Network:** +``` +Build strategic partnerships: +→ Guest posting exchanges +→ Podcast interview swaps +→ Joint webinar hosting +→ Cross-promotion agreements +→ Mastermind group participation +``` + +### **Authority Building:** +``` +Systematic credibility development: +→ Original research publication +→ Industry survey creation +→ Expert roundtable hosting +→ Media interview cultivation +→ Award and recognition pursuit +``` + +## 🔄 **Long-term Brand Evolution** + +### **Year 1: Foundation** +- Establish consistent voice and visual identity +- Build core audience of 1,000+ engaged followers +- Create library of high-quality content +- Develop key relationships in your industry + +### **Year 2: Authority** +- Position as go-to expert in your niche +- Speaking opportunities and media features +- Strategic partnerships and collaborations +- Premium service offerings and higher rates + +### **Year 3: Influence** +- Industry thought leadership recognition +- Book deals, course creation, consulting offers +- Board positions and advisory roles +- Platform for social impact and change + +## 🤝 **What Personal Brand Building Actually Looks Like** + +**Realistic Timeline Expectations:** +- **Month 1-3**: Establish consistent voice and posting schedule +- **Month 3-6**: Build initial audience and engagement patterns +- **Month 6-12**: Start seeing opportunities and recognition +- **Year 1+**: Develop thought leadership and strategic partnerships + +**Common Early Wins:** +- More consistent content creation and messaging +- Better engagement on posts due to AI assistance +- Professional-looking visuals without design skills +- Automated posting saves 5-10 hours per week + +**Typical Growth Patterns:** +- **LinkedIn**: 500-2000 new connections in first 6 months +- **Content quality**: Noticeable improvement in clarity and consistency +- **Time efficiency**: 3-4x faster content creation with AI help +- **Opportunities**: Better quality networking and collaboration requests + +**What Won't Happen Overnight:** +- Instant authority or thought leadership +- Massive follower growth without consistent effort +- Speaking opportunities without proven expertise +- Rate increases without demonstrable value + +## 🔧 **Quick Start Guide** + +### **Week 1: Foundation** +1. Complete brand discovery workshop with Clara +2. Generate visual identity assets with ImageGen +3. Set up basic content calendar +4. Create profiles on 2-3 key platforms +5. Write compelling bio and positioning statements + +### **Week 2: Content Creation** +1. Generate first month of content using templates +2. Set up automated distribution workflows +3. Create lead magnets and valuable resources +4. Start engaging with target audience content +5. Publish first thought leadership pieces + +### **Week 3: Optimization** +1. Analyze early performance metrics +2. Refine content based on audience response +3. Adjust posting schedule for optimal engagement +4. Build relationships with key industry figures +5. Plan first major content campaign + +### **Week 4: Scaling** +1. Expand to additional platforms +2. Create advanced content formats (video, podcasts) +3. Set up email capture and nurturing sequences +4. Launch first collaborative project +5. Plan speaking or guest content opportunities + +--- + +**Ready to build a brand that works for you while you sleep? Start with your unique story and let AI amplify your authentic voice.** diff --git a/docs/ideas/research-assistant.md b/docs/ideas/research-assistant.md new file mode 100644 index 00000000..fc819a41 --- /dev/null +++ b/docs/ideas/research-assistant.md @@ -0,0 +1,427 @@ +--- +title: "Research Assistant" +description: "Academic research workflow with AI analysis and 3D knowledge mapping" +category: "ideas" +order: 4 +lastUpdated: "2025-09-06" +contributors: ["badboysm890"] +--- + +# 📊 Research Assistant + +**Transform how you conduct academic research with AI-powered analysis and visual knowledge mapping** + +This workflow shows how researchers, students, and knowledge workers use ClaraVerse to automate literature review, paper analysis, and knowledge synthesis with stunning 3D visualizations. + +## 💡 **The Problem** + +Academic research is overwhelming: +- Information overload from multiple sources +- Manual paper analysis takes hours per document +- Knowledge connections are hard to visualize +- Literature reviews require reading hundreds of papers +- Citation management is tedious +- Collaboration with colleagues is fragmented +- Insights get lost in folders and notes + +**Researchers spend 60% of time organizing information, 40% actually thinking.** + +## 🎯 **The ClaraVerse Solution** + +Flip the ratio: Let AI handle organization and discovery, focus your brain on insights and creativity. + +``` +Papers → AI Analysis → Knowledge Graph → Insights → Writing → Collaboration + ↑ ↓ + └──────────── Continuous Learning & Discovery ────────────────┘ +``` + +## 🛠️ **The Complete Research Pipeline** + +### **Phase 1: Discovery & Collection (15 minutes)** + +**1. Automated Literature Search** +``` +Clara Agent Mode: +"Find recent papers on [research topic] from top-tier journals" + +→ Searches academic databases +→ Filters by impact factor and recency +→ Downloads PDFs automatically +→ Creates initial bibliography +``` + +**2. Source Diversification** +``` +N8N workflow monitors: +→ ArXiv for preprints +→ Google Scholar alerts +→ Journal RSS feeds +→ Conference proceedings +→ Auto-imports to research Notebook +``` + +### **Phase 2: AI-Powered Analysis (30 minutes for 20+ papers)** + +**3. Batch Paper Processing** +``` +Agent workflow: +→ PDF Input node: Load research papers +→ LLM Analysis node: Extract key findings +→ Structured LLM node: Generate summaries +→ Notebook Writer: Save structured data +``` + +**4. Deep Content Analysis** +``` +Clara analyzes each paper for: +→ Main research question +→ Methodology used +→ Key findings and conclusions +→ Limitations and future work +→ Citation relationships +→ Novel contributions +``` + +**5. Cross-Paper Synthesis** +``` +Clara identifies: +→ Conflicting findings +→ Research gaps +→ Emerging trends +→ Methodological patterns +→ Citation clusters +``` + +### **Phase 3: Knowledge Visualization (10 minutes)** + +**6. 3D Knowledge Mapping** +``` +Notebooks' 3D visualization shows: +→ Concept relationships +→ Paper clustering by topic +→ Citation networks +→ Research evolution over time +→ Knowledge gaps as empty spaces +``` + +**7. Interactive Exploration** +``` +Navigate your research in 3D: +→ Zoom into specific topics +→ Follow citation pathways +→ Discover hidden connections +→ Identify research opportunities +``` + +### **Phase 4: Insight Generation (20 minutes)** + +**8. Research Question Development** +``` +Clara suggests: +→ Novel research angles +→ Unexplored combinations +→ Methodological improvements +→ Replication opportunities +→ Interdisciplinary connections +``` + +**9. Hypothesis Formation** +``` +Based on analysis, Clara helps: +→ Formulate testable hypotheses +→ Design research methodologies +→ Identify required resources +→ Plan experimental approaches +``` + +### **Phase 5: Writing & Communication (45 minutes)** + +**10. Literature Review Generation** +``` +Clara writes comprehensive reviews: +→ Synthesizes findings from all papers +→ Identifies themes and patterns +→ Highlights conflicting evidence +→ Suggests future research directions +→ Proper academic formatting +``` + +**11. Visual Research Summaries** +``` +ImageGen creates: +→ Concept maps and diagrams +→ Research timeline visualizations +→ Methodology flowcharts +→ Results infographics +→ Presentation slides +``` + +## 🎯 **Real Example: AI Ethics Research** + +**Research Question**: "What are the emerging ethical frameworks for AI in healthcare?" + +**Discovery** (15 minutes): +- Found 847 relevant papers from 2020-2025 +- Filtered to 156 high-impact publications +- Identified 23 key authors and research groups + +**Analysis** (45 minutes): +- AI extracted key ethical frameworks from all papers +- Identified 12 major ethical principles +- Found 7 conflicting viewpoints on data privacy +- Discovered gap in pediatric AI ethics research + +**Visualization**: +- 3D map showing evolution of ethical thinking +- Clear clusters around privacy, autonomy, fairness +- Visual timeline of regulatory development + +**Insights**: +- Novel research opportunity in AI ethics for children +- Methodology gap in longitudinal ethics studies +- Need for interdisciplinary collaboration framework + +**Output**: +- 15-page literature review with 156 citations +- Visual presentation with custom diagrams +- Research proposal for grant application + +## 🔄 **Specialized Research Workflows** + +### **Meta-Analysis Pipeline** +``` +Paper collection → Data extraction → Statistical analysis → Results synthesis → Visualization +``` + +### **Systematic Review Process** +``` +Search strategy → Study selection → Quality assessment → Data synthesis → PRISMA reporting +``` + +### **Interdisciplinary Research** +``` +Multi-domain search → Cross-field analysis → Integration opportunities → Novel connections +``` + +## 📚 **Research Domain Examples** + +### **Computer Science Research:** +- Algorithm performance comparisons +- Technology trend analysis +- Open source project evolution +- Conference paper tracking + +### **Medical Research:** +- Clinical trial analysis +- Drug interaction mapping +- Treatment efficacy reviews +- Epidemiological studies + +### **Social Sciences:** +- Survey data analysis +- Qualitative research synthesis +- Policy impact assessment +- Cultural trend studies + +### **Business Research:** +- Market analysis +- Competitive intelligence +- Industry trend forecasting +- Consumer behavior studies + +## 💰 **Time & Cost Savings** + +**Traditional Research Process:** +- Literature search: 8 hours +- Paper reading: 40 hours (2 hours × 20 papers) +- Note organization: 6 hours +- Analysis & synthesis: 12 hours +- Writing: 16 hours +- **Total: 82 hours** + +**ClaraVerse Research Assistant:** +- Setup & discovery: 2 hours +- AI analysis review: 4 hours +- Insight development: 6 hours +- Writing & refinement: 8 hours +- **Total: 20 hours** + +**75% time savings + better insight quality + visual knowledge mapping** + +## 🛠️ **Setup for Different Research Types** + +### **Academic Researchers:** +``` +Components needed: +→ Clara Assistant (paper analysis) +→ Notebooks (knowledge storage & 3D viz) +→ Agents (batch processing) +→ ImageGen (academic diagrams) +→ N8N (source monitoring) +``` + +### **Market Researchers:** +``` +Additional integrations: +→ Google Trends API +→ Social media monitoring +→ Survey data processing +→ Competitive intelligence feeds +``` + +### **Policy Researchers:** +``` +Specialized sources: +→ Government databases +→ Think tank publications +→ Legislative tracking +→ Policy impact analysis +``` + +## 📊 **Quality Assurance Features** + +### **Source Verification:** +- DOI validation +- Journal impact factor checking +- Author credential verification +- Citation accuracy confirmation + +### **Bias Detection:** +- Funding source analysis +- Methodology assessment +- Sample size evaluation +- Statistical significance checking + +### **Reproducibility Support:** +- Methodology documentation +- Data source tracking +- Analysis step recording +- Result verification protocols + +## 🎨 **Visual Research Outputs** + +### **Academic Presentations:** +- Conference presentation slides +- Poster layouts with data visualizations +- Interactive research timelines +- 3D knowledge network demonstrations + +### **Grant Applications:** +- Research gap visualizations +- Methodology flowcharts +- Expected impact diagrams +- Collaboration network maps + +### **Research Papers:** +- Figure generation for publications +- Conceptual framework diagrams +- Results visualization +- Literature map illustrations + +## 🚨 **Best Practices & Pitfalls** + +### **What Works:** +- Start broad, then narrow focus +- Validate AI insights with domain expertise +- Use 3D visualization to spot patterns +- Regular knowledge base updates + +### **Common Mistakes:** +- Over-relying on AI without verification +- Ignoring methodological differences +- Missing interdisciplinary connections +- Poor source quality control + +### **Quality Control:** +- Cross-reference key findings manually +- Verify citation accuracy +- Check for recent developments +- Validate statistical interpretations + +## 📈 **Advanced Features** + +### **Collaboration Tools:** +``` +Shared notebooks → Team analysis → Distributed writing → Version control +``` + +### **Real-time Monitoring:** +``` +Alert system → New paper notifications → Trend detection → Research updates +``` + +### **Impact Tracking:** +``` +Citation monitoring → Research influence → Network analysis → Career development +``` + +## 🔄 **Integration Examples** + +### **With University Systems:** +- Library database connections +- Student thesis supervision +- Faculty collaboration networks +- Research output tracking + +### **With Research Teams:** +- Shared knowledge bases +- Collaborative analysis +- Distributed literature reviews +- Team writing workflows + +### **With Publishing:** +- Journal submission automation +- Peer review management +- Impact factor tracking +- Open access optimization + +## 🤝 **What Researchers Can Realistically Expect** + +**Typical Research Workflow Improvements:** +- **Literature search**: 3-4x faster discovery of relevant papers +- **Paper analysis**: Quick extraction of key points from 10-20 papers at once +- **Knowledge organization**: Visual connections you might have missed +- **Writing support**: AI helps structure and synthesize findings + +**Time Investment vs. Benefits:** +- **Initial setup**: 4-6 hours to configure for your research domain +- **Learning curve**: 1-2 weeks to optimize prompts and workflows +- **Ongoing use**: 2-3 hours saved per research session +- **Quality improvement**: More comprehensive literature coverage + +**Realistic Outcomes:** +- Better organized research notes and references +- Faster identification of research gaps and opportunities +- More thorough literature reviews with visual knowledge mapping +- Improved synthesis of findings across multiple papers + +## 🔧 **Getting Started Guide** + +### **Week 1: Basic Setup** +1. Install Notebooks with RAG container +2. Configure Clara with academic prompts +3. Set up basic paper analysis workflow +4. Test with 5-10 papers in your field + +### **Week 2: Automation** +1. Create N8N workflows for source monitoring +2. Build agent pipelines for batch processing +3. Customize 3D visualization settings +4. Develop quality control checklists + +### **Week 3: Advanced Features** +1. Integrate with institutional databases +2. Set up collaboration workflows +3. Create custom analysis templates +4. Build presentation generation pipeline + +### **Week 4: Optimization** +1. Refine AI prompts for your domain +2. Optimize processing speed +3. Build domain-specific knowledge bases +4. Create reusable workflow templates + +--- + +**Ready to revolutionize your research process? Start with a small literature review and experience the power of AI-assisted discovery.** diff --git a/docs/images/0g-logo.svg b/docs/images/0g-logo.svg deleted file mode 100644 index 7c842d92..00000000 --- a/docs/images/0g-logo.svg +++ /dev/null @@ -1,4 +0,0 @@ - - - - diff --git a/docs/images/README.md b/docs/images/README.md deleted file mode 100644 index 4c802a01..00000000 --- a/docs/images/README.md +++ /dev/null @@ -1,68 +0,0 @@ -# ClaraVerse Branding & Assets - -This directory contains branding assets and screenshots for the ClaraVerse README and documentation. - -## Files - -### Branding Assets -- `logo.png` - Official ClaraVerse logo (19KB) -- `logo-web.png` - Web-optimized logo from claraverse.space (19KB) -- `banner.png` - Clara mascot banner (134KB) -- `clara-mascot.png` - Clara welcome mascot character (2.6MB) - -### Screenshots -- `screenshot-chat.webp` - Chat interface example -- `screenshot-agents.webp` - Multi-agent orchestration interface - -## Usage - -### In README.md -```markdown -![ClaraVerse Logo](docs/images/logo.png) -![Clara Banner](docs/images/banner.png) -``` - -### In Documentation -Reference images using relative paths: -```markdown -![Screenshot](../images/screenshot-chat.webp) -``` - -## Adding New Screenshots - -To add new screenshots: - -1. Take a screenshot of the feature at 1400px+ width -2. Optimize the image: - ```bash - # For PNG (use ImageMagick) - convert screenshot.png -quality 85 -resize 1400x screenshot-optimized.png - - # For WebP (recommended) - cwebp -q 80 screenshot.png -o screenshot.webp - ``` -3. Save to this directory with a descriptive name -4. Update this README with the new file - -## Brand Colors - -Based on claraverse.space: - -- **Primary (Sakura)**: `#f43f5e` - Use for CTAs and highlights -- **Background**: `#fafaf9` (Stone-50) -- **Text Primary**: `#1c1917` (Stone-900) -- **Text Secondary**: `#78716c` (Stone-500) - -## Logo Guidelines - -- Minimum size: 32px height -- Recommended sizes: - - Header: 120px width - - Footer: 80px width - - Social media: 512x512px square -- Maintain aspect ratio -- Use on light backgrounds for best visibility - -## Official Branding - -All assets are sourced from [claraverse.space](https://claraverse.space) and should match the official branding guidelines. diff --git a/docs/images/banner-full.png b/docs/images/banner-full.png deleted file mode 100644 index 4be21b29..00000000 Binary files a/docs/images/banner-full.png and /dev/null differ diff --git a/docs/images/banner.png b/docs/images/banner.png deleted file mode 100644 index b0caaeb4..00000000 Binary files a/docs/images/banner.png and /dev/null differ diff --git a/docs/images/chutes-logo.jpg b/docs/images/chutes-logo.jpg deleted file mode 100644 index a7372d55..00000000 Binary files a/docs/images/chutes-logo.jpg and /dev/null differ diff --git a/docs/images/clara-blog-announcement.png b/docs/images/clara-blog-announcement.png deleted file mode 100644 index 67d1904d..00000000 Binary files a/docs/images/clara-blog-announcement.png and /dev/null differ diff --git a/docs/images/clara-lean-updated.png b/docs/images/clara-lean-updated.png deleted file mode 100644 index b0caaeb4..00000000 Binary files a/docs/images/clara-lean-updated.png and /dev/null differ diff --git a/docs/images/clara-sun.png b/docs/images/clara-sun.png deleted file mode 100644 index 4be21b29..00000000 Binary files a/docs/images/clara-sun.png and /dev/null differ diff --git a/docs/images/image-1.png b/docs/images/image-1.png deleted file mode 100644 index 157bd8ff..00000000 Binary files a/docs/images/image-1.png and /dev/null differ diff --git a/docs/images/image-2.png b/docs/images/image-2.png deleted file mode 100644 index 0cd05d2b..00000000 Binary files a/docs/images/image-2.png and /dev/null differ diff --git a/docs/images/image-3.png b/docs/images/image-3.png deleted file mode 100644 index 10ce85b5..00000000 Binary files a/docs/images/image-3.png and /dev/null differ diff --git a/docs/images/image-4.png b/docs/images/image-4.png deleted file mode 100644 index bc33cb0b..00000000 Binary files a/docs/images/image-4.png and /dev/null differ diff --git a/docs/images/image-banner.png b/docs/images/image-banner.png deleted file mode 100644 index 53b0e3f2..00000000 Binary files a/docs/images/image-banner.png and /dev/null differ diff --git a/docs/images/image.png b/docs/images/image.png deleted file mode 100644 index 1a11aafc..00000000 Binary files a/docs/images/image.png and /dev/null differ diff --git a/docs/images/openrouter-logo.webp b/docs/images/openrouter-logo.webp deleted file mode 100644 index 449e871a..00000000 Binary files a/docs/images/openrouter-logo.webp and /dev/null differ diff --git a/docs/images/privacy-block.png b/docs/images/privacy-block.png deleted file mode 100644 index ac575d4a..00000000 Binary files a/docs/images/privacy-block.png and /dev/null differ diff --git a/docs/images/screenshot-agents.webp b/docs/images/screenshot-agents.webp deleted file mode 100644 index 68410ca0..00000000 Binary files a/docs/images/screenshot-agents.webp and /dev/null differ diff --git a/docs/images/screenshot-chat.webp b/docs/images/screenshot-chat.webp deleted file mode 100644 index 05989afc..00000000 Binary files a/docs/images/screenshot-chat.webp and /dev/null differ diff --git a/docs/images/zai-logo.svg b/docs/images/zai-logo.svg deleted file mode 100644 index 4f511bd7..00000000 --- a/docs/images/zai-logo.svg +++ /dev/null @@ -1,219 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - diff --git a/electron/activityTracking.cjs b/electron/activityTracking.cjs new file mode 100644 index 00000000..c8a4da5a --- /dev/null +++ b/electron/activityTracking.cjs @@ -0,0 +1,70 @@ +const { ipcMain } = require('electron'); +const log = require('electron-log'); +const { getAdaptiveHealthCheckManager } = require('./adaptiveHealthCheckManager.cjs'); + +/** + * Activity Tracking for Adaptive Health Checks + * Records user and service activity to inform adaptive battery management + */ + +function setupActivityTracking() { + const adaptiveManager = getAdaptiveHealthCheckManager(); + + // Track user activity when they interact with any service + const activityHandlers = [ + // N8N activity + { pattern: 'n8n:', service: 'n8n' }, + { pattern: 'start-n8n', service: 'n8n' }, + { pattern: 'stop-n8n', service: 'n8n' }, + + // ComfyUI activity + { pattern: 'comfyui:', service: 'comfyui' }, + { pattern: 'start-comfyui', service: 'comfyui' }, + { pattern: 'stop-comfyui', service: 'comfyui' }, + + // Python backend activity + { pattern: 'start-python', service: 'python' }, + { pattern: 'stop-python', service: 'python' }, + { pattern: 'chat:', service: 'python' }, + { pattern: 'rag:', service: 'python' }, + { pattern: 'tts:', service: 'python' }, + { pattern: 'stt:', service: 'python' }, + + // Clara Core activity + { pattern: 'claracore', service: 'claracore' }, + { pattern: 'model:', service: 'claracore' }, + { pattern: 'ollama:', service: 'claracore' } + ]; + + // Wrap IPC handler to track activity + const originalHandle = ipcMain.handle.bind(ipcMain); + ipcMain.handle = function(channel, handler) { + // Check if this channel should be tracked + const activityHandler = activityHandlers.find(h => channel.includes(h.pattern)); + + if (activityHandler) { + // Wrap handler to record activity + const wrappedHandler = async function(...args) { + // Record user activity + adaptiveManager.recordUserActivity(); + // Record service-specific activity + adaptiveManager.recordServiceActivity(activityHandler.service); + + log.debug(`📊 Activity recorded: ${activityHandler.service} via ${channel}`); + + // Call original handler + return handler(...args); + }; + + return originalHandle(channel, wrappedHandler); + } else { + // No tracking needed, use original handler + return originalHandle(channel, handler); + } + }; + + log.info('✅ Activity tracking enabled for adaptive health checks'); +} + +module.exports = { setupActivityTracking }; + diff --git a/electron/adaptiveHealthCheckManager.cjs b/electron/adaptiveHealthCheckManager.cjs new file mode 100644 index 00000000..afa306d8 --- /dev/null +++ b/electron/adaptiveHealthCheckManager.cjs @@ -0,0 +1,305 @@ +const { powerMonitor } = require('electron'); +const log = require('electron-log'); + +/** + * Adaptive Health Check Manager + * Intelligently adjusts health check intervals based on system activity and service usage + * Dramatically reduces battery drain during idle periods + */ +class AdaptiveHealthCheckManager { + constructor() { + // Health check intervals (in milliseconds) + this.intervals = { + ACTIVE: 30000, // 30 seconds - when actively using services + LIGHT_IDLE: 2 * 60000, // 2 minutes - light idle (< 5 min) + MEDIUM_IDLE: 5 * 60000, // 5 minutes - medium idle (5-15 min) + DEEP_IDLE: 10 * 60000, // 10 minutes - deep idle (15-30 min) + SLEEP: 30 * 60000 // 30 minutes - extended idle (> 30 min) + }; + + // Activity tracking + this.lastUserActivity = Date.now(); + this.lastServiceActivity = new Map(); // Track per-service activity + this.systemIdleTime = 0; + this.currentMode = 'LIGHT_IDLE'; // Start in LIGHT_IDLE instead of ACTIVE to save battery + + // Service activity thresholds (minutes) + this.thresholds = { + LIGHT_IDLE: 5, + MEDIUM_IDLE: 15, + DEEP_IDLE: 30 + }; + + // Monitoring state + this.isMonitoring = false; + this.activityCheckInterval = null; + + // Battery state awareness - check immediately on init + this.isOnBattery = false; + this.batteryLevel = 100; + + // Initialize battery state synchronously if possible + if (powerMonitor && powerMonitor.isOnBatteryPower) { + try { + this.isOnBattery = powerMonitor.isOnBatteryPower(); + if (this.isOnBattery) { + this.currentMode = 'MEDIUM_IDLE'; // More aggressive power saving on battery + log.info('🔋 Adaptive Health Check Manager initialized (on battery - starting in MEDIUM_IDLE mode)'); + } else { + log.info('🔋 Adaptive Health Check Manager initialized (on AC power - starting in LIGHT_IDLE mode)'); + } + } catch (error) { + log.info('🔋 Adaptive Health Check Manager initialized (battery state unknown - starting in LIGHT_IDLE mode)'); + } + } else { + log.info('🔋 Adaptive Health Check Manager initialized (starting in LIGHT_IDLE mode)'); + } + } + + /** + * Start monitoring system activity and battery state + */ + startMonitoring() { + if (this.isMonitoring) return; + + this.isMonitoring = true; + + // Monitor system idle state + this.activityCheckInterval = setInterval(() => { + this.checkSystemIdle(); + }, 30000); // Check every 30 seconds + + // Monitor battery state if available + if (powerMonitor) { + try { + // Check if on battery power + this.isOnBattery = powerMonitor.isOnBatteryPower?.() || false; + + // Listen for power state changes + powerMonitor.on?.('on-battery', () => { + this.isOnBattery = true; + log.info('🔋 Switched to battery power - enabling aggressive power saving'); + this.updateHealthCheckMode(); + }); + + powerMonitor.on?.('on-ac', () => { + this.isOnBattery = false; + log.info('🔌 Plugged in - using standard health check intervals'); + this.updateHealthCheckMode(); + }); + + // Monitor system resume from sleep + powerMonitor.on?.('resume', () => { + log.info('💤 System resumed from sleep - resetting activity timers'); + this.recordUserActivity(); + }); + + // Monitor system suspend + powerMonitor.on?.('suspend', () => { + log.info('💤 System suspending - pausing health checks'); + }); + + // Monitor for user activity (keyboard/mouse) + powerMonitor.on?.('user-did-become-active', () => { + this.recordUserActivity(); + }); + + log.info('✅ Battery and power monitoring enabled'); + } catch (error) { + log.warn('⚠️ Power monitoring not available:', error.message); + } + } + + log.info('✅ Adaptive health check monitoring started'); + } + + /** + * Stop monitoring + */ + stopMonitoring() { + if (this.activityCheckInterval) { + clearInterval(this.activityCheckInterval); + this.activityCheckInterval = null; + } + this.isMonitoring = false; + log.info('🛑 Adaptive health check monitoring stopped'); + } + + /** + * Check system idle time + */ + async checkSystemIdle() { + try { + if (powerMonitor && powerMonitor.getSystemIdleTime) { + this.systemIdleTime = powerMonitor.getSystemIdleTime(); + + // If user becomes active, record it + if (this.systemIdleTime < 10) { // Less than 10 seconds idle + this.recordUserActivity(); + } + } + + this.updateHealthCheckMode(); + } catch (error) { + log.debug('Could not get system idle time:', error.message); + } + } + + /** + * Record user activity + */ + recordUserActivity() { + const now = Date.now(); + const wasIdle = this.currentMode !== 'ACTIVE'; + + this.lastUserActivity = now; + this.updateHealthCheckMode(); + + if (wasIdle) { + log.info('👤 User activity detected - returning to active monitoring mode'); + } + } + + /** + * Record service activity (service was used) + */ + recordServiceActivity(serviceName) { + this.lastServiceActivity.set(serviceName, Date.now()); + + // Any service activity counts as user activity + this.recordUserActivity(); + + log.debug(`📊 Service activity recorded: ${serviceName}`); + } + + /** + * Get time since last activity for a service + */ + getServiceIdleTime(serviceName) { + const lastActivity = this.lastServiceActivity.get(serviceName); + if (!lastActivity) return Infinity; + + return Date.now() - lastActivity; + } + + /** + * Update health check mode based on activity + */ + updateHealthCheckMode() { + const minutesSinceActivity = (Date.now() - this.lastUserActivity) / 60000; + let newMode = 'ACTIVE'; + + // Determine mode based on idle time + if (minutesSinceActivity > this.thresholds.DEEP_IDLE) { + newMode = 'SLEEP'; + } else if (minutesSinceActivity > this.thresholds.MEDIUM_IDLE) { + newMode = 'DEEP_IDLE'; + } else if (minutesSinceActivity > this.thresholds.LIGHT_IDLE) { + newMode = 'MEDIUM_IDLE'; + } else if (minutesSinceActivity > 1) { + newMode = 'LIGHT_IDLE'; + } + + // Be more aggressive on battery power + if (this.isOnBattery) { + // Accelerate idle mode progression on battery + if (minutesSinceActivity > 10 && newMode === 'LIGHT_IDLE') { + newMode = 'MEDIUM_IDLE'; + } else if (minutesSinceActivity > 20 && newMode === 'MEDIUM_IDLE') { + newMode = 'DEEP_IDLE'; + } + } + + // Log mode changes + if (newMode !== this.currentMode) { + const interval = this.intervals[newMode]; + log.info(`🔄 Health check mode: ${this.currentMode} → ${newMode} (interval: ${interval / 1000}s)`); + this.currentMode = newMode; + } + } + + /** + * Get current health check interval for a service + */ + getHealthCheckInterval(serviceName, baseInterval = 30000) { + // If service was used recently, use more frequent checks + const serviceIdleTime = this.getServiceIdleTime(serviceName); + const serviceIdleMinutes = serviceIdleTime / 60000; + + // If service is actively being used (< 2 minutes), use base interval + if (serviceIdleMinutes < 2) { + return baseInterval; + } + + // Otherwise, use adaptive interval based on system-wide activity + const adaptiveInterval = this.intervals[this.currentMode]; + + // On battery, use longer intervals + if (this.isOnBattery) { + return Math.max(adaptiveInterval, baseInterval * 2); + } + + return Math.max(adaptiveInterval, baseInterval); + } + + /** + * Should we skip health check? (during very deep idle) + */ + shouldSkipHealthCheck(serviceName) { + // Never skip if actively using the service + const serviceIdleMinutes = this.getServiceIdleTime(serviceName) / 60000; + if (serviceIdleMinutes < 2) { + return false; + } + + // In SLEEP mode on battery, only check critical services + if (this.currentMode === 'SLEEP' && this.isOnBattery) { + // Define critical services that always need checking + const criticalServices = ['claracore']; + return !criticalServices.includes(serviceName); + } + + return false; + } + + /** + * Get battery-aware configuration + */ + getBatteryAwareConfig() { + return { + isOnBattery: this.isOnBattery, + currentMode: this.currentMode, + minutesIdle: (Date.now() - this.lastUserActivity) / 60000, + recommendedInterval: this.intervals[this.currentMode], + aggressiveSaving: this.isOnBattery && this.currentMode !== 'ACTIVE' + }; + } + + /** + * Get status for logging + */ + getStatus() { + return { + mode: this.currentMode, + isOnBattery: this.isOnBattery, + batteryLevel: this.batteryLevel, + minutesSinceActivity: Math.round((Date.now() - this.lastUserActivity) / 60000), + systemIdleTime: this.systemIdleTime, + currentInterval: this.intervals[this.currentMode], + activeServices: this.lastServiceActivity.size + }; + } +} + +// Singleton instance +let instance = null; + +module.exports = { + getAdaptiveHealthCheckManager: () => { + if (!instance) { + instance = new AdaptiveHealthCheckManager(); + } + return instance; + } +}; + diff --git a/electron/assets/tray-icon.png b/electron/assets/tray-icon.png new file mode 100644 index 00000000..140e6dc4 Binary files /dev/null and b/electron/assets/tray-icon.png differ diff --git a/electron/centralServiceManager.cjs b/electron/centralServiceManager.cjs new file mode 100644 index 00000000..0b2e5119 --- /dev/null +++ b/electron/centralServiceManager.cjs @@ -0,0 +1,792 @@ +const { EventEmitter } = require('events'); +const log = require('electron-log'); +const { getAdaptiveHealthCheckManager } = require('./adaptiveHealthCheckManager.cjs'); + +/** + * Central Service Manager + * Single source of truth for all ClaraVerse services + * Supports both Docker and manual deployment modes + * Replaces scattered service management across multiple files + */ +class CentralServiceManager extends EventEmitter { + constructor(configManager = null) { + super(); + + // Get adaptive health check manager + this.adaptiveHealthManager = getAdaptiveHealthCheckManager(); + + // Service registry - single source of truth + this.services = new Map(); + this.serviceStates = new Map(); + this.platformInfo = this.detectPlatform(); + + // NEW: Configuration manager for deployment modes + this.configManager = configManager; + + // Service lifecycle states + this.states = { + STOPPED: 'stopped', + STARTING: 'starting', + RUNNING: 'running', + STOPPING: 'stopping', + ERROR: 'error', + RESTARTING: 'restarting' + }; + + // Global configuration (now with adaptive health checks) + this.config = { + startupTimeout: 30000, + shutdownTimeout: 15000, + baseHealthCheckInterval: 30000, // Base interval when active + maxRestartAttempts: 3, + restartDelay: 5000, + useAdaptiveChecks: true // Enable adaptive health checking + }; + + this.isShuttingDown = false; + this.startupPromise = null; + + log.info('🎯 Central Service Manager initialized with deployment mode support'); + } + + /** + * Register a service with the central manager + */ + registerService(name, serviceConfig) { + const service = { + name, + ...serviceConfig, + state: this.states.STOPPED, + restartAttempts: 0, + lastHealthCheck: null, + lastError: null, + instance: null + }; + + this.services.set(name, service); + this.serviceStates.set(name, this.states.STOPPED); + this.emit('service-registered', { name, service }); + + log.info(`📝 Registered service: ${name}`); + return service; + } + + /** + * Start all services in proper order + */ + async startAllServices() { + if (this.startupPromise) { + return this.startupPromise; + } + + this.startupPromise = this._startServicesSequence(); + return this.startupPromise; + } + + async _startServicesSequence() { + try { + log.info('🚀 Starting all ClaraVerse services...'); + + // Get services ordered by priority + const orderedServices = this.getStartupOrder(); + + for (const serviceName of orderedServices) { + const service = this.services.get(serviceName); + if (!service) continue; + + try { + await this.startService(serviceName); + } catch (error) { + log.error(`❌ Failed to start ${serviceName}:`, error); + + // Decide if this is a critical failure + if (service.critical) { + throw new Error(`Critical service ${serviceName} failed to start: ${error.message}`); + } + + // Mark as error but continue with non-critical services + this.setState(serviceName, this.states.ERROR); + service.lastError = error; + } + } + + // Start adaptive health monitoring + if (this.config.useAdaptiveChecks) { + this.adaptiveHealthManager.startMonitoring(); + log.info('🔋 Adaptive health monitoring enabled', this.adaptiveHealthManager.getStatus()); + } + + // Start health monitoring + this.startHealthMonitoring(); + + log.info('✅ Service startup sequence completed'); + this.emit('all-services-started'); + + } catch (error) { + log.error('💥 Service startup failed:', error); + this.emit('startup-failed', error); + throw error; + } finally { + this.startupPromise = null; + } + } + + /** + * Start individual service (Enhanced for deployment modes) + */ + async startService(serviceName) { + const service = this.services.get(serviceName); + if (!service) { + throw new Error(`Service ${serviceName} not registered`); + } + + if (service.state === this.states.RUNNING) { + log.info(`⏭️ Service ${serviceName} already running`); + return; + } + + this.setState(serviceName, this.states.STARTING); + + try { + // NEW: Determine deployment mode + const deploymentMode = this.getDeploymentMode(serviceName); + log.info(`🔄 Starting service: ${serviceName} (mode: ${deploymentMode})`); + + // Start service based on deployment mode + if (deploymentMode === 'manual' || deploymentMode === 'remote') { + service.instance = await this.startManualService(serviceName, service); + } else if (deploymentMode === 'local' || deploymentMode === 'native') { + // Local binary service + service.instance = await this.startNativeService(serviceName, service); + } else { + // Default to Docker mode (backward compatibility) + const startupMethod = this.getStartupMethod(service); + service.instance = await startupMethod(service); + } + + // Wait for service to become healthy + await this.waitForHealthy(serviceName); + + this.setState(serviceName, this.states.RUNNING); + service.restartAttempts = 0; + service.deploymentMode = deploymentMode; + + log.info(`✅ Service ${serviceName} started successfully (${deploymentMode} mode)`); + this.emit('service-started', { + name: serviceName, + service, + deploymentMode: deploymentMode + }); + + } catch (error) { + this.setState(serviceName, this.states.ERROR); + service.lastError = error; + log.error(`❌ Failed to start service ${serviceName}:`, error); + throw error; + } + } + + /** + * Stop all services gracefully + */ + async stopAllServices() { + this.isShuttingDown = true; + + // Stop adaptive monitoring + if (this.adaptiveHealthManager) { + this.adaptiveHealthManager.stopMonitoring(); + } + + // Clear health check timer + if (this.healthCheckTimer) { + clearTimeout(this.healthCheckTimer); + this.healthCheckTimer = null; + } + + try { + log.info('🛑 Stopping all ClaraVerse services...'); + + // Get services in reverse startup order + const orderedServices = this.getStartupOrder().reverse(); + + const stopPromises = orderedServices.map(serviceName => + this.stopService(serviceName).catch(error => { + log.error(`Error stopping ${serviceName}:`, error); + }) + ); + + await Promise.allSettled(stopPromises); + + log.info('✅ All services stopped'); + this.emit('all-services-stopped'); + + } catch (error) { + log.error('Error during service shutdown:', error); + throw error; + } finally { + this.isShuttingDown = false; + } + } + + /** + * Stop individual service + */ + async stopService(serviceName) { + const service = this.services.get(serviceName); + if (!service || service.state === this.states.STOPPED) { + return; + } + + this.setState(serviceName, this.states.STOPPING); + + try { + log.info(`🔄 Stopping service: ${serviceName}`); + + const stopMethod = this.getStopMethod(service); + await stopMethod(service); + + this.setState(serviceName, this.states.STOPPED); + service.instance = null; + + log.info(`✅ Service ${serviceName} stopped`); + this.emit('service-stopped', { name: serviceName, service }); + + } catch (error) { + log.error(`❌ Error stopping service ${serviceName}:`, error); + service.lastError = error; + this.setState(serviceName, this.states.ERROR); + throw error; + } + } + + /** + * Restart service with exponential backoff + */ + async restartService(serviceName) { + const service = this.services.get(serviceName); + if (!service) { + throw new Error(`Service ${serviceName} not registered`); + } + + if (service.restartAttempts >= this.config.maxRestartAttempts) { + throw new Error(`Service ${serviceName} exceeded max restart attempts`); + } + + this.setState(serviceName, this.states.RESTARTING); + service.restartAttempts++; + + try { + // Stop first + await this.stopService(serviceName); + + // Wait with exponential backoff + const delay = this.config.restartDelay * Math.pow(2, service.restartAttempts - 1); + await new Promise(resolve => setTimeout(resolve, delay)); + + // Start again + await this.startService(serviceName); + + log.info(`🔄 Service ${serviceName} restarted successfully`); + + } catch (error) { + log.error(`❌ Failed to restart service ${serviceName}:`, error); + this.setState(serviceName, this.states.ERROR); + throw error; + } + } + + /** + * Get service startup order based on dependencies + */ + getStartupOrder() { + // Define service dependencies + const dependencies = { + 'docker': [], + 'claracore': [], // No dependencies - runs independently + 'python-backend': ['docker'], + 'comfyui': ['docker', 'python-backend'], + 'n8n': ['docker'], + 'mcp': ['python-backend', 'claracore'], + 'watchdog': ['docker', 'python-backend', 'claracore'] + }; + + // Topological sort to determine startup order + return this.topologicalSort(dependencies); + } + + /** + * Topological sort for dependency resolution + */ + topologicalSort(dependencies) { + const visited = new Set(); + const temp = new Set(); + const result = []; + + const visit = (node) => { + if (temp.has(node)) { + throw new Error(`Circular dependency detected involving ${node}`); + } + if (!visited.has(node)) { + temp.add(node); + + const deps = dependencies[node] || []; + deps.forEach(visit); + + temp.delete(node); + visited.add(node); + result.push(node); + } + }; + + Object.keys(dependencies).forEach(visit); + return result; + } + + /** + * Platform detection + */ + detectPlatform() { + const os = require('os'); + return { + platform: os.platform(), + arch: os.arch(), + isWindows: os.platform() === 'win32', + isMac: os.platform() === 'darwin', + isLinux: os.platform() === 'linux' + }; + } + + /** + * Set service state and emit events + */ + setState(serviceName, state) { + const previousState = this.serviceStates.get(serviceName); + this.serviceStates.set(serviceName, state); + + const service = this.services.get(serviceName); + if (service) { + service.state = state; + } + + this.emit('service-state-changed', { + name: serviceName, + previousState, + currentState: state + }); + } + + /** + * Set service URL (for remote/manual deployments) + */ + setServiceUrl(serviceName, url) { + const service = this.services.get(serviceName); + if (service) { + service.serviceUrl = url; + log.info(`✅ Set ${serviceName} URL to: ${url}`); + } else { + log.warn(`⚠️ Service ${serviceName} not found when setting URL`); + } + } + + /** + * Get current state of all services (Enhanced with deployment mode info) + */ + getServicesStatus() { + const status = {}; + + for (const [name, service] of this.services) { + // Calculate uptime + let uptime = 0; + if (service.instance && service.instance.startTime) { + uptime = Date.now() - service.instance.startTime; + } + + // Get deployment mode and URL info + const deploymentMode = service.deploymentMode || this.getDeploymentMode(name); + let serviceUrl = null; + + if ((deploymentMode === 'manual' || deploymentMode === 'remote') && this.configManager) { + // Manual and remote services get URL from config manager + serviceUrl = this.configManager.getServiceUrl(name); + } else if (deploymentMode === 'docker' && service.serviceUrl) { + // Docker services get URL from service state (set by main.cjs) + serviceUrl = service.serviceUrl; + } else if (deploymentMode === 'docker' && service.instance && service.instance.url) { + // Docker services get URL from container instance (fallback) + serviceUrl = service.instance.url; + } else if (deploymentMode === 'docker') { + // Fallback: construct default URLs for known Docker services + const defaultPorts = { + 'n8n': 5678, + 'python-backend': 5001, + 'comfyui': 8188, + 'claracore': 8091 + }; + if (defaultPorts[name]) { + serviceUrl = `http://localhost:${defaultPorts[name]}`; + } + } + + status[name] = { + state: service.state, + deploymentMode: deploymentMode, + restartAttempts: service.restartAttempts, + lastHealthCheck: service.lastHealthCheck, + lastError: service.lastError?.message, + uptime: uptime, + // NEW: Deployment mode specific information + serviceUrl: serviceUrl, + isManual: deploymentMode === 'manual' || deploymentMode === 'remote', + canRestart: (deploymentMode !== 'manual' && deploymentMode !== 'remote') && service.autoRestart, + // Platform compatibility info + supportedModes: this.configManager?.getSupportedModes(name) || ['docker'] + }; + } + + return status; + } + + /** + * Schedule next adaptive health check + */ + scheduleNextHealthCheck() { + if (this.isShuttingDown) return; + + // Get adaptive interval + let nextInterval = this.config.baseHealthCheckInterval; + + if (this.config.useAdaptiveChecks && this.adaptiveHealthManager) { + // Use the most conservative interval (longest) from all services + for (const [serviceName] of this.services) { + const serviceInterval = this.adaptiveHealthManager.getHealthCheckInterval( + serviceName, + this.config.baseHealthCheckInterval + ); + nextInterval = Math.max(nextInterval, serviceInterval); + } + + // Log adaptive interval changes + const status = this.adaptiveHealthManager.getStatus(); + if (status.mode !== 'ACTIVE') { + log.debug(`🔋 Adaptive health check: next in ${nextInterval / 1000}s (${status.mode} mode)`); + } + } + + // Clear existing timer + if (this.healthCheckTimer) { + clearTimeout(this.healthCheckTimer); + } + + // Schedule next check + this.healthCheckTimer = setTimeout(async () => { + await this.performHealthMonitoring(); + this.scheduleNextHealthCheck(); // Schedule next one after this completes + }, nextInterval); + } + + /** + * Health monitoring for all services (Enhanced for deployment modes and adaptive checking) + */ + startHealthMonitoring() { + // Start the first health check cycle + this.scheduleNextHealthCheck(); + } + + /** + * Perform health monitoring on all services + */ + async performHealthMonitoring() { + if (this.isShuttingDown) return; + + for (const [serviceName, service] of this.services) { + if (service.state === this.states.RUNNING) { + // Skip services during deep idle if adaptive checking is enabled + if (this.config.useAdaptiveChecks && + this.adaptiveHealthManager && + this.adaptiveHealthManager.shouldSkipHealthCheck(serviceName)) { + log.debug(`⏭️ Skipping ${serviceName} health check - deep idle mode`); + continue; + } + + try { + // Determine health check method based on deployment mode + let healthCheck; + let isHealthy = false; + + if (service.instance && service.instance.healthCheck) { + // Manual service with custom health check + healthCheck = service.instance.healthCheck; + isHealthy = await healthCheck(); + } else if (service.healthCheck) { + // Service-defined health check + const serviceUrl = this.configManager?.getServiceUrl(serviceName); + isHealthy = await service.healthCheck(serviceUrl); + } else { + // No health check defined, assume healthy + isHealthy = true; + } + + service.lastHealthCheck = Date.now(); + + // Record service activity for adaptive checking when healthy + if (isHealthy && this.adaptiveHealthManager) { + this.adaptiveHealthManager.recordServiceActivity(serviceName); + } + + if (!isHealthy) { + log.warn(`⚠️ Service ${serviceName} health check failed (${service.deploymentMode || 'docker'} mode)`); + this.emit('service-unhealthy', { + name: serviceName, + service, + deploymentMode: service.deploymentMode || 'docker' + }); + + // Auto-restart if configured (only for Docker services) + if (service.autoRestart && service.deploymentMode !== 'manual') { + log.info(`🔄 Auto-restarting service ${serviceName}`); + await this.restartService(serviceName); + } else if (service.deploymentMode === 'manual') { + log.warn(`⚠️ Manual service ${serviceName} is unhealthy, manual intervention required`); + this.setState(serviceName, this.states.ERROR); + service.lastError = new Error('Manual service health check failed'); + } + } + + } catch (error) { + log.error(`❌ Health check error for ${serviceName}:`, error); + service.lastError = error; + + // For manual services, mark as error since we can't restart them + if (service.deploymentMode === 'manual') { + this.setState(serviceName, this.states.ERROR); + } + } + } + } + } + + /** + * Wait for service to become healthy (Enhanced for deployment modes) + */ + async waitForHealthy(serviceName, timeout = this.config.startupTimeout) { + const service = this.services.get(serviceName); + + // Determine health check method based on deployment mode + let healthCheck; + + if (service.instance && service.instance.healthCheck) { + // Manual service with custom health check + healthCheck = service.instance.healthCheck; + } else if (service.healthCheck) { + // Service-defined health check + const serviceUrl = this.configManager?.getServiceUrl(serviceName); + healthCheck = () => service.healthCheck(serviceUrl); + } else { + // No health check defined + log.warn(`⚠️ No health check defined for service ${serviceName}, assuming healthy`); + return; + } + + const startTime = Date.now(); + let lastError = null; + + while (Date.now() - startTime < timeout) { + try { + const isHealthy = await healthCheck(); + if (isHealthy) { + log.info(`💚 Service ${serviceName} is healthy`); + return; + } + } catch (error) { + lastError = error; + log.debug(`Health check failed for ${serviceName}:`, error.message); + } + + await new Promise(resolve => setTimeout(resolve, 1000)); + } + + const errorMessage = lastError + ? `Service ${serviceName} failed to become healthy within ${timeout}ms. Last error: ${lastError.message}` + : `Service ${serviceName} failed to become healthy within ${timeout}ms`; + + throw new Error(errorMessage); + } + + /** + * NEW: Get deployment mode for a service + */ + getDeploymentMode(serviceName) { + // Special case: ClaraCore always defaults to 'local' mode (native binary) + // Check this FIRST before config manager to ensure it always runs locally + if (serviceName === 'claracore') { + // Unless explicitly configured otherwise, always use local mode + if (this.configManager) { + const mode = this.configManager.getServiceMode(serviceName); + // Only allow 'local' or 'remote', never docker for claracore + if (mode === 'local' || mode === 'remote') { + return mode; + } + } + return 'local'; + } + + if (this.configManager) { + const mode = this.configManager.getServiceMode(serviceName); + if (mode) return mode; + } + + // Fallback: check if service supports manual mode + const { getSupportedDeploymentModes } = require('./serviceDefinitions.cjs'); + const supportedModes = getSupportedDeploymentModes(serviceName, this.platformInfo.platform); + + return supportedModes.includes('docker') ? 'docker' : supportedModes[0] || 'docker'; + } + + /** + * NEW: Start manual service (BYOS - Bring Your Own Service) + */ + async startManualService(serviceName, service) { + if (!this.configManager) { + throw new Error(`Manual service mode requires configuration manager`); + } + + const serviceUrl = this.configManager.getServiceUrl(serviceName); + if (!serviceUrl) { + throw new Error(`Manual service ${serviceName} requires URL configuration`); + } + + log.info(`🔗 Connecting to manual service ${serviceName} at ${serviceUrl}`); + + // Create manual service health check + const { createManualHealthCheck } = require('./serviceDefinitions.cjs'); + const healthEndpoint = service.manual?.healthEndpoint || '/'; + const healthCheck = createManualHealthCheck(serviceUrl, healthEndpoint); + + // Test connectivity immediately + const isHealthy = await healthCheck(); + if (!isHealthy) { + throw new Error(`Manual service ${serviceName} at ${serviceUrl} is not accessible or unhealthy`); + } + + // Return manual service instance + return { + type: 'manual', + url: serviceUrl, + healthEndpoint: healthEndpoint, + healthCheck: healthCheck, + startTime: Date.now(), + deploymentMode: 'manual' + }; + } + + /** + * NEW: Start native service (binary executable) + */ + async startNativeService(serviceName, service) { + log.info(`🔧 Starting native service ${serviceName}`); + + // Use the service's custom start method if available + if (service.customStart) { + const instance = await service.customStart(); + + // IMPORTANT: Don't spread the instance! Methods are on the prototype and won't be copied. + // Store the instance directly and add metadata properties to it + instance.type = 'native'; + instance.startTime = Date.now(); + instance.deploymentMode = 'local'; + + // Ensure healthCheck is bound properly + if (instance.checkHealth && !instance.healthCheck) { + instance.healthCheck = instance.checkHealth.bind(instance); + } + + // Return the instance with all its methods intact + return instance; + } + + // Fallback: generic native service startup + throw new Error(`Native service ${serviceName} does not define a customStart method`); + } + + /** + * Get platform-specific startup method + */ + getStartupMethod(service) { + // Return appropriate startup method based on service type and platform + if (service.dockerContainer) { + return this.startDockerService.bind(this); + } else if (service.binaryPath) { + return this.startBinaryService.bind(this); + } else if (service.customStart) { + return service.customStart; + } + + throw new Error(`No startup method defined for service ${service.name}`); + } + + /** + * Get platform-specific stop method + */ + getStopMethod(service) { + if (service.dockerContainer) { + return this.stopDockerService.bind(this); + } else if (service.instance && service.instance.stopService) { + // Native service instance (like ClaraCore local binary) + return this.stopNativeService.bind(this); + } else if (service.process) { + return this.stopProcessService.bind(this); + } else if (service.customStop) { + return service.customStop; + } + + return () => Promise.resolve(); // No-op if no stop method + } + + /** + * Docker service startup + */ + async startDockerService(service) { + // Implementation will integrate with existing dockerSetup.cjs logic + const DockerSetup = require('./dockerSetup.cjs'); + // ... Docker startup logic + } + + /** + * Docker service stop + */ + async stopDockerService(service) { + // Docker stop logic + } + + /** + * Binary service startup + */ + async startBinaryService(service) { + const { spawn } = require('child_process'); + // Binary startup logic + } + + /** + * Process service stop + */ + async stopProcessService(service) { + if (service.instance && service.instance.kill) { + service.instance.kill('SIGTERM'); + } + } + + /** + * Native service stop (for ClaraCore local binary and similar) + */ + async stopNativeService(service) { + if (service.instance && typeof service.instance.stopService === 'function') { + log.info(`🛑 Stopping native service instance: ${service.name}`); + await service.instance.stopService(); + log.info(`✅ Native service instance stopped: ${service.name}`); + } else { + log.warn(`⚠️ Native service ${service.name} has no stopService method`); + } + } +} + +module.exports = CentralServiceManager; \ No newline at end of file diff --git a/electron/claraCoreDockerService.cjs b/electron/claraCoreDockerService.cjs new file mode 100644 index 00000000..7c5ce2d9 --- /dev/null +++ b/electron/claraCoreDockerService.cjs @@ -0,0 +1,727 @@ +const Docker = require('dockerode'); +const { execSync } = require('child_process'); +const log = require('electron-log'); +const os = require('os'); +const path = require('path'); +const fs = require('fs'); + +/** + * ClaraCore Docker Service Manager + * Manages ClaraCore in Docker with GPU detection and acceleration support + */ +class ClaraCoreDockerService { + constructor() { + this.docker = new Docker(); + this.containerName = 'clara_core'; + this.isRunning = false; + this.gpuType = null; + this.detectedImage = null; + } + + /** + * Detect GPU type and determine appropriate Docker image + */ + async detectGPU() { + const platform = os.platform(); + + log.info('🔍 Detecting GPU for ClaraCore Docker...'); + + try { + // Check for NVIDIA GPU + if (platform === 'win32') { + try { + const nvidiaCheck = execSync('nvidia-smi --query-gpu=name --format=csv,noheader', { + encoding: 'utf8', + timeout: 5000 + }); + + if (nvidiaCheck && nvidiaCheck.trim()) { + log.info(`✅ NVIDIA GPU detected: ${nvidiaCheck.trim()}`); + this.gpuType = 'cuda'; + this.detectedImage = 'clara17verse/claracore:cuda'; + return { type: 'cuda', name: nvidiaCheck.trim() }; + } + } catch (error) { + log.info('NVIDIA GPU not detected or nvidia-smi not available'); + } + } else if (platform === 'linux') { + // Linux: Check for NVIDIA + try { + const nvidiaCheck = execSync('nvidia-smi --query-gpu=name --format=csv,noheader', { + encoding: 'utf8', + timeout: 5000 + }); + + if (nvidiaCheck && nvidiaCheck.trim()) { + log.info(`✅ NVIDIA GPU detected: ${nvidiaCheck.trim()}`); + this.gpuType = 'cuda'; + this.detectedImage = 'clara17verse/claracore:cuda'; + return { type: 'cuda', name: nvidiaCheck.trim() }; + } + } catch (error) { + log.info('NVIDIA GPU not detected'); + } + + // Linux: Check for AMD ROCm + try { + const rocmCheck = execSync('rocm-smi --showproductname', { + encoding: 'utf8', + timeout: 5000 + }); + + if (rocmCheck && rocmCheck.includes('GPU')) { + log.info(`✅ AMD ROCm GPU detected`); + this.gpuType = 'rocm'; + this.detectedImage = 'clara17verse/claracore:rocm'; + return { type: 'rocm', name: 'AMD GPU (ROCm)' }; + } + } catch (error) { + log.info('AMD ROCm GPU not detected'); + } + } else if (platform === 'win32') { + // Windows: Check for AMD GPU via WSL2 + // ROCm on Windows requires Docker Desktop in WSL2 mode + AMD Radeon Software 25.8.1+ + try { + const wslCheck = execSync('wsl -l -v', { + encoding: 'utf8', + timeout: 5000 + }); + + // Check if WSL2 is available + if (wslCheck && wslCheck.includes('Version 2')) { + // Try to detect AMD GPU through WSL + try { + const amdCheck = execSync('wsl lspci | findstr "AMD"', { + encoding: 'utf8', + timeout: 5000 + }); + + if (amdCheck && (amdCheck.includes('Radeon') || amdCheck.includes('AMD'))) { + log.info(`✅ AMD GPU detected in WSL2 - ROCm may be available`); + log.warn('⚠️ ROCm on Windows requires: Windows 11 + AMD Radeon Software 25.8.1+ + WSL2'); + log.warn('⚠️ Docker Desktop must be in WSL2 mode (not Hyper-V)'); + // Note: We don't auto-select ROCm on Windows due to complexity + // User should manually select it if they have it properly configured + } + } catch (amdError) { + log.info('AMD GPU not detected in WSL2'); + } + } + } catch (error) { + log.info('WSL2 not available or not configured'); + } + } + + // Check for Vulkan support (fallback for AMD/Intel on Linux) + if (platform === 'linux') { + try { + const vulkanCheck = execSync('vulkaninfo --summary', { + encoding: 'utf8', + timeout: 5000 + }); + + if (vulkanCheck && vulkanCheck.includes('Vulkan')) { + log.info(`✅ Vulkan GPU support detected`); + this.gpuType = 'vulkan'; + this.detectedImage = 'clara17verse/claracore:vulkan'; + return { type: 'vulkan', name: 'Vulkan-compatible GPU' }; + } + } catch (error) { + log.info('Vulkan support not detected'); + } + } + + // Fallback to CPU + log.info('⚠️ No GPU detected, using CPU mode'); + this.gpuType = 'cpu'; + this.detectedImage = 'clara17verse/claracore:cpu'; + return { type: 'cpu', name: 'CPU Only' }; + + } catch (error) { + log.error('Error during GPU detection:', error); + this.gpuType = 'cpu'; + this.detectedImage = 'clara17verse/claracore:cpu'; + return { type: 'cpu', name: 'CPU Only (detection failed)' }; + } + } + + /** + * Ensure Docker is running + */ + async ensureDockerRunning() { + try { + await this.docker.ping(); + log.info('✅ Docker daemon is running'); + return true; + } catch (error) { + log.error('❌ Docker daemon is not running:', error.message); + + // On Windows, provide specific instructions + if (os.platform() === 'win32') { + throw new Error( + 'Docker Desktop is not running or not responding.\n\n' + + 'Please try:\n' + + '1. Open Docker Desktop from Start Menu\n' + + '2. Wait for Docker to fully start (whale icon in system tray should be steady)\n' + + '3. If Docker is already running, restart it: Right-click whale icon → Restart\n' + + '4. If that fails, completely quit Docker Desktop and start it again\n\n' + + 'Error: ' + error.message + ); + } + + throw new Error('Docker is not running. Please start Docker Desktop and try again.'); + } + } + + /** + * Check if container exists + */ + async containerExists() { + try { + const containers = await this.docker.listContainers({ all: true }); + return containers.some(c => c.Names.includes(`/${this.containerName}`)); + } catch (error) { + log.error('Error checking container existence:', error); + return false; + } + } + + /** + * Get container instance + */ + async getContainer() { + return this.docker.getContainer(this.containerName); + } + + /** + * Pull Docker image + */ + async pullImage(imageName) { + log.info(`🔽 Pulling Docker image: ${imageName}`); + + return new Promise((resolve, reject) => { + this.docker.pull(imageName, (err, stream) => { + if (err) { + log.error(`Failed to pull image ${imageName}:`, err); + return reject(err); + } + + this.docker.modem.followProgress(stream, + (err, output) => { + if (err) { + log.error('Error during image pull:', err); + return reject(err); + } + log.info(`✅ Successfully pulled image: ${imageName}`); + resolve(output); + }, + (event) => { + // Progress logging + if (event.status === 'Downloading' || event.status === 'Extracting') { + log.info(`${event.status}: ${event.progress || ''}`); + } + } + ); + }); + }); + } + + /** + * Check if port 8091 is in use and kill the process + */ + async killProcessOnPort8091() { + const platform = os.platform(); + + try { + log.info('🔍 Checking if port 8091 is in use...'); + + if (platform === 'win32') { + const { execSync } = require('child_process'); + try { + const netstatOutput = execSync('netstat -ano | findstr :8091 | findstr LISTENING', { + encoding: 'utf8', + timeout: 5000 + }); + + if (netstatOutput) { + const lines = netstatOutput.trim().split('\n'); + for (const line of lines) { + const parts = line.trim().split(/\s+/); + const pid = parts[parts.length - 1]; + + if (pid && !isNaN(pid)) { + log.warn(`⚠️ Port 8091 is in use by process ${pid}. Attempting to kill...`); + try { + execSync(`taskkill /F /PID ${pid}`, { encoding: 'utf8', timeout: 5000 }); + log.info(`✅ Killed process ${pid} on port 8091`); + // Wait for port to be released + await new Promise(resolve => setTimeout(resolve, 2000)); + } catch (killError) { + log.error(`❌ Failed to kill process ${pid}:`, killError.message); + } + } + } + } else { + log.info('✅ Port 8091 is free'); + } + } catch (error) { + // No process on port, which is good + log.info('✅ Port 8091 is free'); + } + } else { + // Linux/Mac: Use lsof + const { execSync } = require('child_process'); + try { + const lsofOutput = execSync('lsof -ti:8091', { encoding: 'utf8', timeout: 5000 }).trim(); + + if (lsofOutput) { + const pids = lsofOutput.split('\n').filter(pid => pid); + for (const pid of pids) { + log.warn(`⚠️ Port 8091 is in use by process ${pid}. Attempting to kill...`); + try { + execSync(`kill -9 ${pid}`, { encoding: 'utf8', timeout: 5000 }); + log.info(`✅ Killed process ${pid} on port 8091`); + // Wait for port to be released + await new Promise(resolve => setTimeout(resolve, 2000)); + } catch (killError) { + log.error(`❌ Failed to kill process ${pid}:`, killError.message); + } + } + } else { + log.info('✅ Port 8091 is free'); + } + } catch (error) { + // No process on port, which is good + log.info('✅ Port 8091 is free'); + } + } + } catch (error) { + log.warn('Error checking port 8091:', error.message); + } + } + + /** + * Cleanup stuck or corrupted containers (Windows-specific) + */ + async cleanupStuckContainers() { + try { + log.info('🔍 Checking for stuck containers...'); + + const containers = await this.docker.listContainers({ all: true }); + const claraCoreContainers = containers.filter(c => + c.Names.some(name => name.includes('clara_core')) + ); + + for (const containerInfo of claraCoreContainers) { + try { + const container = this.docker.getContainer(containerInfo.Id); + const info = await container.inspect(); + + // Check if container is in a problematic state + if (info.State.Status === 'dead' || + info.State.Status === 'exited' && info.State.ExitCode !== 0 || + info.State.Error) { + log.warn(`⚠️ Found stuck container ${containerInfo.Names[0]} in state: ${info.State.Status}`); + + // Try to remove it + try { + if (info.State.Running) { + await container.stop({ t: 5 }); + } + await container.remove({ force: true }); + log.info(`✅ Cleaned up stuck container ${containerInfo.Names[0]}`); + } catch (removeError) { + log.error(`Failed to remove stuck container: ${removeError.message}`); + } + } + } catch (inspectError) { + log.error(`Error inspecting container: ${inspectError.message}`); + } + } + } catch (error) { + log.warn('Error during cleanup check:', error.message); + // Don't throw - this is a best-effort cleanup + } + } + + /** + * Create and start Clara Core container + */ + async start(options = {}) { + try { + await this.ensureDockerRunning(); + + // Kill any process using port 8091 (including local ClaraCore binary) + await this.killProcessOnPort8091(); + + // On Windows, check if there are any stuck/corrupted containers and clean them + if (os.platform() === 'win32') { + await this.cleanupStuckContainers(); + } + + // Auto-detect GPU if not specified + let gpuInfo; + if (!options.gpuType) { + gpuInfo = await this.detectGPU(); + log.info(`Auto-detected GPU: ${gpuInfo.type} - ${gpuInfo.name}`); + } else { + this.gpuType = options.gpuType; + this.detectedImage = `clara17verse/claracore:${options.gpuType}`; + gpuInfo = { type: this.gpuType, name: `Manual: ${this.gpuType}` }; + } + + // Check if container already exists + const exists = await this.containerExists(); + + if (exists) { + log.info('Container exists, checking if it\'s running...'); + const container = await this.getContainer(); + const info = await container.inspect(); + + if (info.State.Running) { + log.info('✅ ClaraCore container is already running'); + this.isRunning = true; + return { success: true, message: 'Container already running', gpuType: this.gpuType }; + } else { + log.info('Starting existing container...'); + await container.start(); + this.isRunning = true; + return { success: true, message: 'Container started', gpuType: this.gpuType }; + } + } + + // Pull image if not present + const images = await this.docker.listImages(); + const imageExists = images.some(img => + img.RepoTags && img.RepoTags.includes(this.detectedImage) + ); + + if (!imageExists) { + log.info(`Image ${this.detectedImage} not found locally, pulling...`); + await this.pullImage(this.detectedImage); + } + + // Base container configuration + // Container runs on port 5890 internally, mapped to host port 8091 + // Only need a single volume for downloads (models are managed internally) + const containerConfig = { + Image: this.detectedImage, + name: this.containerName, + Hostname: 'clara-core', + ExposedPorts: { + '5890/tcp': {} + }, + Env: [ + 'NODE_ENV=production', + 'CLARA_PORT=5890' + ], + HostConfig: { + PortBindings: { + '5890/tcp': [{ HostPort: '8091' }] + }, + // Use named volume for downloads persistence + Binds: [ + 'claracore:/app/downloads' + ], + RestartPolicy: { + Name: 'unless-stopped' + } + } + }; + + // Add GPU-specific configurations + if (this.gpuType === 'cuda') { + containerConfig.Env.push('NVIDIA_VISIBLE_DEVICES=all'); + containerConfig.Env.push('NVIDIA_DRIVER_CAPABILITIES=compute,utility'); + + // Platform-specific GPU configuration + // Windows: Use Runtime property (DeviceRequests causes Docker daemon crashes on Windows) + // Linux: Use DeviceRequests for better compatibility + if (os.platform() === 'win32') { + containerConfig.HostConfig.Runtime = 'nvidia'; + } else { + // Linux: Use modern DeviceRequests API + containerConfig.HostConfig.DeviceRequests = [{ + Driver: 'nvidia', + Count: -1, // All GPUs + Capabilities: [['gpu', 'compute', 'utility']] + }]; + } + } else if (this.gpuType === 'rocm') { + // ROCm works on Windows ONLY through WSL2 + // Docker Desktop must be running in WSL2 mode (not Hyper-V) + if (os.platform() === 'win32') { + log.warn('⚠️ ROCm on Windows requires Docker Desktop in WSL2 mode'); + log.warn('Ensure: Windows 11 + AMD Radeon Software 25.8.1+ + WSL2 with Ubuntu'); + } + + containerConfig.HostConfig.Devices = [ + { PathOnHost: '/dev/kfd', PathInContainer: '/dev/kfd', CgroupPermissions: 'rwm' }, + { PathOnHost: '/dev/dri', PathInContainer: '/dev/dri', CgroupPermissions: 'rwm' } + ]; + containerConfig.Env.push('HSA_OVERRIDE_GFX_VERSION=10.3.0'); + } else if (this.gpuType === 'vulkan') { + containerConfig.HostConfig.Devices = [ + { PathOnHost: '/dev/dri', PathInContainer: '/dev/dri', CgroupPermissions: 'rwm' } + ]; + containerConfig.Env.push('VK_ICD_FILENAMES=/usr/share/vulkan/icd.d/nvidia_icd.json'); + } + + log.info('Creating ClaraCore container with config:', JSON.stringify(containerConfig, null, 2)); + + // Create and start container + let container; + try { + container = await this.docker.createContainer(containerConfig); + } catch (createError) { + // If CUDA/GPU creation fails on Windows, fall back to CPU mode + if (this.gpuType === 'cuda' && os.platform() === 'win32' && + (createError.message.includes('runtime') || createError.message.includes('nvidia') || createError.message.includes('gpu'))) { + log.warn('⚠️ Failed to create container with GPU support, falling back to CPU mode'); + log.warn('GPU Error:', createError.message); + + // Retry with CPU image + this.gpuType = 'cpu'; + this.detectedImage = 'clara17verse/claracore:cpu'; + containerConfig.Image = this.detectedImage; + + // Remove GPU configurations + delete containerConfig.HostConfig.Runtime; + delete containerConfig.HostConfig.DeviceRequests; + containerConfig.Env = containerConfig.Env.filter(env => + !env.startsWith('NVIDIA_') + ); + + log.info('Retrying with CPU configuration...'); + container = await this.docker.createContainer(containerConfig); + } else { + throw createError; + } + } + + await container.start(); + + log.info('✅ ClaraCore container created and started successfully'); + this.isRunning = true; + + // Wait for service to be ready + await this.waitForHealthy(); + + return { + success: true, + message: 'ClaraCore started in Docker', + gpuType: this.gpuType, + image: this.detectedImage + }; + + } catch (error) { + log.error('❌ Failed to start ClaraCore Docker container:', error); + this.isRunning = false; + + // Provide user-friendly error message for port conflicts + if (error.message && error.message.includes('port') && error.message.includes('8091')) { + const friendlyError = new Error( + 'Port 8091 is still in use. Please:\n' + + '1. Stop any running ClaraCore instances (Local mode)\n' + + '2. Check Task Manager for processes using port 8091\n' + + '3. Try restarting the application' + ); + friendlyError.originalError = error; + throw friendlyError; + } + + throw error; + } + } + + /** + * Stop Clara Core container + */ + async stop() { + try { + const exists = await this.containerExists(); + + if (!exists) { + log.info('Container does not exist, nothing to stop'); + this.isRunning = false; + return { success: true, message: 'Container not found' }; + } + + const container = await this.getContainer(); + const info = await container.inspect(); + + if (info.State.Running) { + log.info('Stopping ClaraCore container...'); + await container.stop({ t: 10 }); // 10 second graceful shutdown + log.info('✅ ClaraCore container stopped'); + } else { + log.info('Container is not running'); + } + + this.isRunning = false; + return { success: true, message: 'Container stopped' }; + + } catch (error) { + log.error('❌ Failed to stop ClaraCore container:', error); + throw error; + } + } + + /** + * Restart Clara Core container + */ + async restart() { + log.info('Restarting ClaraCore container...'); + await this.stop(); + await new Promise(resolve => setTimeout(resolve, 2000)); + return await this.start(); + } + + /** + * Remove Clara Core container + */ + async remove() { + try { + const exists = await this.containerExists(); + + if (!exists) { + log.info('Container does not exist, nothing to remove'); + return { success: true, message: 'Container not found' }; + } + + const container = await this.getContainer(); + const info = await container.inspect(); + + // Stop if running + if (info.State.Running) { + await container.stop({ t: 10 }); + } + + // Remove container + await container.remove(); + log.info('✅ ClaraCore container removed'); + + this.isRunning = false; + return { success: true, message: 'Container removed' }; + + } catch (error) { + log.error('❌ Failed to remove ClaraCore container:', error); + throw error; + } + } + + /** + * Wait for container to be healthy + */ + async waitForHealthy(maxAttempts = 30, interval = 2000) { + const http = require('http'); + + for (let i = 0; i < maxAttempts; i++) { + try { + const isHealthy = await new Promise((resolve) => { + const req = http.get('http://localhost:8091/health', (res) => { + resolve(res.statusCode === 200); + }); + + req.on('error', () => resolve(false)); + req.setTimeout(3000, () => { + req.destroy(); + resolve(false); + }); + }); + + if (isHealthy) { + log.info('✅ ClaraCore is healthy and ready'); + return true; + } + } catch (error) { + // Ignore errors during health check attempts + } + + await new Promise(resolve => setTimeout(resolve, interval)); + } + + log.warn('⚠️ ClaraCore health check timeout, but container is running'); + return false; + } + + /** + * Check container health + */ + async checkHealth() { + const http = require('http'); + + return new Promise((resolve) => { + const req = http.get('http://localhost:8091/health', (res) => { + resolve(res.statusCode === 200); + }); + + req.on('error', () => resolve(false)); + req.setTimeout(3000, () => { + req.destroy(); + resolve(false); + }); + }); + } + + /** + * Get container status + */ + async getStatus() { + try { + const exists = await this.containerExists(); + + if (!exists) { + return { + exists: false, + running: false, + gpuType: this.gpuType, + image: this.detectedImage + }; + } + + const container = await this.getContainer(); + const info = await container.inspect(); + + return { + exists: true, + running: info.State.Running, + status: info.State.Status, + started: info.State.StartedAt, + gpuType: this.gpuType, + image: info.Config.Image, + ports: info.NetworkSettings.Ports + }; + + } catch (error) { + log.error('Error getting container status:', error); + return { + exists: false, + running: false, + error: error.message + }; + } + } + + /** + * Get container logs + */ + async getLogs(options = { tail: 100 }) { + try { + const container = await this.getContainer(); + const logs = await container.logs({ + stdout: true, + stderr: true, + tail: options.tail || 100, + timestamps: true + }); + + return logs.toString('utf8'); + } catch (error) { + log.error('Error getting container logs:', error); + throw error; + } + } +} + +module.exports = ClaraCoreDockerService; diff --git a/electron/claraCoreRemoteService.cjs b/electron/claraCoreRemoteService.cjs new file mode 100644 index 00000000..1801717d --- /dev/null +++ b/electron/claraCoreRemoteService.cjs @@ -0,0 +1,1323 @@ +const { Client } = require('ssh2'); +const log = require('electron-log'); + +/** + * ClaraCore Remote Deployment Service + * Handles SSH connection, hardware detection, and Docker deployment + */ +class ClaraCoreRemoteService { + constructor() { + this.conn = null; + // SECURITY NOTE: sudoPassword is only stored temporarily during deployment + // It is: + // 1. Set at deployment start + // 2. Used only for sudo operations during deployment + // 3. Cleared immediately after deployment (success or failure) + // 4. Never persisted to disk or logs + // 5. Transmitted only over encrypted SSH connection + this.sudoPassword = null; + } + + /** + * Test SSH connection and detect hardware + */ + async testSetup(config) { + return new Promise((resolve, reject) => { + const conn = new Client(); + let isResolved = false; + + const timeout = setTimeout(() => { + if (!isResolved) { + isResolved = true; + conn.end(); + reject(new Error('Connection timeout after 30 seconds')); + } + }, 30000); + + conn.on('ready', async () => { + log.info('SSH connection established'); + + try { + // Detect hardware + const hardware = await this.detectHardware(conn); + + clearTimeout(timeout); + conn.end(); + + if (!isResolved) { + isResolved = true; + resolve({ + success: true, + hardware + }); + } + } catch (error) { + clearTimeout(timeout); + conn.end(); + + if (!isResolved) { + isResolved = true; + resolve({ + success: false, + error: error.message + }); + } + } + }); + + // Handle keyboard-interactive authentication (required for Raspberry Pi and similar SSH servers) + conn.on('keyboard-interactive', (name, instructions, instructionsLang, prompts, finish) => { + finish([config.password]); + }); + + conn.on('error', (err) => { + clearTimeout(timeout); + log.error('SSH connection error:', err); + + if (!isResolved) { + isResolved = true; + resolve({ + success: false, + error: err.message + }); + } + }); + + // Connect + conn.connect({ + host: config.host, + port: config.port || 22, + username: config.username, + password: config.password, + tryKeyboard: true, // Enable keyboard-interactive auth (required for some SSH servers like Raspberry Pi) + readyTimeout: 30000 + }); + }); + } + + /** + * Detect hardware and recommend container image + */ + async detectHardware(conn) { + const details = { + docker: false, + nvidia: false, + rocm: false, + vulkan: false, + strix: false, + architecture: 'unknown' + }; + + try { + // Check CPU Architecture + const archInfo = await this.execCommand(conn, 'uname -m'); + if (archInfo) { + details.architecture = archInfo.trim(); + log.info(`Detected architecture: ${details.architecture}`); + } + + // Check Docker + const dockerVersion = await this.execCommand(conn, 'docker --version 2>/dev/null'); + if (dockerVersion && !dockerVersion.includes('command not found')) { + details.docker = true; + details.dockerVersion = dockerVersion.trim(); + } + + // Check NVIDIA GPU + const nvidiaInfo = await this.execCommand(conn, 'nvidia-smi --query-gpu=name --format=csv,noheader 2>/dev/null'); + if (nvidiaInfo && !nvidiaInfo.includes('command not found') && nvidiaInfo.trim()) { + details.nvidia = true; + details.gpuInfo = nvidiaInfo.trim(); + + // Check CUDA version + const cudaVersion = await this.execCommand(conn, 'nvcc --version 2>/dev/null | grep "release" | awk \'{print $5}\''); + if (cudaVersion && cudaVersion.trim()) { + details.cudaVersion = cudaVersion.trim().replace(',', ''); + } + } + + // Check AMD ROCm + const rocmInfo = await this.execCommand(conn, 'rocm-smi --showproductname 2>/dev/null'); + if (rocmInfo && !rocmInfo.includes('command not found') && rocmInfo.trim()) { + details.rocm = true; + + const rocmVersion = await this.execCommand(conn, 'cat /opt/rocm/.info/version 2>/dev/null'); + if (rocmVersion && rocmVersion.trim()) { + details.rocmVersion = rocmVersion.trim(); + } + + // Check if ROCm devices are accessible (critical for container usage) + const kfdCheck = await this.execCommand(conn, 'test -e /dev/kfd && echo "exists" || echo "missing"'); + details.rocmDeviceAccessible = (kfdCheck.trim() === 'exists'); + + log.info(`[Remote] ROCm detected: v${details.rocmVersion || 'unknown'}, /dev/kfd: ${details.rocmDeviceAccessible ? 'available' : 'MISSING'}`); + } + + // Check for Vulkan support (fallback for AMD GPUs when ROCm devices unavailable) + // First check if /dev/dri exists (this is what matters for container GPU access) + const driCheck = await this.execCommand(conn, 'test -e /dev/dri && echo "exists" || echo "missing"'); + details.vulkanDeviceAccessible = (driCheck.trim() === 'exists'); + + // Then check if vulkaninfo is installed + const vulkanCheck = await this.execCommand(conn, 'vulkaninfo --summary 2>/dev/null | grep -i "Vulkan Instance Version"'); + if (vulkanCheck && !vulkanCheck.includes('command not found') && vulkanCheck.trim()) { + details.vulkan = true; + details.vulkanInstalled = true; + + // Try to get Vulkan device name + const vulkanDevice = await this.execCommand(conn, 'vulkaninfo 2>/dev/null | grep "deviceName" | head -1'); + if (vulkanDevice && vulkanDevice.trim()) { + details.vulkanDevice = vulkanDevice.replace(/.*deviceName\s*=\s*/, '').trim(); + } + } else if (details.vulkanDeviceAccessible) { + // /dev/dri exists but vulkaninfo not installed - we can still use Vulkan! + // setupVulkan will install the necessary packages + details.vulkan = true; + details.vulkanInstalled = false; + log.info('[Remote] Vulkan: /dev/dri available, vulkaninfo not installed (will be installed during setup)'); + } + + if (details.vulkanDeviceAccessible) { + log.info(`[Remote] Vulkan compatible: YES (/dev/dri available, installed: ${details.vulkanInstalled ? 'YES' : 'NO'})`); + } + + // Check for Strix Halo (Ryzen AI Max) + const cpuInfo = await this.execCommand(conn, 'lscpu | grep "Model name"'); + if (cpuInfo) { + details.cpuModel = cpuInfo.replace('Model name:', '').trim(); + + // Check for Strix Halo keywords + if (cpuInfo.includes('Ryzen AI Max') || cpuInfo.includes('Strix') || cpuInfo.includes('8040')) { + details.strix = true; + } + } + + // Check if ARM architecture (not supported yet) + const isARM = details.architecture.includes('arm') || + details.architecture.includes('aarch'); + + if (isARM) { + return { + detected: 'unsupported', + confidence: 'high', + details, + error: `ARM architecture (${details.architecture}) is not supported yet. ClaraCore Docker images are currently only available for x86_64/amd64 architecture.`, + unsupportedReason: 'arm' + }; + } + + // Determine recommendation with smart fallback logic + let detected = 'cpu'; + let confidence = 'high'; + let fallbackReason = null; + + if (details.nvidia) { + detected = 'cuda'; + confidence = details.cudaVersion ? 'high' : 'medium'; + } else if (details.strix) { + detected = 'strix'; + confidence = 'high'; + } else if (details.rocm && details.rocmDeviceAccessible) { + // ROCm is available AND devices are accessible + detected = 'rocm'; + confidence = 'high'; + } else if (details.rocm && !details.rocmDeviceAccessible && details.vulkan && details.vulkanDeviceAccessible) { + // ROCm installed but /dev/kfd missing, fall back to Vulkan + detected = 'vulkan'; + confidence = 'high'; // Changed to high since we know /dev/dri works + fallbackReason = 'ROCm detected but /dev/kfd not accessible. Using Vulkan as fallback for GPU acceleration.'; + log.warn(`[Remote] ⚠️ ${fallbackReason}`); + log.info(`[Remote] ℹ️ ROCm version: ${details.rocmVersion || 'unknown'}`); + log.info(`[Remote] ℹ️ /dev/kfd: missing`); + log.info(`[Remote] ℹ️ /dev/dri: available`); + log.info(`[Remote] ✅ Vulkan will provide GPU acceleration without requiring /dev/kfd`); + } else if (details.vulkan && details.vulkanDeviceAccessible) { + // Vulkan available (no ROCm or ROCm not accessible) + detected = 'vulkan'; + confidence = 'high'; + log.info(`[Remote] ℹ️ Vulkan GPU acceleration available via /dev/dri`); + } + + return { + detected, + confidence, + details, + fallbackReason + }; + + } catch (error) { + log.error('Hardware detection error:', error); + throw error; + } + } + + /** + * Deploy ClaraCore using native installation script + * For ROCm, Vulkan, Strix Halo, and CPU modes + * Uses port 5800 + */ + async deployNative(conn, config, hardwareType) { + try { + log.info(`[Remote] Starting native ClaraCore installation for ${hardwareType.toUpperCase()}...`); + + // 1. Download install.sh script + log.info('[Remote] Downloading ClaraCore installation script...'); + await this.execCommand(conn, 'curl -fsSL https://raw.githubusercontent.com/claraverse-space/ClaraCore/main/scripts/install.sh -o /tmp/claracore-install.sh'); + + // 2. Make executable + await this.execCommand(conn, 'chmod +x /tmp/claracore-install.sh'); + + // 3. Execute installation script + log.info('[Remote] Running ClaraCore installation (this may take a few minutes)...'); + try { + await this.execCommandWithOutput(conn, 'sudo bash /tmp/claracore-install.sh'); + } catch (installError) { + // Check if error is just due to script output or actual failure + log.warn(`[Remote] Installation script completed with warnings: ${installError.message}`); + } + + // 4. Wait for service to start + log.info('[Remote] Waiting for ClaraCore service to start...'); + await this.sleep(8000); // Give systemd time to start the service + + // 5. Check if service is running via systemd + const serviceStatus = await this.execCommand(conn, 'systemctl --user is-active claracore 2>&1 || echo "not-active"'); + + if (serviceStatus.includes('active')) { + log.info('[Remote] ✅ ClaraCore service is active'); + } else { + log.warn('[Remote] ⚠️ Service may not be active yet. Status: ' + serviceStatus.trim()); + + // Try to start it manually + log.info('[Remote] Attempting to start service manually...'); + await this.execCommand(conn, 'systemctl --user start claracore 2>&1 || true'); + await this.sleep(5000); + } + + // 6. Verify service is responding on port 5800 + log.info('[Remote] Verifying ClaraCore is responding on port 5800...'); + let healthCheckSuccess = false; + + for (let i = 0; i < 10; i++) { + const healthCheck = await this.execCommand(conn, 'curl -sf http://localhost:5800/ 2>&1 || echo "not-ready"'); + + if (!healthCheck.includes('not-ready') && !healthCheck.includes('Connection refused')) { + healthCheckSuccess = true; + log.info('[Remote] ✅ ClaraCore is responding on port 5800'); + break; + } + + log.info(`[Remote] Waiting for service to respond... (attempt ${i + 1}/10)`); + await this.sleep(3000); + } + + if (!healthCheckSuccess) { + // Get service logs for debugging + const serviceLogs = await this.execCommand(conn, 'systemctl --user status claracore 2>&1 || journalctl --user -u claracore -n 20 2>&1 || echo "No logs available"'); + log.warn(`[Remote] Service logs:\n${serviceLogs}`); + + throw new Error('ClaraCore service did not respond on port 5800 after installation. Check service logs.'); + } + + // 7. Clean up installation script + await this.execCommand(conn, 'rm -f /tmp/claracore-install.sh'); + + log.info(`[Remote] ✅ Native ClaraCore installation completed successfully (${hardwareType.toUpperCase()})`); + + return { + success: true, + url: `http://${config.host}:5800`, + port: 5800, + deploymentMethod: 'native', + hardwareType: hardwareType, + message: `Successfully deployed ClaraCore via native installation (${hardwareType.toUpperCase()})`, + containerName: null // No container for native installation + }; + + } catch (error) { + log.error('[Remote] Native installation failed:', error); + throw error; + } + } + + /** + * Manage native ClaraCore service via systemd + * @param {Object} conn - SSH connection + * @param {string} action - Action: 'start', 'stop', 'restart', 'status', 'logs' + * @returns {Promise} - Action result + */ + async manageNativeService(conn, action) { + try { + log.info(`[Remote] Managing ClaraCore service: ${action}`); + + let command; + switch (action) { + case 'start': + command = 'systemctl --user start claracore'; + break; + case 'stop': + command = 'systemctl --user stop claracore'; + break; + case 'restart': + command = 'systemctl --user restart claracore'; + break; + case 'status': + command = 'systemctl --user status claracore'; + break; + case 'logs': + command = 'journalctl --user -u claracore -n 50 --no-pager'; + break; + default: + throw new Error(`Unknown action: ${action}`); + } + + const result = await this.execCommand(conn, command); + + if (action === 'status') { + // Parse status output + const isActive = result.includes('active (running)'); + const isEnabled = result.includes('enabled'); + + return { + success: true, + action, + isActive, + isEnabled, + output: result + }; + } else if (action === 'logs') { + return { + success: true, + action, + logs: result + }; + } else { + // start, stop, restart + return { + success: true, + action, + message: `Service ${action} completed successfully` + }; + } + } catch (error) { + log.error(`[Remote] Service ${action} failed:`, error); + return { + success: false, + action, + error: error.message + }; + } + } + + /** + * Deploy ClaraCore using Docker container + * For CUDA mode only + * Uses port 5890 + */ + async deployDocker(conn, config, hardwareType) { + try { + const imageName = `clara17verse/claracore:${hardwareType}`; + const containerName = `claracore-${hardwareType}`; + + log.info(`[Remote] Deploying ${imageName} via Docker...`); + + // 1. Check if Docker is installed + const hasDocker = await this.checkDocker(conn); + if (!hasDocker) { + log.info('[Remote] Installing Docker...'); + await this.installDocker(conn); + } + + // 1.5. Ensure clara_network exists + log.info('[Remote] Setting up Clara network...'); + const networkCheck = await this.execCommand(conn, 'docker network ls --filter name=clara_network --format "{{.Name}}"'); + if (!networkCheck || !networkCheck.includes('clara_network')) { + await this.execCommand(conn, 'docker network create clara_network --driver bridge --subnet 172.25.0.0/16'); + log.info('[Remote] ✓ Clara network created'); + } else { + log.info('[Remote] ✓ Clara network exists'); + } + + // 2. Install CUDA prerequisites (only for CUDA) + if (hardwareType === 'cuda') { + await this.setupCuda(conn); + } + + // 3. Stop and remove existing container + log.info('[Remote] Cleaning up existing containers...'); + await this.execCommand(conn, `docker stop ${containerName} 2>/dev/null || true`); + await this.execCommand(conn, `docker rm ${containerName} 2>/dev/null || true`); + + // 4. Pull the image + log.info(`[Remote] Pulling image ${imageName}...`); + await this.execCommandWithOutput(conn, `docker pull ${imageName}`); + + // 5. Run the container with CUDA flags + log.info(`[Remote] Starting container ${containerName}...`); + const runCommand = this.buildDockerRunCommand(hardwareType, containerName, imageName, []); + + try { + await this.execCommand(conn, runCommand); + } catch (runError) { + log.error(`[Remote] Docker run command failed: ${runError.message}`); + throw new Error(`Failed to start container: ${runError.message}`); + } + + // 6. Wait for container to be healthy + log.info('[Remote] Waiting for container to start...'); + await this.sleep(5000); + + // 7. Verify container is running + const isRunning = await this.execCommand(conn, `docker ps -q -f name=${containerName}`); + if (!isRunning || !isRunning.trim()) { + const logs = await this.execCommand(conn, `docker logs ${containerName} 2>&1 || echo "No logs available"`); + const inspectResult = await this.execCommand(conn, `docker inspect ${containerName} --format='{{.State.Status}}: {{.State.Error}}' 2>&1 || echo "Container not found"`); + throw new Error(`Container failed to start.\n\nStatus: ${inspectResult}\n\nLogs:\n${logs.substring(0, 500)}`); + } + + log.info('[Remote] ✅ Container started successfully!'); + + // 8. Check if service is responding + log.info('[Remote] Verifying service health...'); + const healthCheck = await this.execCommand(conn, `curl -sf http://localhost:5890/health 2>&1 || echo "Health check not available"`); + if (healthCheck.includes('Health check not available')) { + log.warn('[Remote] Service health endpoint not available, but container is running'); + } else { + log.info('[Remote] ✅ Service is healthy and responding'); + } + + return { + success: true, + url: `http://${config.host}:5890`, + port: 5890, + deploymentMethod: 'docker', + containerName: containerName, + hardwareType: hardwareType, + message: `Successfully deployed ClaraCore via Docker (${hardwareType.toUpperCase()})` + }; + + } catch (error) { + log.error('[Remote] Docker deployment failed:', error); + throw error; + } + } + + /** + * Deploy ClaraCore - routes to Docker or Native installation based on hardware type + * - CUDA: Uses Docker (port 5890) + * - ROCm, Vulkan, Strix, CPU: Uses Native installation (port 5800) + */ + async deploy(config) { + return new Promise((resolve, reject) => { + const conn = new Client(); + let isResolved = false; + + // Store password temporarily for this deployment session only + // It will be cleared in all exit paths (success/failure/timeout) + this.sudoPassword = config.password; + + const timeout = setTimeout(() => { + if (!isResolved) { + isResolved = true; + conn.end(); + this.sudoPassword = null; + reject(new Error('Deployment timeout after 5 minutes')); + } + }, 300000); // 5 minutes + + conn.on('ready', async () => { + log.info('SSH connection established for deployment'); + + try { + const { hardwareType } = config; + + // Choose deployment method based on hardware type + let result; + if (hardwareType === 'cuda') { + log.info('[Remote] Using Docker deployment for CUDA'); + result = await this.deployDocker(conn, config, hardwareType); + } else { + log.info(`[Remote] Using native installation for ${hardwareType.toUpperCase()}`); + result = await this.deployNative(conn, config, hardwareType); + } + + // Deployment successful - clean up and resolve + clearTimeout(timeout); + conn.end(); + + // Clear password from memory + this.sudoPassword = null; + + if (!isResolved) { + isResolved = true; + resolve(result); + } + + } catch (error) { + log.error('Deployment error:', error); + clearTimeout(timeout); + conn.end(); + + // Clear password from memory + this.sudoPassword = null; + + if (!isResolved) { + isResolved = true; + + // Provide better error messages + let errorMessage = error.message; + if (errorMessage.includes('incorrect password')) { + errorMessage = 'Incorrect sudo password. Please verify your SSH password and try again.'; + } else if (errorMessage.includes('Permission denied')) { + errorMessage = 'SSH authentication failed. Please check your credentials.'; + } + + resolve({ + success: false, + error: errorMessage + }); + } + } + }); + + // Handle keyboard-interactive authentication (required for Raspberry Pi and similar SSH servers) + conn.on('keyboard-interactive', (name, instructions, instructionsLang, prompts, finish) => { + finish([config.password]); + }); + + conn.on('error', (err) => { + clearTimeout(timeout); + log.error('SSH connection error during deployment:', err); + + // Clear password from memory + this.sudoPassword = null; + + if (!isResolved) { + isResolved = true; + + let errorMessage = err.message; + if (err.level === 'client-authentication') { + errorMessage = 'SSH authentication failed. Please check your username and password.'; + } else if (err.code === 'ECONNREFUSED') { + errorMessage = 'Connection refused. Please check the host and port.'; + } else if (err.code === 'ETIMEDOUT' || err.code === 'ENOTFOUND') { + errorMessage = 'Connection timeout. Please check the host address and your network connection.'; + } + + resolve({ + success: false, + error: errorMessage + }); + } + }); + + // Connect + conn.connect({ + host: config.host, + port: config.port || 22, + username: config.username, + password: config.password, + tryKeyboard: true, // Enable keyboard-interactive auth (required for some SSH servers like Raspberry Pi) + readyTimeout: 30000 + }); + }); + } + + /** + * Detect available DRI devices on the remote server + */ + async detectDRIDevices(conn) { + try { + // List all devices in /dev/dri/ + const devices = await this.execCommand(conn, 'ls -1 /dev/dri/ 2>/dev/null | grep -E "^(card|renderD)" || echo ""'); + + if (!devices || !devices.trim()) { + log.warn('[Remote] No DRI devices found in /dev/dri/'); + return []; + } + + // Parse device list and create full paths + const deviceList = devices.trim().split('\n') + .filter(d => d.trim()) + .map(d => `/dev/dri/${d.trim()}`); + + return deviceList; + } catch (error) { + log.error('[Remote] Error detecting DRI devices:', error.message); + return []; + } + } + + /** + * Build Docker run command based on hardware type + * Handles different contexts (Docker Desktop vs Docker Engine) + */ + buildDockerRunCommand(hardwareType, containerName, imageName, availableDevices = []) { + // Use clara_network and expose on both ports (8091 standard, 5890 legacy) + // Use 172.17.0.1 (default bridge gateway) to access host services from custom network + const baseCmd = `docker run -d --name ${containerName} --network clara_network --restart unless-stopped -p 8091:5890 -p 5890:5890 --add-host=host.docker.internal:172.17.0.1`; + const volume = `-v claracore-${hardwareType}-downloads:/app/downloads`; + + switch (hardwareType) { + case 'cuda': + // For CUDA, try --gpus all (requires nvidia runtime) + // If using Docker Engine with proper setup, this should work + return `${baseCmd} --gpus all ${volume} ${imageName}`; + + case 'rocm': + // AMD ROCm requires specific device access (/dev/kfd + DRI render devices) + // Use dynamically detected devices + const rocmDevices = availableDevices.map(d => `--device=${d}`).join(' '); + return `${baseCmd} --device=/dev/kfd ${rocmDevices} --group-add video --group-add render --ipc=host --cap-add=SYS_PTRACE --security-opt seccomp=unconfined ${volume} ${imageName}`; + + case 'vulkan': + // Vulkan only requires DRI render devices (no /dev/kfd needed) + // Use privileged mode if Docker can't see individual devices (namespace issue) + if (availableDevices.length === 0) { + log.warn('[Remote] No DRI devices detected, using --privileged mode for full device access'); + return `${baseCmd} --privileged --group-add video --group-add render -e VK_ICD_FILENAMES=/usr/share/vulkan/icd.d/radeon_icd.x86_64.json ${volume} ${imageName}`; + } + const vulkanDevices = availableDevices.map(d => `--device=${d}`).join(' '); + return `${baseCmd} ${vulkanDevices} --group-add video --group-add render --security-opt seccomp=unconfined -e VK_ICD_FILENAMES=/usr/share/vulkan/icd.d/radeon_icd.x86_64.json ${volume} ${imageName}`; + + case 'strix': + // Strix Halo (Ryzen AI Max) uses iGPU with Vulkan + // Use privileged mode if Docker can't see individual devices + if (availableDevices.length === 0) { + log.warn('[Remote] No DRI devices detected, using --privileged mode for full device access'); + return `${baseCmd} --privileged --group-add video --group-add render ${volume} ${imageName}`; + } + const strixDevices = availableDevices.map(d => `--device=${d}`).join(' '); + return `${baseCmd} ${strixDevices} --group-add video --group-add render --security-opt seccomp=unconfined ${volume} ${imageName}`; + + case 'cpu': + default: + // CPU-only version + return `${baseCmd} ${volume} ${imageName}`; + } + } + + /** + * Check if Docker is installed + */ + async checkDocker(conn) { + try { + const result = await this.execCommand(conn, 'docker --version 2>/dev/null'); + return result && !result.includes('command not found'); + } catch { + return false; + } + } + + /** + * Install Docker using official convenience script + * This is more reliable and works across all major Linux distributions + */ + async installDocker(conn) { + try { + log.info('[Remote] Detecting Linux distribution...'); + + // Detect the distribution + const osRelease = await this.execCommand(conn, 'cat /etc/os-release'); + const distro = this.detectDistro(osRelease); + + log.info(`[Remote] Detected distribution: ${distro}`); + + // For simplicity and reliability, use Docker's official convenience script + // This works across Ubuntu, Debian, Fedora, CentOS, and other distros + log.info('[Remote] Downloading Docker installation script...'); + await this.execCommand(conn, 'curl -fsSL https://get.docker.com -o /tmp/get-docker.sh'); + + log.info('[Remote] Installing Docker (this may take a few minutes)...'); + await this.execCommandWithOutput(conn, 'sudo sh /tmp/get-docker.sh'); + + // Clean up + await this.execCommand(conn, 'rm /tmp/get-docker.sh'); + + // Get current username + const username = await this.execCommand(conn, 'whoami'); + const user = username.trim() || 'ubuntu'; + + log.info(`[Remote] Adding user ${user} to docker group...`); + await this.execCommand(conn, `sudo usermod -aG docker ${user}`); + + log.info('[Remote] Starting Docker service...'); + await this.execCommand(conn, 'sudo systemctl start docker'); + await this.execCommand(conn, 'sudo systemctl enable docker'); + + log.info('[Remote] Docker installed successfully'); + + // Important: Warn about group membership + log.info('[Remote] Note: User needs to log out and back in for docker group to take effect'); + + } catch (error) { + log.error('[Remote] Docker installation failed:', error); + throw new Error(`Failed to install Docker: ${error.message}`); + } + } + + /** + * Detect Linux distribution from /etc/os-release + */ + detectDistro(osRelease) { + if (osRelease.includes('Ubuntu')) return 'Ubuntu'; + if (osRelease.includes('Debian')) return 'Debian'; + if (osRelease.includes('Fedora')) return 'Fedora'; + if (osRelease.includes('CentOS')) return 'CentOS'; + if (osRelease.includes('Red Hat')) return 'RHEL'; + if (osRelease.includes('Arch')) return 'Arch Linux'; + return 'Unknown Linux'; + } + + /** + * Setup NVIDIA CUDA with proper runtime configuration + */ + async setupCuda(conn) { + try { + // Check if nvidia-smi works (GPU drivers installed) + const nvidiaCheck = await this.execCommand(conn, 'nvidia-smi 2>/dev/null'); + if (!nvidiaCheck || nvidiaCheck.includes('command not found')) { + throw new Error('NVIDIA drivers not found. Please install NVIDIA drivers first.'); + } + + log.info('[Remote] NVIDIA drivers detected'); + + // Check if nvidia-container-toolkit is installed + const hasToolkit = await this.execCommand(conn, 'which nvidia-ctk 2>/dev/null'); + + if (!hasToolkit || !hasToolkit.trim()) { + log.info('[Remote] Installing NVIDIA Container Toolkit...'); + + // Detect package manager and distro + const hasApt = await this.execCommand(conn, 'which apt-get 2>/dev/null'); + const hasYum = await this.execCommand(conn, 'which yum 2>/dev/null'); + + if (hasApt && hasApt.trim()) { + await this.installNvidiaToolkitApt(conn); + } else if (hasYum && hasYum.trim()) { + await this.installNvidiaToolkitYum(conn); + } else { + throw new Error('Unsupported package manager. Only apt and yum are supported.'); + } + } else { + log.info('[Remote] NVIDIA Container Toolkit already installed'); + } + + // Configure Docker runtime + log.info('[Remote] Configuring NVIDIA runtime for Docker...'); + await this.execCommand(conn, 'sudo nvidia-ctk runtime configure --runtime=docker'); + + // Reload systemd and restart Docker + log.info('[Remote] Restarting Docker service...'); + await this.execCommand(conn, 'sudo systemctl daemon-reload'); + await this.execCommand(conn, 'sudo systemctl restart docker'); + + // Wait for Docker to be ready + await this.sleep(3000); + + // Check if Docker context needs to be switched from desktop-linux to default + const dockerContext = await this.execCommand(conn, 'docker context show 2>/dev/null'); + if (dockerContext && dockerContext.includes('desktop-linux')) { + log.info('[Remote] Switching from Docker Desktop to Docker Engine context...'); + await this.execCommand(conn, 'docker context use default'); + + // Get current user and ensure they're in docker group + const username = await this.execCommand(conn, 'whoami'); + const user = username.trim(); + await this.execCommand(conn, `sudo usermod -aG docker ${user}`); + + log.info('[Remote] Note: User may need to log out and back in for docker group to take effect'); + } + + // Verify NVIDIA runtime is available + const runtimeCheck = await this.execCommand(conn, 'docker info 2>/dev/null | grep -i runtime'); + if (runtimeCheck && runtimeCheck.includes('nvidia')) { + log.info('[Remote] NVIDIA Container Toolkit configured successfully'); + } else { + log.warn('[Remote] NVIDIA runtime may not be properly configured. Container may need manual intervention.'); + } + + } catch (error) { + log.error('[Remote] CUDA setup failed:', error); + throw error; + } + } + + /** + * Install NVIDIA Container Toolkit on Debian/Ubuntu (apt-based) + */ + async installNvidiaToolkitApt(conn) { + const commands = [ + // Add NVIDIA GPG key + { + cmd: 'curl -fsSL https://nvidia.github.io/libnvidia-container/gpgkey | sudo gpg --dearmor -o /usr/share/keyrings/nvidia-container-toolkit-keyring.gpg', + desc: 'Adding NVIDIA GPG key' + }, + // Add repository + { + cmd: 'curl -s -L https://nvidia.github.io/libnvidia-container/stable/deb/nvidia-container-toolkit.list | sed \'s#deb https://#deb [signed-by=/usr/share/keyrings/nvidia-container-toolkit-keyring.gpg] https://#g\' | sudo tee /etc/apt/sources.list.d/nvidia-container-toolkit.list', + desc: 'Adding NVIDIA repository' + }, + // Update and install + { cmd: 'sudo apt-get update', desc: 'Updating package lists' }, + { cmd: 'sudo apt-get install -y nvidia-container-toolkit', desc: 'Installing NVIDIA Container Toolkit' } + ]; + + for (const { cmd, desc } of commands) { + log.info(`[Remote] ${desc}...`); + await this.execCommandWithOutput(conn, cmd); + } + } + + /** + * Install NVIDIA Container Toolkit on RHEL/CentOS/Fedora (yum-based) + */ + async installNvidiaToolkitYum(conn) { + const commands = [ + { + cmd: 'curl -s -L https://nvidia.github.io/libnvidia-container/stable/rpm/nvidia-container-toolkit.repo | sudo tee /etc/yum.repos.d/nvidia-container-toolkit.repo', + desc: 'Adding NVIDIA repository' + }, + { cmd: 'sudo yum install -y nvidia-container-toolkit', desc: 'Installing NVIDIA Container Toolkit' } + ]; + + for (const { cmd, desc } of commands) { + log.info(`[Remote] ${desc}...`); + await this.execCommandWithOutput(conn, cmd); + } + } + + /** + * Setup AMD ROCm + */ + async setupRocm(conn) { + try { + log.info('[Remote] Validating ROCm device access...'); + + // More thorough check: verify /dev/kfd is actually a character device + const kfdCheck = await this.execCommand(conn, 'ls -la /dev/kfd 2>&1'); + log.info(`[Remote] DEBUG: ls -la /dev/kfd output:\n${kfdCheck}`); + + // Check if it's a character device (starts with 'c') + if (!kfdCheck || kfdCheck.includes('No such file') || kfdCheck.includes('cannot access')) { + throw new Error('ROCm device /dev/kfd not found. ROCm kernel drivers may not be loaded.\n\nTry loading the module: sudo modprobe amdgpu\nOr install ROCm drivers: https://rocmdocs.amd.com/en/latest/Installation_Guide/Installation-Guide.html'); + } + + if (!kfdCheck.startsWith('c')) { + throw new Error('/dev/kfd exists but is not a character device. ROCm kernel module (amdkfd) may not be loaded.\n\nTry: sudo modprobe amdgpu'); + } + + // Check if /dev/dri exists and is accessible + const driCheck = await this.execCommand(conn, 'ls -la /dev/dri 2>&1'); + log.info(`[Remote] DEBUG: ls -la /dev/dri output:\n${driCheck}`); + + if (!driCheck || driCheck.includes('No such file') || driCheck.includes('cannot access')) { + throw new Error('Device /dev/dri not found. AMD GPU drivers may not be installed correctly.'); + } + + log.info('[Remote] ✅ ROCm devices validated: /dev/kfd and /dev/dri are accessible'); + + // Ensure user is in video and render groups + const username = await this.execCommand(conn, 'whoami'); + const user = username.trim(); + log.info(`[Remote] Adding user ${user} to video and render groups...`); + await this.execCommand(conn, `sudo usermod -a -G video,render ${user}`); + + log.info('[Remote] ROCm setup complete. Note: User may need to log out and back in for group changes to take effect.'); + } catch (error) { + log.error('[Remote] ROCm setup failed:', error); + throw error; + } + } + + /** + * Setup Vulkan GPU acceleration (Fallback for AMD GPUs when ROCm unavailable) + * Only requires /dev/dri, no ROCm kernel drivers needed + */ + async setupVulkan(conn) { + try { + log.info('[Remote] Setting up Vulkan GPU acceleration...'); + + // 1. Validate DRI device (required for GPU access) + log.info('[Remote] Checking GPU device access...'); + const driCheck = await this.execCommand(conn, 'test -e /dev/dri && echo "exists" || echo "missing"'); + if (driCheck.trim() === 'missing') { + throw new Error('Device /dev/dri not found. AMD GPU drivers (amdgpu) may not be installed.\n\nPlease ensure the Linux kernel has amdgpu drivers loaded.'); + } + log.info('[Remote] ✓ GPU device found: /dev/dri'); + + // 2. Check for Vulkan support (critical for GPU acceleration) + log.info('[Remote] Checking Vulkan support...'); + const vulkanCheck = await this.execCommand(conn, 'which vulkaninfo 2>/dev/null'); + + if (!vulkanCheck || !vulkanCheck.trim()) { + // Vulkan not found - need to install + log.info('[Remote] Vulkan not found. Installing Vulkan drivers...'); + + // Detect distro + const osRelease = await this.execCommand(conn, 'cat /etc/os-release'); + const distro = this.detectDistro(osRelease); + log.info(`[Remote] Detected distribution: ${distro}`); + + // Install based on distro + if (osRelease.includes('Ubuntu') || osRelease.includes('Debian')) { + log.info('[Remote] Installing Vulkan packages for Ubuntu/Debian...'); + await this.execCommandWithOutput(conn, 'sudo apt-get update'); + await this.execCommandWithOutput(conn, 'sudo apt-get install -y mesa-vulkan-drivers vulkan-tools libvulkan1'); + } else if (osRelease.includes('Fedora') || osRelease.includes('Red Hat') || osRelease.includes('CentOS')) { + log.info('[Remote] Installing Vulkan packages for Fedora/RHEL...'); + await this.execCommandWithOutput(conn, 'sudo dnf install -y mesa-vulkan-drivers vulkan-tools vulkan-loader'); + } else if (osRelease.includes('Arch')) { + log.info('[Remote] Installing Vulkan packages for Arch Linux...'); + await this.execCommandWithOutput(conn, 'sudo pacman -S --noconfirm vulkan-radeon vulkan-tools'); + } else { + throw new Error(`Unsupported distribution: ${distro}. Please install mesa-vulkan-drivers and vulkan-tools manually.`); + } + + // Verify Vulkan installation + log.info('[Remote] Verifying Vulkan installation...'); + const vulkanVerify = await this.execCommand(conn, 'vulkaninfo --summary 2>&1 | grep -i "Vulkan Instance Version" || echo "failed"'); + if (vulkanVerify.includes('failed')) { + throw new Error('Vulkan installation verification failed. Please check the installation logs.'); + } + log.info('[Remote] ✓ Vulkan installed successfully'); + } else { + log.info('[Remote] ✓ Vulkan is already installed'); + } + + // 3. Verify Vulkan can detect GPU + log.info('[Remote] Verifying Vulkan GPU detection...'); + const vulkanDevices = await this.execCommand(conn, 'vulkaninfo 2>/dev/null | grep "deviceName" | head -1 || echo "No GPU detected"'); + if (vulkanDevices.includes('No GPU detected')) { + throw new Error('Vulkan is installed but cannot detect any GPU. Please check AMD GPU drivers.'); + } + log.info(`[Remote] ✓ Vulkan GPU detected: ${vulkanDevices.trim()}`); + + // 4. Ensure user is in video and render groups (required for GPU access) + const username = await this.execCommand(conn, 'whoami'); + const user = username.trim(); + log.info(`[Remote] Adding user ${user} to video and render groups...`); + await this.execCommand(conn, `sudo usermod -a -G video,render ${user}`); + + log.info('[Remote] ✅ Vulkan GPU acceleration setup complete!'); + log.info('[Remote] Note: This provides good GPU performance without requiring ROCm kernel drivers.'); + } catch (error) { + log.error('[Remote] Vulkan setup failed:', error); + throw error; + } + } + + /** + * Setup Strix Halo (Ryzen AI Max with integrated GPU) + * Focuses on Vulkan support for GPU acceleration + */ + async setupStrix(conn) { + try { + log.info('[Remote] Setting up Strix Halo (Ryzen AI Max) with Vulkan support...'); + + // 1. Validate DRI device (required for GPU access) + log.info('[Remote] Checking GPU device access...'); + const driCheck = await this.execCommand(conn, 'test -e /dev/dri && echo "exists" || echo "missing"'); + if (driCheck.trim() === 'missing') { + throw new Error('Device /dev/dri not found. AMD GPU drivers (amdgpu) may not be installed.\n\nPlease ensure the Linux kernel has amdgpu drivers loaded.'); + } + log.info('[Remote] ✓ GPU device found: /dev/dri'); + + // 2. Check for Vulkan support (critical for GPU acceleration) + log.info('[Remote] Checking Vulkan support...'); + const vulkanCheck = await this.execCommand(conn, 'which vulkaninfo 2>/dev/null'); + + if (!vulkanCheck || !vulkanCheck.trim()) { + // Vulkan not found - need to install + log.info('[Remote] Vulkan not found. Installing Vulkan drivers...'); + + // Detect distro + const osRelease = await this.execCommand(conn, 'cat /etc/os-release'); + const distro = this.detectDistro(osRelease); + log.info(`[Remote] Detected distribution: ${distro}`); + + // Install based on distro + if (osRelease.includes('Ubuntu') || osRelease.includes('Debian')) { + log.info('[Remote] Installing Vulkan packages for Ubuntu/Debian...'); + await this.execCommandWithOutput(conn, 'sudo apt-get update'); + await this.execCommandWithOutput(conn, 'sudo apt-get install -y mesa-vulkan-drivers vulkan-tools libvulkan1'); + } else if (osRelease.includes('Fedora') || osRelease.includes('Red Hat') || osRelease.includes('CentOS')) { + log.info('[Remote] Installing Vulkan packages for Fedora/RHEL...'); + await this.execCommandWithOutput(conn, 'sudo dnf install -y mesa-vulkan-drivers vulkan-tools vulkan-loader'); + } else if (osRelease.includes('Arch')) { + log.info('[Remote] Installing Vulkan packages for Arch Linux...'); + await this.execCommandWithOutput(conn, 'sudo pacman -S --noconfirm vulkan-radeon vulkan-tools'); + } else { + throw new Error(`Unsupported distribution: ${distro}. Please install mesa-vulkan-drivers and vulkan-tools manually.`); + } + + // Verify Vulkan installation + log.info('[Remote] Verifying Vulkan installation...'); + const vulkanVerify = await this.execCommand(conn, 'vulkaninfo --summary 2>&1 | grep -i "Vulkan Instance Version" || echo "failed"'); + if (!vulkanVerify.includes('failed')) { + log.info('[Remote] ✓ Vulkan installed and detected successfully'); + } else { + log.warn('[Remote] ⚠ Vulkan installed but may not be functioning. A system reboot might be required.'); + } + } else { + log.info('[Remote] ✓ Vulkan already installed'); + + // Quick Vulkan validation + const vulkanDevices = await this.execCommand(conn, 'vulkaninfo --summary 2>&1 | grep -i "deviceName" || echo "none"'); + if (!vulkanDevices.includes('none')) { + log.info(`[Remote] ✓ Vulkan GPU detected: ${vulkanDevices.trim()}`); + } + } + + // 3. Set up user permissions for GPU access + const username = await this.execCommand(conn, 'whoami'); + const user = username.trim(); + log.info(`[Remote] Adding user ${user} to video and render groups...`); + await this.execCommand(conn, `sudo usermod -a -G video,render ${user}`); + + log.info('[Remote] ✓ Strix Halo setup complete! GPU will be available via Vulkan.'); + log.info('[Remote] Note: User may need to log out and back in for group changes to take effect.'); + + } catch (error) { + log.error('[Remote] Strix Halo setup failed:', error); + throw error; + } + } + + /** + * Execute command with sudo support (using temporarily stored password) + */ + execCommand(conn, command) { + return new Promise((resolve, reject) => { + // Handle sudo commands with password properly + let execCommand = command; + + if (this.sudoPassword && command.includes('sudo')) { + // Escape password for shell + const escapedPassword = this.sudoPassword.replace(/'/g, "'\\''"); + + // For commands with pipes that contain sudo + if (command.includes('|') && command.includes('sudo')) { + const escapedCommand = command.replace(/'/g, "'\\''"); + execCommand = `bash -c "echo '${escapedPassword}' | ${command.replace(/sudo/g, 'sudo -S')}"`; + } else if (command.trim().startsWith('sudo ')) { + // Simple sudo command at start + execCommand = `echo '${escapedPassword}' | ${command.replace(/^sudo\s+/, 'sudo -S ')}`; + } + } + + conn.exec(execCommand, (err, stream) => { + if (err) { + reject(err); + return; + } + + let output = ''; + let errorOutput = ''; + + stream.on('close', (code) => { + if (code !== 0 && errorOutput) { + log.warn(`Command failed (code ${code}): ${command}`); + log.warn(`Error: ${errorOutput}`); + } + resolve(output || errorOutput); + }); + + stream.on('data', (data) => { + output += data.toString(); + }); + + stream.stderr.on('data', (data) => { + errorOutput += data.toString(); + }); + }); + }); + } + + /** + * Execute command and stream output (for long-running commands, using temporarily stored password) + */ + execCommandWithOutput(conn, command) { + return new Promise((resolve, reject) => { + // Handle sudo commands with password properly + let execCommand = command; + + if (this.sudoPassword && command.includes('sudo')) { + // For commands with pipes that contain sudo, we need to handle it specially + // Replace all instances of 'sudo' with proper password handling + if (command.includes('|') && command.includes('sudo')) { + // Wrap the entire command in a bash -c with password provided via -S + const escapedPassword = this.sudoPassword.replace(/'/g, "'\\''"); + const escapedCommand = command.replace(/'/g, "'\\''"); + execCommand = `bash -c "echo '${escapedPassword}' | ${command.replace(/sudo/g, 'sudo -S')}"`; + } else if (command.trim().startsWith('sudo ')) { + // Simple sudo command at start + const escapedPassword = this.sudoPassword.replace(/'/g, "'\\''"); + execCommand = `echo '${escapedPassword}' | ${command.replace(/^sudo\s+/, 'sudo -S ')}`; + } + } + + conn.exec(execCommand, (err, stream) => { + if (err) { + reject(err); + return; + } + + let hasOutput = false; + + let stderrOutput = ''; + + stream.on('close', (code) => { + if (code === 0) { + resolve(); + } else { + const errorMsg = stderrOutput ? + `Command failed with code ${code}: ${stderrOutput}` : + `Command failed with code ${code}`; + reject(new Error(errorMsg)); + } + }); + + stream.on('data', (data) => { + const output = data.toString().trim(); + if (output && !output.includes('[sudo] password') && !output.includes('Sorry, try again')) { + hasOutput = true; + log.info(`[Remote] ${output}`); + } + }); + + stream.stderr.on('data', (data) => { + const output = data.toString().trim(); + // Capture stderr for error reporting + if (output) { + stderrOutput += output + '\n'; + } + // Filter out sudo password prompts and sudo warnings for logging + if (output && + !output.includes('[sudo] password') && + !output.includes('Sorry, try again') && + !output.includes('sudo: a password is required')) { + log.info(`[Remote] ${output}`); + } + }); + }); + }); + } + + /** + * Monitor remote ClaraCore services + * Returns status of all ClaraCore containers running on remote server + */ + async monitorRemoteServices(config) { + return new Promise((resolve, reject) => { + const conn = new Client(); + let isResolved = false; + + const timeout = setTimeout(() => { + if (!isResolved) { + isResolved = true; + conn.end(); + reject(new Error('Monitor timeout after 15 seconds')); + } + }, 15000); + + conn.on('ready', async () => { + try { + // List all claracore containers + const containerListCmd = 'docker ps -a --filter "name=claracore-" --format "{{.Names}}|{{.Status}}|{{.Ports}}"'; + const containerList = await this.execCommand(conn, containerListCmd); + + const services = []; + + if (containerList && containerList.trim()) { + const lines = containerList.trim().split('\n'); + + for (const line of lines) { + const [name, status, ports] = line.split('|'); + + // Extract hardware type from container name (claracore-cuda, claracore-rocm, etc.) + const hardwareType = name.replace('claracore-', ''); + const isRunning = status.toLowerCase().includes('up'); + + // Check health if running + let isHealthy = false; + if (isRunning) { + try { + const healthCheck = await this.execCommand(conn, `curl -sf http://localhost:5890/health 2>&1`); + isHealthy = healthCheck && !healthCheck.includes('Failed to connect'); + } catch { + isHealthy = false; + } + } + + services.push({ + name, + hardwareType, + status: isRunning ? 'running' : 'stopped', + isHealthy: isRunning ? isHealthy : false, + ports: ports || 'N/A', + url: isRunning ? `http://${config.host}:5890` : null + }); + } + } + + clearTimeout(timeout); + conn.end(); + + if (!isResolved) { + isResolved = true; + resolve({ + success: true, + host: config.host, + services, + totalServices: services.length, + runningServices: services.filter(s => s.status === 'running').length, + healthyServices: services.filter(s => s.isHealthy).length, + timestamp: new Date().toISOString() + }); + } + } catch (error) { + log.error('Monitor error:', error); + clearTimeout(timeout); + conn.end(); + + if (!isResolved) { + isResolved = true; + reject(error); + } + } + }); + + // Handle keyboard-interactive authentication (required for Raspberry Pi and similar SSH servers) + conn.on('keyboard-interactive', (name, instructions, instructionsLang, prompts, finish) => { + finish([config.password]); + }); + + conn.on('error', (err) => { + if (!isResolved) { + isResolved = true; + clearTimeout(timeout); + reject(err); + } + }); + + conn.connect({ + host: config.host, + port: config.port || 22, + username: config.username, + password: config.password, + tryKeyboard: true, // Enable keyboard-interactive auth (required for some SSH servers like Raspberry Pi) + readyTimeout: 15000 + }); + }); + } + + /** + * Sleep helper + */ + sleep(ms) { + return new Promise(resolve => setTimeout(resolve, ms)); + } +} + +module.exports = ClaraCoreRemoteService; diff --git a/electron/claraCoreService.cjs b/electron/claraCoreService.cjs new file mode 100644 index 00000000..2559d4d4 --- /dev/null +++ b/electron/claraCoreService.cjs @@ -0,0 +1,591 @@ +const { spawn } = require('child_process'); +const path = require('path'); +const fs = require('fs'); +const log = require('electron-log'); +const { app } = require('electron'); +const os = require('os'); + +/** + * ClaraCore Service Manager + * Manages the ClaraCore AI Engine binary service + * Supports local, remote, and docker deployment modes + */ +class ClaraCoreService { + constructor() { + this.process = null; + this.isRunning = false; + this.startTime = null; + this.restartAttempts = 0; + this.maxRestartAttempts = 3; + this.lastError = null; + this.isStarting = false; // Prevent concurrent start attempts + this.isIntentionallyStopped = false; // Track intentional stops + this.restartTimer = null; // Track auto-restart timer + + // Set up writable config directory in user data path + this.configDir = path.join(app.getPath('userData'), 'claracore'); + this.configPath = path.join(this.configDir, 'config.yaml'); + } + + /** + * Get the platform-specific binary path + */ + getBinaryPath() { + const platform = os.platform(); + const arch = os.arch(); + + let binaryName; + + if (platform === 'win32') { + binaryName = 'claracore-windows-amd64.exe'; + } else if (platform === 'darwin') { + binaryName = arch === 'arm64' + ? 'claracore-darwin-arm64' + : 'claracore-darwin-amd64'; + } else if (platform === 'linux') { + binaryName = arch === 'arm64' + ? 'claracore-linux-arm64' + : 'claracore-linux-amd64'; + } else { + throw new Error(`Unsupported platform: ${platform}`); + } + + // Multiple paths to try (in order of preference) + const pathsToTry = []; + + // 1. Production: electron app resources + if (process.resourcesPath) { + pathsToTry.push(path.join(process.resourcesPath, 'electron', 'claracore', binaryName)); + pathsToTry.push(path.join(process.resourcesPath, 'claracore', binaryName)); + } + + // 2. Development: relative to electron directory + pathsToTry.push(path.join(__dirname, 'claracore', binaryName)); + + // 3. Development: relative to project root + const projectRoot = path.resolve(__dirname, '..'); + pathsToTry.push(path.join(projectRoot, 'electron', 'claracore', binaryName)); + pathsToTry.push(path.join(projectRoot, 'claracore', binaryName)); + + // Check which path exists + for (const tryPath of pathsToTry) { + if (fs.existsSync(tryPath)) { + log.info(`ClaraCore binary found at: ${tryPath}`); + return tryPath; + } + } + + // If none found, log all tried paths for debugging + log.error(`ClaraCore binary not found. Tried paths: ${pathsToTry.join(', ')}`); + throw new Error(`ClaraCore binary not found. Tried ${pathsToTry.length} paths.`); + } + + /** + * Check if a port is in use + */ + async isPortInUse(port) { + const net = require('net'); + + return new Promise((resolve) => { + const server = net.createServer(); + + server.once('error', (err) => { + if (err.code === 'EADDRINUSE') { + resolve(true); // Port is in use + } else { + resolve(false); + } + }); + + server.once('listening', () => { + server.close(); + resolve(false); // Port is available + }); + + server.listen(port); + }); + } + + /** + * Check if ClaraCore is running on the port by hitting its API + */ + async isClaraCoreRunningOnPort() { + try { + const http = require('http'); + + return new Promise((resolve) => { + const options = { + hostname: 'localhost', + port: 8091, + path: '/health', + method: 'GET', + timeout: 2000 + }; + + const req = http.request(options, (res) => { + let data = ''; + res.on('data', (chunk) => { data += chunk; }); + res.on('end', () => { + // If we get a response, assume it's ClaraCore + resolve(true); + }); + }); + + req.on('error', () => { + // Can't reach the service, probably not ClaraCore + resolve(false); + }); + + req.on('timeout', () => { + req.destroy(); + resolve(false); + }); + + req.end(); + }); + } catch (error) { + return false; + } + } + + /** + * Kill any existing ClaraCore process running on port 8091 + */ + async killExistingClaraCore() { + try { + log.info('Attempting to gracefully shutdown existing ClaraCore process on port 8091...'); + + // First, try to shutdown gracefully via HTTP using ClaraCore's restart endpoint + try { + const axios = require('axios'); + // Use the hard restart endpoint which actually shuts down the service + await axios.post('http://127.0.0.1:8091/api/server/restart/hard', {}, { timeout: 3000 }); + log.info('✅ Sent hard restart/shutdown request to existing ClaraCore'); + await new Promise(resolve => setTimeout(resolve, 2000)); + + // Check if it actually stopped + if (!(await this.isPortInUse(8091))) { + log.info('✅ ClaraCore gracefully shut down via HTTP'); + return true; + } + } catch (shutdownError) { + // Graceful shutdown failed, continue with force kill + log.debug('HTTP shutdown failed, attempting force kill...', shutdownError.message); + } + + if (os.platform() === 'win32') { + // Windows: Find PID using netstat and kill it + const { execSync } = require('child_process'); + try { + const netstatOutput = execSync('netstat -ano | findstr :8091 | findstr LISTENING', { encoding: 'utf8' }); + + if (netstatOutput) { + // Extract PID from netstat output + const lines = netstatOutput.trim().split('\n'); + + for (const line of lines) { + const parts = line.trim().split(/\s+/); + const pid = parts[parts.length - 1]; + + if (pid && !isNaN(pid)) { + try { + execSync(`taskkill /F /PID ${pid}`, { encoding: 'utf8' }); + log.info(`✅ Killed existing ClaraCore process (PID: ${pid})`); + + // Wait a bit for the port to be released + await new Promise(resolve => setTimeout(resolve, 1000)); + return true; + } catch (killError) { + log.warn(`Failed to kill process ${pid}:`, killError.message); + // Don't throw - continue execution + } + } + } + } + } catch (netstatError) { + log.debug('No process found on port 8091'); + } + } else { + // Unix-like systems: Use lsof + const { execSync } = require('child_process'); + try { + const lsofOutput = execSync('lsof -ti:8091', { encoding: 'utf8' }).trim(); + + if (lsofOutput) { + const pids = lsofOutput.split('\n').filter(pid => pid); + for (const pid of pids) { + try { + // First try SIGTERM (graceful) + execSync(`kill -15 ${pid}`, { encoding: 'utf8' }); + log.info(`✅ Sent SIGTERM to ClaraCore process (PID: ${pid})`); + + // Wait for graceful shutdown + await new Promise(resolve => setTimeout(resolve, 2000)); + + // Check if still running + try { + execSync(`kill -0 ${pid}`, { encoding: 'utf8' }); + // Still running, try SIGKILL + execSync(`kill -9 ${pid}`, { encoding: 'utf8' }); + log.info(`✅ Force killed ClaraCore process (PID: ${pid})`); + } catch { + // Process already dead, that's good + log.info(`✅ ClaraCore process (PID: ${pid}) terminated`); + } + + await new Promise(resolve => setTimeout(resolve, 1000)); + return true; + } catch (killError) { + if (killError.message.includes('Permission denied')) { + log.error(`❌ Permission denied: Cannot kill ClaraCore process ${pid}`); + log.error(` The process may be owned by another user or started with sudo.`); + log.error(` Please manually stop ClaraCore or run: sudo kill ${pid}`); + + // Don't crash the app - return false to indicate failure + return false; + } else { + log.warn(`Failed to kill process ${pid}:`, killError.message); + } + } + } + } + } catch (lsofError) { + log.debug('No process found on port 8091'); + } + } + + return false; + } catch (error) { + log.warn('Error while trying to kill existing ClaraCore:', error.message); + // Don't crash the app - return false to indicate failure + return false; + } + } + + /** + * Start the ClaraCore service + */ + async start() { + if (this.isRunning) { + log.warn('ClaraCore service is already running'); + return; + } + + if (this.isStarting) { + log.warn('ClaraCore service is already starting, skipping duplicate start attempt'); + return; + } + + this.isStarting = true; + this.isIntentionallyStopped = false; + + // Check if port 8091 is already in use + const portInUse = await this.isPortInUse(8091); + if (portInUse) { + log.warn('⚠️ Port 8091 is already in use. Checking if it\'s ClaraCore...'); + + // Check if it's ClaraCore running on the port + const isClaraCore = await this.isClaraCoreRunningOnPort(); + + if (isClaraCore) { + log.info('🔄 Detected existing ClaraCore instance. Attempting to kill and restart...'); + const killed = await this.killExistingClaraCore(); + + if (!killed) { + throw new Error('❌ ClaraCore is already running on port 8091, but failed to kill the existing instance. Please manually stop it and try again.'); + } + + // Double-check the port is now free + const stillInUse = await this.isPortInUse(8091); + if (stillInUse) { + throw new Error('❌ Port 8091 is still in use after killing the process. Please wait a moment and try again.'); + } + + log.info('✅ Successfully killed existing ClaraCore instance. Starting new instance...'); + } else { + // Not ClaraCore - some other service is using the port + throw new Error('❌ ClaraCore cannot start: Port 8091 is already in use by another service. Please free up the port and try again.'); + } + } + + try { + const binaryPath = this.getBinaryPath(); + + // Ensure config directory exists + if (!fs.existsSync(this.configDir)) { + fs.mkdirSync(this.configDir, { recursive: true }); + log.info(`Created ClaraCore config directory: ${this.configDir}`); + } + + // Create downloads directory in the writable location + const downloadsDir = path.join(this.configDir, 'downloads'); + if (!fs.existsSync(downloadsDir)) { + fs.mkdirSync(downloadsDir, { recursive: true }); + log.info(`Created ClaraCore downloads directory: ${downloadsDir}`); + } + + const args = ['-listen', ':8091', '-config', this.configPath]; + + log.info(`Starting ClaraCore service: ${binaryPath} ${args.join(' ')}`); + log.info(`ClaraCore working directory: ${this.configDir}`); + + // Ensure binary has execute permissions on Unix-like systems + if (os.platform() !== 'win32') { + try { + fs.chmodSync(binaryPath, '755'); + log.info('Set execute permissions on ClaraCore binary'); + } catch (chmodError) { + log.warn('Failed to set execute permissions:', chmodError); + } + } + + // Spawn the ClaraCore process with the writable config path + // Set cwd to our writable directory so downloads go there + this.process = spawn(binaryPath, args, { + stdio: ['ignore', 'pipe', 'pipe'], + detached: false, + windowsHide: true, + cwd: this.configDir // Set working directory to writable location + }); + + this.isRunning = true; + this.startTime = Date.now(); + this.restartAttempts = 0; + + // Handle process events + this.process.on('spawn', () => { + log.info('ClaraCore service spawned successfully'); + log.info(`ClaraCore PID: ${this.process.pid}`); + }); + + this.process.on('error', (error) => { + log.error('ClaraCore service error:', error); + this.isRunning = false; + this.process = null; + }); + + this.process.on('exit', (code, signal) => { + log.info(`ClaraCore service exited with code ${code}, signal ${signal}`); + this.isRunning = false; + this.process = null; + this.isStarting = false; + + // Don't auto-restart if it was intentionally stopped + if (this.isIntentionallyStopped) { + log.info('ClaraCore was intentionally stopped, skipping auto-restart'); + return; + } + + // Check if the error was due to port binding - don't auto-restart in this case + const isPortBindingError = this.lastError && ( + this.lastError.includes('bind:') || + this.lastError.includes('address already in use') || + this.lastError.includes('Only one usage of each socket address') + ); + + if (isPortBindingError) { + log.error('❌ ClaraCore failed to start due to port 8091 being in use. Auto-restart disabled. Please manually stop any conflicting services.'); + this.restartAttempts = this.maxRestartAttempts; // Prevent further restart attempts + return; + } + + // Auto-restart if it wasn't a clean shutdown + if (code !== 0 && code !== null && this.restartAttempts < this.maxRestartAttempts) { + this.restartAttempts++; + log.info(`Auto-restarting ClaraCore service (attempt ${this.restartAttempts}/${this.maxRestartAttempts})`); + // Store the timer so it can be cleared if stop() is called + this.restartTimer = setTimeout(() => { + this.restartTimer = null; + this.start(); + }, 5000); // Wait 5 seconds before restart + } + }); + + // Handle stdout + this.process.stdout.on('data', (data) => { + const output = data.toString().trim(); + if (output) { + log.info(`[ClaraCore] ${output}`); + } + }); + + // Handle stderr + this.process.stderr.on('data', (data) => { + const output = data.toString().trim(); + if (output) { + log.error(`[ClaraCore Error] ${output}`); + // Track last error for detecting port binding issues + this.lastError = output; + } + }); + + // Wait a bit to ensure the service starts properly + await new Promise(resolve => setTimeout(resolve, 2000)); + + // Verify the service is healthy + const isHealthy = await this.checkHealth(); + if (!isHealthy) { + this.isStarting = false; + throw new Error('ClaraCore service failed health check after startup'); + } + + log.info('ClaraCore service started successfully and is healthy'); + this.isStarting = false; // Clear starting flag on success + + } catch (error) { + log.error('Failed to start ClaraCore service:', error); + this.isRunning = false; + this.process = null; + this.isStarting = false; // Clear starting flag on error + throw error; + } + } + + /** + * Stop the ClaraCore service + */ + async stop() { + // Always set intentionally stopped flag FIRST, before any checks + this.isIntentionallyStopped = true; + + // Clear any pending auto-restart timer + if (this.restartTimer) { + clearTimeout(this.restartTimer); + this.restartTimer = null; + log.info('Cleared pending auto-restart timer'); + } + + try { + log.info('Stopping ClaraCore service...'); + + // ALWAYS try to kill any process on port 8091, regardless of internal state + // This ensures we actually stop ClaraCore even if our state tracking is wrong + const portInUse = await this.isPortInUse(8091); + + if (!portInUse && !this.isRunning && !this.process) { + log.info('ClaraCore service is not running (port 8091 is free)'); + this.isRunning = false; + this.process = null; + return; + } + + if (portInUse) { + log.info('Port 8091 is in use, attempting to kill process...'); + } + + // On Windows, SIGTERM doesn't work reliably, so force kill + if (os.platform() === 'win32') { + const killed = await this.killExistingClaraCore(); + + // Wait a moment for the process to fully terminate + await new Promise(resolve => setTimeout(resolve, 500)); + + // Verify the port is actually free now + const stillRunning = await this.isPortInUse(8091); + + if (stillRunning) { + log.warn('⚠️ Port 8091 still in use after first kill attempt, retrying...'); + // Try one more time + await this.killExistingClaraCore(); + await new Promise(resolve => setTimeout(resolve, 1000)); + + // Final check + const finalCheck = await this.isPortInUse(8091); + + if (finalCheck) { + log.error('❌ Failed to free port 8091 even after multiple kill attempts'); + throw new Error('Failed to stop ClaraCore - port 8091 still in use'); + } + } + + this.isRunning = false; + this.process = null; + log.info('✅ ClaraCore service stopped successfully (verified port is free)'); + return; + } + + // For Unix systems or if killExistingClaraCore failed, try process.kill + return new Promise((resolve, reject) => { + const timeout = setTimeout(() => { + log.warn('ClaraCore service did not stop gracefully, force killing...'); + if (this.process) { + this.process.kill('SIGKILL'); + } + this.isRunning = false; + this.process = null; + resolve(); + }, 5000); // 5 second timeout + + if (this.process) { + this.process.on('exit', () => { + clearTimeout(timeout); + this.isRunning = false; + this.process = null; + log.info('ClaraCore service stopped successfully'); + resolve(); + }); + + // Send SIGTERM for graceful shutdown + this.process.kill('SIGTERM'); + } else { + clearTimeout(timeout); + this.isRunning = false; + this.process = null; + resolve(); + } + }); + + } catch (error) { + log.error('❌ Error stopping ClaraCore service:', error); + this.isRunning = false; + this.process = null; + throw error; + } + } + + /** + * Restart the ClaraCore service + */ + async restart() { + log.info('Restarting ClaraCore service...'); + await this.stop(); + await new Promise(resolve => setTimeout(resolve, 2000)); // Wait 2 seconds + await this.start(); + } + + /** + * Check if the service is healthy + */ + async checkHealth() { + const http = require('http'); + + return new Promise((resolve) => { + const req = http.get('http://localhost:8091/health', (res) => { + resolve(res.statusCode === 200); + }); + + req.on('error', () => { + resolve(false); + }); + + req.setTimeout(3000, () => { + req.destroy(); + resolve(false); + }); + }); + } + + /** + * Get service status + */ + getStatus() { + return { + isRunning: this.isRunning, + pid: this.process?.pid || null, + uptime: this.startTime ? Date.now() - this.startTime : 0, + restartAttempts: this.restartAttempts + }; + } +} + +module.exports = ClaraCoreService; diff --git a/electron/claracore/API.md b/electron/claracore/API.md new file mode 100644 index 00000000..2b7968f0 --- /dev/null +++ b/electron/claracore/API.md @@ -0,0 +1,1484 @@ +# ClaraCore Complete API Reference + +## Overview + +ClaraCore provides a comprehensive HTTP API for managing AI models, system configuration, and OpenAI-compatible endpoints. The API is designed for both programmatic access and integration with the React UI. + +**Base URLs:** +- Local UI: `http://localhost:5800/ui/` +- API Base: `http://localhost:5800/api` +- OpenAI Base: `http://localhost:5800/v1` +- Health Check: `http://localhost:5800/health` + +**Default Port:** 5800 (configurable via `config.yaml`) + +--- + +## Table of Contents + +1. [Authentication](#authentication) +2. [OpenAI-Compatible Endpoints](#openai-compatible-endpoints) +3. [System Management](#system-management) +4. [Model Management](#model-management) +5. [Configuration Management](#configuration-management) +6. [Download Management](#download-management) +7. [Monitoring & Events](#monitoring--events) +8. [Binary Management](#binary-management) +9. [Error Handling](#error-handling) +10. [Examples & Use Cases](#examples--use-cases) + +--- + +## Authentication + +ClaraCore supports optional API key authentication for all endpoints except system settings configuration. + +### Headers +```http +Authorization: Bearer +# OR +X-API-Key: +# OR (for EventSource/limited clients) +?api_key= +``` + +### Configure API Key +```bash +curl -X POST http://localhost:5800/api/settings/system \ + -H 'Content-Type: application/json' \ + -d '{ + "requireApiKey": true, + "apiKey": "your-secret-key-here" + }' +``` + +--- + +## OpenAI-Compatible Endpoints + +ClaraCore provides full OpenAI API compatibility for seamless integration with existing tools and applications. + +### Chat Completions + +**Endpoint:** `POST /v1/chat/completions` + +**Request:** +```bash +curl -X POST http://localhost:5800/v1/chat/completions \ + -H 'Content-Type: application/json' \ + -H 'Authorization: Bearer your-api-key' \ + -d '{ + "model": "llama-3.2-3b-instruct", + "messages": [ + { + "role": "user", + "content": "Explain quantum computing in simple terms" + } + ], + "temperature": 0.7, + "max_tokens": 150, + "stream": false + }' +``` + +**Response:** +```json +{ + "id": "chatcmpl-abc123", + "object": "chat.completion", + "created": 1699999999, + "model": "llama-3.2-3b-instruct", + "choices": [ + { + "index": 0, + "message": { + "role": "assistant", + "content": "Quantum computing is like having a super-powered calculator..." + }, + "finish_reason": "stop" + } + ], + "usage": { + "prompt_tokens": 15, + "completion_tokens": 45, + "total_tokens": 60 + } +} +``` + +### Streaming Chat Completions + +**Request:** +```bash +curl -X POST http://localhost:5800/v1/chat/completions \ + -H 'Content-Type: application/json' \ + -H 'Authorization: Bearer your-api-key' \ + -d '{ + "model": "llama-3.2-3b-instruct", + "messages": [ + { + "role": "user", + "content": "Write a haiku about programming" + } + ], + "stream": true + }' +``` + +**Response:** (Server-Sent Events) +``` +data: {"id":"chatcmpl-123","object":"chat.completion.chunk","created":1699999999,"model":"llama-3.2-3b-instruct","choices":[{"index":0,"delta":{"role":"assistant"},"finish_reason":null}]} + +data: {"id":"chatcmpl-123","object":"chat.completion.chunk","created":1699999999,"model":"llama-3.2-3b-instruct","choices":[{"index":0,"delta":{"content":"Code"},"finish_reason":null}]} + +data: {"id":"chatcmpl-123","object":"chat.completion.chunk","created":1699999999,"model":"llama-3.2-3b-instruct","choices":[{"index":0,"delta":{"content":" flows"},"finish_reason":null}]} + +data: [DONE] +``` + +### Text Completions + +**Endpoint:** `POST /v1/completions` + +**Request:** +```bash +curl -X POST http://localhost:5800/v1/completions \ + -H 'Content-Type: application/json' \ + -H 'Authorization: Bearer your-api-key' \ + -d '{ + "model": "codellama-7b", + "prompt": "def fibonacci(n):", + "max_tokens": 100, + "temperature": 0.2 + }' +``` + +### Embeddings + +**Endpoint:** `POST /v1/embeddings` + +**Request:** +```bash +curl -X POST http://localhost:5800/v1/embeddings \ + -H 'Content-Type: application/json' \ + -H 'Authorization: Bearer your-api-key' \ + -d '{ + "model": "nomic-embed-text-v1.5", + "input": ["Hello world", "How are you?"] + }' +``` + +**Response:** +```json +{ + "object": "list", + "data": [ + { + "object": "embedding", + "embedding": [0.1234, -0.5678, ...], + "index": 0 + }, + { + "object": "embedding", + "embedding": [0.9876, -0.1234, ...], + "index": 1 + } + ], + "model": "nomic-embed-text-v1.5", + "usage": { + "prompt_tokens": 4, + "total_tokens": 4 + } +} +``` + +### List Models + +**Endpoint:** `GET /v1/models` + +**Request:** +```bash +curl -X GET http://localhost:5800/v1/models \ + -H 'Authorization: Bearer your-api-key' +``` + +**Response:** +```json +{ + "object": "list", + "data": [ + { + "id": "llama-3.2-3b-instruct", + "object": "model", + "created": 1699999999, + "owned_by": "claracore", + "permission": [], + "root": "llama-3.2-3b-instruct", + "parent": null + }, + { + "id": "nomic-embed-text-v1.5", + "object": "model", + "created": 1699999999, + "owned_by": "claracore", + "permission": [], + "root": "nomic-embed-text-v1.5", + "parent": null + } + ] +} +``` + +### Audio Endpoints + +#### Text-to-Speech +**Endpoint:** `POST /v1/audio/speech` + +```bash +curl -X POST http://localhost:5800/v1/audio/speech \ + -H 'Content-Type: application/json' \ + -H 'Authorization: Bearer your-api-key' \ + -d '{ + "model": "tts-model", + "input": "Hello, this is a test.", + "voice": "alloy" + }' \ + --output speech.mp3 +``` + +#### Speech-to-Text +**Endpoint:** `POST /v1/audio/transcriptions` + +```bash +curl -X POST http://localhost:5800/v1/audio/transcriptions \ + -H 'Authorization: Bearer your-api-key' \ + -F file=@audio.wav \ + -F model=whisper-1 +``` + +**Response:** +```json +{ + "text": "Hello, this is a test transcription." +} +``` + +### Reranking + +**Endpoints:** `POST /v1/rerank`, `POST /v1/reranking` + +```bash +curl -X POST http://localhost:5800/v1/rerank \ + -H 'Content-Type: application/json' \ + -H 'Authorization: Bearer your-api-key' \ + -d '{ + "model": "rerank-model", + "query": "What is machine learning?", + "documents": [ + "Machine learning is a subset of AI", + "Cooking recipes for beginners", + "Deep learning uses neural networks" + ] + }' +``` + +**Response:** +```json +{ + "object": "list", + "data": [ + { + "index": 0, + "relevance_score": 0.95, + "document": "Machine learning is a subset of AI" + }, + { + "index": 2, + "relevance_score": 0.78, + "document": "Deep learning uses neural networks" + }, + { + "index": 1, + "relevance_score": 0.05, + "document": "Cooking recipes for beginners" + } + ], + "model": "rerank-model", + "usage": { + "total_tokens": 25 + } +} +``` + +### Additional Endpoints + +**Code Infilling:** `POST /infill` +**Completion:** `POST /completion` + +--- + +## System Management + +### System Specifications + +**Endpoint:** `GET /api/system/specs` + +**Request:** +```bash +curl -X GET http://localhost:5800/api/system/specs +``` + +**Response:** +```json +{ + "totalRAM": 34359738368, + "availableRAM": 25769803776, + "totalVRAM": 12884901888, + "availableVRAM": 10737418240, + "cpuCores": 16, + "gpuName": "NVIDIA RTX 4070", + "diskSpace": 500000000000 +} +``` + +### System Detection (Comprehensive) + +**Endpoint:** `GET /api/system/detection` + +**Request:** +```bash +curl -X GET http://localhost:5800/api/system/detection +``` + +**Response:** +```json +{ + "detectionQuality": "excellent", + "platform": "windows", + "arch": "amd64", + "gpuDetected": true, + "gpuTypes": ["NVIDIA (RTX, GTX)", "CPU Only"], + "primaryGPU": { + "name": "NVIDIA GeForce RTX 4070", + "brand": "nvidia", + "vramGB": 12.0 + }, + "totalRAMGB": 32.0, + "availableRAMGB": 24.0, + "recommendedBackends": ["cuda", "vulkan", "cpu"], + "supportedBackends": ["cuda", "vulkan", "cpu"], + "recommendedContextSizes": [8192, 16384, 32768, 65536, 131072], + "maxRecommendedContextSize": 131072, + "recommendations": { + "primaryBackend": "cuda", + "fallbackBackend": "cpu", + "suggestedContextSize": 65536, + "suggestedVRAMAllocation": 9, + "suggestedRAMAllocation": 16, + "throughputFirst": true, + "notes": [ + "Detected cuda with 12.0GB VRAM", + "Recommended context size: 65536 tokens", + "Performance priority: Speed (Higher throughput)" + ] + }, + "detectionTimestamp": "2024-01-01T12:00:00Z" +} +``` + +### System Settings + +#### Get Settings +**Endpoint:** `GET /api/settings/system` + +```bash +curl -X GET http://localhost:5800/api/settings/system +``` + +**Response:** +```json +{ + "settings": { + "gpuType": "nvidia", + "backend": "cuda", + "vramGB": 12.0, + "ramGB": 32.0, + "preferredContext": 65536, + "throughputFirst": true, + "enableJinja": true, + "requireApiKey": false + } +} +``` + +#### Save Settings +**Endpoint:** `POST /api/settings/system` + +```bash +curl -X POST http://localhost:5800/api/settings/system \ + -H 'Content-Type: application/json' \ + -d '{ + "gpuType": "nvidia", + "backend": "cuda", + "vramGB": 12.0, + "ramGB": 32.0, + "preferredContext": 65536, + "throughputFirst": true, + "enableJinja": true, + "requireApiKey": true, + "apiKey": "your-secret-key" + }' +``` + +### Server Management + +#### Soft Restart +**Endpoint:** `POST /api/server/restart` + +Reloads configuration and restarts model processes without killing the main server. + +```bash +curl -X POST http://localhost:5800/api/server/restart +``` + +**Response:** +```json +{ + "message": "Soft restart initiated - reloading config and restarting models", + "status": "restarting" +} +``` + +#### Hard Restart +**Endpoint:** `POST /api/server/restart/hard` + +Spawns a new server process and exits the current one. + +```bash +curl -X POST http://localhost:5800/api/server/restart/hard +``` + +**Response:** +```json +{ + "message": "Hard restart initiated - spawning new process", + "status": "restarting" +} +``` + +--- + +## Model Management + +### Model Status + +Access real-time model status through the events endpoint. Models can be in various states: + +**States:** `ready`, `starting`, `stopping`, `shutdown`, `stopped`, `unknown` + +**Example Model Status:** +```json +[ + { + "id": "llama-3.2-3b-instruct", + "name": "Llama 3.2 3B Instruct", + "description": "Meta's Llama 3.2 3B instruction-tuned model", + "state": "ready", + "unlisted": false, + "proxyUrl": "http://127.0.0.1:8200" + }, + { + "id": "nomic-embed-text-v1.5", + "name": "Nomic Embed Text v1.5", + "description": "Nomic's embedding model", + "state": "starting", + "unlisted": false, + "proxyUrl": "http://127.0.0.1:8201" + } +] +``` + +### Unload All Models + +**Endpoint:** `POST /api/models/unload` + +```bash +curl -X POST http://localhost:5800/api/models/unload +``` + +**Response:** +```json +{ + "msg": "ok" +} +``` + +--- + +## Download Management + +### Model Downloads + +#### Start Download +**Endpoint:** `POST /api/models/download` + +```bash +curl -X POST http://localhost:5800/api/models/download \ + -H 'Content-Type: application/json' \ + -d '{ + "url": "https://huggingface.co/microsoft/Phi-3.5-mini-instruct-GGUF/resolve/main/Phi-3.5-mini-instruct-Q4_K_M.gguf", + "modelId": "phi-3.5-mini-instruct", + "filename": "phi-3.5-mini-instruct-q4-k-m.gguf", + "hfApiKey": "hf_your_token_here" + }' +``` + +**Response:** +```json +{ + "downloadId": "download_abc123", + "status": "download started", + "modelId": "phi-3.5-mini-instruct", + "filename": "phi-3.5-mini-instruct-q4-k-m.gguf" +} +``` + +#### List Downloads +**Endpoint:** `GET /api/models/downloads` + +```bash +curl -X GET http://localhost:5800/api/models/downloads +``` + +**Response:** +```json +{ + "download_abc123": { + "id": "download_abc123", + "modelId": "phi-3.5-mini-instruct", + "filename": "phi-3.5-mini-instruct-q4-k-m.gguf", + "url": "https://huggingface.co/...", + "status": "downloading", + "progress": 45.2, + "downloadedBytes": 1234567890, + "totalBytes": 2730000000, + "speed": "15.2 MB/s", + "eta": "2m 15s" + } +} +``` + +#### Get Download Status +**Endpoint:** `GET /api/models/downloads/:id` + +```bash +curl -X GET http://localhost:5800/api/models/downloads/download_abc123 +``` + +#### Pause Download +**Endpoint:** `POST /api/models/downloads/:id/pause` + +```bash +curl -X POST http://localhost:5800/api/models/downloads/download_abc123/pause +``` + +#### Resume Download +**Endpoint:** `POST /api/models/downloads/:id/resume` + +```bash +curl -X POST http://localhost:5800/api/models/downloads/download_abc123/resume +``` + +#### Cancel Download +**Endpoint:** `POST /api/models/download/cancel` + +```bash +curl -X POST http://localhost:5800/api/models/download/cancel \ + -H 'Content-Type: application/json' \ + -d '{ + "downloadId": "download_abc123" + }' +``` + +### HuggingFace API Key Management + +#### Get HF API Key Status +**Endpoint:** `GET /api/settings/hf-api-key` + +```bash +curl -X GET http://localhost:5800/api/settings/hf-api-key +``` + +#### Set HF API Key +**Endpoint:** `POST /api/settings/hf-api-key` + +```bash +curl -X POST http://localhost:5800/api/settings/hf-api-key \ + -H 'Content-Type: application/json' \ + -d '{ + "apiKey": "hf_your_token_here" + }' +``` + +--- + +## Configuration Management + +### Get Current Configuration + +**Endpoint:** `GET /api/config` + +```bash +curl -X GET http://localhost:5800/api/config +``` + +**Response:** +```json +{ + "yaml": "healthCheckTimeout: 300\nlogLevel: info\n...", + "config": { + "healthCheckTimeout": 300, + "logLevel": "info", + "startPort": 8100, + "downloadDir": "./downloads", + "models": { + "llama-3.2-3b": { + "name": "Llama 3.2 3B", + "cmd": "...", + "proxy": "http://127.0.0.1:${PORT}" + } + }, + "groups": { + "large-models": { + "exclusive": true, + "members": ["llama-3.2-3b"], + "startPort": 8200 + } + } + } +} +``` + +### Update Configuration + +**Endpoint:** `POST /api/config` + +```bash +curl -X POST http://localhost:5800/api/config \ + -H 'Content-Type: application/json' \ + -d '{ + "yaml": "healthCheckTimeout: 300\nlogLevel: debug\n..." + }' +``` + +### Scan Model Folders + +**Endpoint:** `POST /api/config/scan-folder` + +Scan folders for GGUF models with intelligent detection. + +```bash +curl -X POST http://localhost:5800/api/config/scan-folder \ + -H 'Content-Type: application/json' \ + -d '{ + "folderPaths": [ + "C:\\AI\\Models\\Llama", + "D:\\HuggingFace\\Models" + ], + "recursive": true, + "addToDatabase": true + }' +``` + +**Response:** +```json +{ + "models": [ + { + "modelId": "llama-3.2-3b-instruct", + "filename": "llama-3.2-3b-instruct-q4-k-m.gguf", + "name": "Llama 3.2 3B Instruct", + "size": 2100000000, + "sizeFormatted": "2.1GB", + "path": "C:\\AI\\Models\\Llama\\llama-3.2-3b-instruct-q4-k-m.gguf", + "relativePath": "llama-3.2-3b-instruct-q4-k-m.gguf", + "quantization": "Q4_K_M", + "isInstruct": true, + "isDraft": false, + "isEmbedding": false, + "contextLength": 131072, + "numLayers": 28, + "isMoE": false + } + ], + "scanSummary": [ + { + "folder": "C:\\AI\\Models\\Llama", + "status": "success", + "models": 5 + } + ], + "totalModels": 5, + "foldersScanned": 1 +} +``` + +### Add Single Model + +**Endpoint:** `POST /api/config/append-model` + +Add a single model to existing configuration with smart parameter detection. + +```bash +curl -X POST http://localhost:5800/api/config/append-model \ + -H 'Content-Type: application/json' \ + -d '{ + "filePath": "C:\\AI\\Models\\phi-3.5-mini-instruct-q4-k-m.gguf", + "options": { + "enableJinja": true, + "throughputFirst": true, + "minContext": 16384, + "preferredContext": 32768 + } + }' +``` + +**Response:** +```json +{ + "status": "Model successfully appended to config.yaml", + "modelId": "phi-3.5-mini-instruct", + "modelInfo": { + "name": "Phi 3.5 Mini Instruct", + "size": "2.4GB", + "quantization": "Q4_K_M", + "isInstruct": true, + "isEmbedding": false, + "contextLength": 131072 + }, + "requiresRestart": true, + "restartMessage": "New model has been added to configuration. Would you like to restart the server to apply changes?" +} +``` + +### Update Model Parameters + +**Endpoint:** `POST /api/config/model/:id` + +Update specific parameters for a model without destroying the configuration structure. + +```bash +curl -X POST http://localhost:5800/api/config/model/llama-3.2-3b \ + -H 'Content-Type: application/json' \ + -d '{ + "contextSize": 65536, + "layers": 999, + "cacheType": "q4_0", + "batchSize": 2048 + }' +``` + +**Response:** +```json +{ + "status": "Model parameters updated successfully", + "model": "llama-3.2-3b", + "backup": "config.yaml.backup.1699999999", + "updated": { + "contextSize": 65536, + "layers": 999, + "cacheType": "q4_0", + "batchSize": 2048 + }, + "requiresRestart": true, + "restartMessage": "Model configuration has been updated. Would you like to restart the server to apply changes?" +} +``` + +### Model Folders Database + +#### Get Tracked Folders +**Endpoint:** `GET /api/config/folders` + +```bash +curl -X GET http://localhost:5800/api/config/folders +``` + +**Response:** +```json +{ + "folders": [ + { + "path": "C:\\AI\\Models", + "addedAt": "2024-01-01T12:00:00Z", + "lastScanned": "2024-01-01T13:00:00Z", + "modelCount": 12, + "recursive": true, + "enabled": true + } + ], + "lastScan": "2024-01-01T13:00:00Z", + "version": "1.0", + "totalCount": 1 +} +``` + +#### Add Folders to Database +**Endpoint:** `POST /api/config/folders` + +```bash +curl -X POST http://localhost:5800/api/config/folders \ + -H 'Content-Type: application/json' \ + -d '{ + "folderPaths": [ + "C:\\AI\\Models\\New", + "D:\\External\\Models" + ], + "recursive": true + }' +``` + +#### Remove Folders from Database +**Endpoint:** `DELETE /api/config/folders` + +```bash +curl -X DELETE http://localhost:5800/api/config/folders \ + -H 'Content-Type: application/json' \ + -d '{ + "folderPaths": [ + "C:\\AI\\Models\\Old" + ] + }' +``` + +### Regenerate Configuration from Database + +**Endpoint:** `POST /api/config/regenerate-from-db` + +Regenerate entire configuration using the same logic as CLI autosetup. + +```bash +curl -X POST http://localhost:5800/api/config/regenerate-from-db \ + -H 'Content-Type: application/json' \ + -d '{ + "options": { + "enableJinja": true, + "throughputFirst": true, + "minContext": 16384, + "preferredContext": 65536, + "forceBackend": "cuda", + "forceVRAM": 10.0, + "forceRAM": 24.0 + } + }' +``` + +**Response:** +```json +{ + "status": "Configuration regenerated using CLI autosetup function", + "totalModels": 15, + "foldersScanned": 3, + "scanSummary": [ + { + "folder": "C:\\AI\\Models", + "status": "success", + "models": 10 + } + ], + "config": "healthCheckTimeout: 300\n...", + "source": "autosetup.AutoSetupWithOptions() - identical to CLI", + "primaryFolder": "C:\\AI\\Models", + "note": "Using same function as CLI for guaranteed consistency", + "autoRestart": "Soft restart triggered automatically" +} +``` + +### Smart Generation + +**Endpoint:** `POST /api/config/generate-all` + +Intelligently generate configuration for all models in tracked folders with system optimization. + +```bash +curl -X POST http://localhost:5800/api/config/generate-all \ + -H 'Content-Type: application/json' \ + -d '{ + "folderPath": "C:\\AI\\Models", + "options": { + "enableJinja": true, + "throughputFirst": true, + "minContext": 16384, + "preferredContext": 32768, + "forceBackend": "cuda", + "forceVRAM": 10.0, + "forceRAM": 24.0 + } + }' +``` + +### Configuration Utilities + +#### Validate Configuration +**Endpoint:** `GET /api/config/validate` + +```bash +curl -X POST http://localhost:5800/api/config/validate \ + -H 'Content-Type: application/json' \ + -d '{ + "yaml": "healthCheckTimeout: 300\n..." + }' +``` + +**Response:** +```json +{ + "valid": true, + "modelCount": 5, + "groupCount": 2, + "macroCount": 2, + "startPort": 8100, + "downloadDir": "./downloads" +} +``` + +#### Validate Models on Disk +**Endpoint:** `POST /api/config/validate-models` + +Remove models from configuration if their files no longer exist. + +```bash +curl -X POST http://localhost:5800/api/config/validate-models +``` + +**Response:** +```json +{ + "status": "Config validation completed", + "removedModels": [ + "missing-model-1 (C:\\Path\\To\\missing.gguf)", + "missing-model-2 (D:\\Path\\To\\deleted.gguf)" + ], + "message": "Removed 2 missing models from config" +} +``` + +#### Cleanup Duplicates +**Endpoint:** `POST /api/config/cleanup-duplicates` + +Remove duplicate models that point to the same file. + +```bash +curl -X POST http://localhost:5800/api/config/cleanup-duplicates +``` + +**Response:** +```json +{ + "message": "Cleanup completed. Removed 3 duplicate models.", + "duplicatesRemoved": 3, + "removedModels": ["duplicate-1", "duplicate-2", "duplicate-3"], + "keptModels": ["original-model"] +} +``` + +--- + +## Monitoring & Events + +### Real-time Events (Server-Sent Events) + +**Endpoint:** `GET /api/events` + +Subscribe to real-time server events including model status, logs, metrics, and download progress. + +```javascript +const eventSource = new EventSource('/api/events'); + +eventSource.onmessage = function(event) { + const envelope = JSON.parse(event.data); + + switch(envelope.type) { + case 'modelStatus': + const models = JSON.parse(envelope.data); + console.log('Models:', models); + break; + + case 'logData': + const logInfo = JSON.parse(envelope.data); + console.log(`[${logInfo.source}]:`, logInfo.data); + break; + + case 'metrics': + const metrics = JSON.parse(envelope.data); + console.log('Metrics:', metrics); + break; + + case 'downloadProgress': + const downloadInfo = JSON.parse(envelope.data); + console.log('Download progress:', downloadInfo); + break; + + case 'configProgress': + const configInfo = JSON.parse(envelope.data); + console.log('Config generation:', configInfo); + break; + } +}; +``` + +### Metrics + +**Endpoint:** `GET /api/metrics` + +Get current performance metrics in JSON format. + +```bash +curl -X GET http://localhost:5800/api/metrics +``` + +**Response:** +```json +[ + { + "modelId": "llama-3.2-3b", + "requestCount": 42, + "avgResponseTime": 1250, + "tokensPerSecond": 35.2, + "lastActivity": "2024-01-01T13:30:00Z", + "memoryUsage": 2100000000 + } +] +``` + +### Setup Progress + +**Endpoint:** `GET /api/setup/progress` + +Monitor configuration generation progress during model setup. + +```bash +curl -X GET http://localhost:5800/api/setup/progress +``` + +**Response:** +```json +{ + "status": "processing", + "current_step": "Analyzing models", + "progress": 65.5, + "total_models": 10, + "processed_models": 6, + "current_model": "llama-3.2-3b-instruct", + "error": null, + "completed": false, + "started_at": "2024-01-01T13:00:00Z", + "updated_at": "2024-01-01T13:02:30Z" +} +``` + +--- + +## Binary Management + +### Get Binary Status + +**Endpoint:** `GET /api/binary/status` + +Check the status of the llama-server binary. + +```bash +curl -X GET http://localhost:5800/api/binary/status +``` + +**Response:** +```json +{ + "exists": true, + "path": "binaries/llama-server/build/bin/llama-server.exe", + "hasMetadata": true, + "currentVersion": "b3990", + "currentType": "cuda", + "latestVersion": "b4000", + "optimalType": "cuda", + "isOptimal": true, + "isUpToDate": false, + "updateAvailable": true +} +``` + +### Update Binary + +**Endpoint:** `POST /api/binary/update` + +Update the llama-server binary to the latest version. + +```bash +curl -X POST http://localhost:5800/api/binary/update +``` + +**Response:** +```json +{ + "status": "updated", + "message": "Binary updated successfully", + "version": "b4000", + "type": "cuda", + "path": "binaries/llama-server/build/bin/llama-server.exe", + "wasForced": false +} +``` + +### Force Update Binary + +**Endpoint:** `POST /api/binary/update/force` + +Force update the binary even if it's already up-to-date. + +```bash +curl -X POST http://localhost:5800/api/binary/update/force +``` + +--- + +## Error Handling + +### Standard Error Response + +All API endpoints return consistent error responses: + +```json +{ + "error": "Descriptive error message", + "details": "Additional context if available", + "code": "ERROR_CODE" +} +``` + +### Common HTTP Status Codes + +- **200 OK** - Success +- **400 Bad Request** - Invalid request parameters +- **401 Unauthorized** - API key required or invalid +- **404 Not Found** - Resource not found +- **409 Conflict** - Resource already exists +- **500 Internal Server Error** - Server error + +### Error Examples + +```bash +# Missing required parameter +{ + "error": "folderPath is required" +} + +# Authentication required +{ + "error": "API key required or invalid" +} + +# Model already exists +{ + "error": "Model already exists in config with ID: llama-3.2-3b", + "existingModelId": "llama-3.2-3b", + "filePath": "C:\\Models\\llama.gguf" +} + +# File not found +{ + "error": "Model file not found: C:\\Models\\missing.gguf" +} +``` + +--- + +## Examples & Use Cases + +### Complete Model Setup Workflow + +```bash +# 1. Detect system capabilities +curl -X GET http://localhost:5800/api/system/detection + +# 2. Configure system settings (one-time) +curl -X POST http://localhost:5800/api/settings/system \ + -H 'Content-Type: application/json' \ + -d '{ + "gpuType": "nvidia", + "backend": "cuda", + "vramGB": 12.0, + "ramGB": 32.0, + "preferredContext": 65536, + "throughputFirst": true, + "enableJinja": true + }' + +# 3. Add model folders to database +curl -X POST http://localhost:5800/api/config/folders \ + -H 'Content-Type: application/json' \ + -d '{ + "folderPaths": ["C:\\AI\\Models"], + "recursive": true + }' + +# 4. Generate configuration from all tracked folders +curl -X POST http://localhost:5800/api/config/regenerate-from-db \ + -H 'Content-Type: application/json' \ + -d '{ + "options": { + "enableJinja": true, + "throughputFirst": true, + "preferredContext": 65536 + } + }' + +# 5. Monitor setup progress via SSE +# (Server automatically restarts when configuration is complete) + +# 6. Use OpenAI-compatible endpoints +curl -X POST http://localhost:5800/v1/chat/completions \ + -H 'Content-Type: application/json' \ + -d '{ + "model": "llama-3.2-3b-instruct", + "messages": [{"role": "user", "content": "Hello!"}] + }' +``` + +### Download and Auto-Configure Workflow + +```bash +# 1. Start model download +curl -X POST http://localhost:5800/api/models/download \ + -H 'Content-Type: application/json' \ + -d '{ + "url": "https://huggingface.co/microsoft/Phi-3.5-mini-instruct-GGUF/resolve/main/Phi-3.5-mini-instruct-Q4_K_M.gguf", + "modelId": "phi-3.5-mini", + "filename": "phi-3.5-mini-q4-k-m.gguf" + }' + +# 2. Monitor download progress via SSE +# (Backend automatically adds folder to database and regenerates config when complete) + +# 3. Model is automatically available for use +curl -X POST http://localhost:5800/v1/chat/completions \ + -H 'Content-Type: application/json' \ + -d '{ + "model": "phi-3.5-mini", + "messages": [{"role": "user", "content": "Hello!"}] + }' +``` + +### Configuration Management + +```bash +# Add a single model to existing setup +curl -X POST http://localhost:5800/api/config/append-model \ + -H 'Content-Type: application/json' \ + -d '{ + "filePath": "C:\\Models\\new-model.gguf", + "options": {"enableJinja": true} + }' + +# Update model parameters +curl -X POST http://localhost:5800/api/config/model/new-model \ + -H 'Content-Type: application/json' \ + -d '{ + "contextSize": 32768, + "layers": 999, + "batchSize": 1024 + }' + +# Soft restart to apply changes +curl -X POST http://localhost:5800/api/server/restart +``` + +### Monitoring and Maintenance + +```bash +# Check system status +curl -X GET http://localhost:5800/api/system/specs +curl -X GET http://localhost:5800/api/metrics + +# Validate configuration +curl -X POST http://localhost:5800/api/config/validate-models +curl -X POST http://localhost:5800/api/config/cleanup-duplicates + +# Update binary +curl -X POST http://localhost:5800/api/binary/update +``` + +--- + +## Integration Examples + +### Python Client Example + +```python +import requests +import json +from sseclient import SSEClient + +class ClaraCoreClient: + def __init__(self, base_url="http://localhost:5800", api_key=None): + self.base_url = base_url + self.headers = {"Content-Type": "application/json"} + if api_key: + self.headers["Authorization"] = f"Bearer {api_key}" + + def chat_completion(self, model, messages, **kwargs): + """OpenAI-compatible chat completion""" + data = { + "model": model, + "messages": messages, + **kwargs + } + response = requests.post( + f"{self.base_url}/v1/chat/completions", + headers=self.headers, + json=data + ) + return response.json() + + def download_model(self, url, model_id, filename): + """Start model download""" + data = { + "url": url, + "modelId": model_id, + "filename": filename + } + response = requests.post( + f"{self.base_url}/api/models/download", + headers=self.headers, + json=data + ) + return response.json() + + def monitor_events(self): + """Monitor real-time events""" + url = f"{self.base_url}/api/events" + if "Authorization" in self.headers: + # Add API key as query param for SSE + api_key = self.headers["Authorization"].replace("Bearer ", "") + url += f"?api_key={api_key}" + + for event in SSEClient(url): + if event.data: + yield json.loads(event.data) + +# Usage +client = ClaraCoreClient(api_key="your-api-key") + +# Chat with model +response = client.chat_completion( + model="llama-3.2-3b-instruct", + messages=[{"role": "user", "content": "Hello!"}], + temperature=0.7 +) +print(response["choices"][0]["message"]["content"]) + +# Monitor events +for event in client.monitor_events(): + if event["type"] == "modelStatus": + models = json.loads(event["data"]) + print(f"Active models: {[m['id'] for m in models if m['state'] == 'ready']}") +``` + +### JavaScript/Node.js Example + +```javascript +const axios = require('axios'); +const EventSource = require('eventsource'); + +class ClaraCoreClient { + constructor(baseUrl = 'http://localhost:5800', apiKey = null) { + this.baseUrl = baseUrl; + this.headers = { 'Content-Type': 'application/json' }; + if (apiKey) { + this.headers['Authorization'] = `Bearer ${apiKey}`; + } + } + + async chatCompletion(model, messages, options = {}) { + const response = await axios.post(`${this.baseUrl}/v1/chat/completions`, { + model, + messages, + ...options + }, { headers: this.headers }); + + return response.data; + } + + async downloadModel(url, modelId, filename) { + const response = await axios.post(`${this.baseUrl}/api/models/download`, { + url, + modelId, + filename + }, { headers: this.headers }); + + return response.data; + } + + monitorEvents(callback) { + let url = `${this.baseUrl}/api/events`; + if (this.headers['Authorization']) { + const apiKey = this.headers['Authorization'].replace('Bearer ', ''); + url += `?api_key=${apiKey}`; + } + + const eventSource = new EventSource(url); + eventSource.onmessage = (event) => { + const data = JSON.parse(event.data); + callback(data); + }; + + return eventSource; + } +} + +// Usage +const client = new ClaraCoreClient('http://localhost:5800', 'your-api-key'); + +// Chat completion +client.chatCompletion('llama-3.2-3b-instruct', [ + { role: 'user', content: 'Explain quantum computing' } +], { temperature: 0.7 }).then(response => { + console.log(response.choices[0].message.content); +}); + +// Monitor events +const eventSource = client.monitorEvents((event) => { + if (event.type === 'downloadProgress') { + const progress = JSON.parse(event.data); + console.log(`Download progress: ${progress.info.progress}%`); + } +}); +``` + +--- + +## Rate Limiting and Best Practices + +### API Rate Limiting +- No built-in rate limiting (designed for local use) +- If exposing publicly, use a reverse proxy with rate limiting + +### Best Practices + +1. **Use System Detection** before first-time setup +2. **Save System Settings** to persist preferences across regenerations +3. **Monitor Events** for real-time status updates +4. **Validate Configurations** after manual edits +5. **Use Folder Database** for automatic model management +6. **Implement Proper Error Handling** in your applications + +### Performance Tips + +1. **Enable Jinja** for better prompt processing +2. **Use Throughput First** for speed over quality when appropriate +3. **Set Optimal Context Size** based on your use case and available VRAM +4. **Monitor Metrics** to optimize performance +5. **Use Model Groups** to manage resource allocation + +--- + +This comprehensive API documentation covers all available endpoints in ClaraCore. The API provides both OpenAI compatibility for easy integration and powerful configuration management for advanced users. \ No newline at end of file diff --git a/electron/claracore/SETUP.md b/electron/claracore/SETUP.md new file mode 100644 index 00000000..4199b050 --- /dev/null +++ b/electron/claracore/SETUP.md @@ -0,0 +1,617 @@ +# ClaraCore Setup Guide + +This guide covers installation, configuration, and getting started with ClaraCore. + +## 🚀 Quick Installation + +### Option 1: Automated Installation (Recommended) + +#### Linux and macOS +```bash +curl -fsSL https://raw.githubusercontent.com/claraverse-space/ClaraCore/main/scripts/install.sh | bash +``` + +**Note**: For containers, WSL, or systemd-less environments, see our [Container Setup Guide](CONTAINER_SETUP.md). + +Or download and run manually: +```bash +wget https://raw.githubusercontent.com/claraverse-space/ClaraCore/main/scripts/install.sh +chmod +x install.sh +./install.sh +``` + +#### Windows (PowerShell as Administrator) +```powershell +irm https://raw.githubusercontent.com/claraverse-space/ClaraCore/main/scripts/install.ps1 | iex +``` + +Or download and run manually: +```powershell +Invoke-WebRequest -Uri "https://raw.githubusercontent.com/claraverse-space/ClaraCore/main/scripts/install.ps1" -OutFile "install.ps1" +.\install.ps1 +``` + +**Installation Features:** +- Downloads latest release automatically +- Sets up system service for auto-start +- Creates default configuration +- Adds to system PATH +- Creates desktop shortcut (Windows) + +### Option 2: Manual Installation + +#### Download Binary + +**Windows:** +```powershell +# Download the latest release +curl -L -o claracore.exe https://github.com/badboysm890/ClaraCore/releases/latest/download/claracore-windows-amd64.exe + +# Or build from source +python build.py +``` + +**Linux/macOS:** +```bash +# Download the latest release +curl -L -o claracore https://github.com/badboysm890/ClaraCore/releases/latest/download/claracore-linux-amd64 +chmod +x claracore + +# Or build from source +go build -o claracore . +``` + +### 2. Quick Setup + +#### After Installation + +If you used the automated installer, ClaraCore is ready to use: + +1. **Start the service** (if not auto-started): + ```bash + # Linux/macOS + sudo systemctl start claracore + + # Windows + Start-Service ClaraCore + ``` + +2. **Configure models**: + ```bash + # Point to your models folder + claracore --models-folder /path/to/your/gguf/models + + # Or use the web interface + # Visit: http://localhost:5800/ui/setup + ``` + +#### Manual Setup + +**Automatic Setup (Recommended):** +```bash +# Point ClaraCore at your models folder - it does the rest! +./claracore --models-folder /path/to/your/gguf/models + +# For specific backend +./claracore --models-folder /path/to/models --backend vulkan +``` + +**Manual Setup:** +```bash +# 1. Start ClaraCore +./claracore + +# 2. Open web interface +# Visit: http://localhost:5800/ui/setup + +# 3. Follow the setup wizard +# - Add model folders +# - Select backend (CUDA/Vulkan/ROCm/Metal/CPU) +# - Configure system settings +# - Generate configuration +``` + +### 3. Verify Installation + +```bash +# Test version info (should show proper version as of v0.1.1+) +claracore --version + +# Check if models are loaded +curl http://localhost:5800/v1/models + +# Test chat completion +curl -X POST http://localhost:5800/v1/chat/completions \ + -H "Content-Type: application/json" \ + -d '{ + "model": "your-model-name", + "messages": [{"role": "user", "content": "Hello!"}], + "max_tokens": 100 + }' +``` + +--- + +## 🎯 Setup Options + +### Command Line Arguments + +```bash +./claracore [options] + +Options: + --models-folder string Path to GGUF models folder + --backend string Force backend (cuda/rocm/vulkan/metal/cpu) + --port int Server port (default: 5800) + --host string Server host (default: 127.0.0.1) + --config string Config file path (default: config.yaml) + --vram float Override VRAM detection (GB) + --ram float Override RAM detection (GB) + --context int Preferred context length + --help Show help message +``` + +### Environment Variables + +```bash +export CLARA_MODELS_FOLDER="/path/to/models" +export CLARA_BACKEND="vulkan" +export CLARA_PORT="5800" +export CLARA_VRAM="8.0" +export CLARA_RAM="16.0" +``` + +--- + +## 🖥️ Backend Selection + +ClaraCore automatically detects your hardware, but you can override: + +### NVIDIA GPUs +```bash +# CUDA (recommended for NVIDIA) +./claracore --models-folder /path/to/models --backend cuda + +# Vulkan (universal, good fallback) +./claracore --models-folder /path/to/models --backend vulkan +``` + +### AMD GPUs +```bash +# ROCm (Linux only, for AMD GPUs) +./claracore --models-folder /path/to/models --backend rocm + +# Vulkan (cross-platform, good for AMD) +./claracore --models-folder /path/to/models --backend vulkan +``` + +### Apple Silicon +```bash +# Metal (macOS M1/M2/M3) +./claracore --models-folder /path/to/models --backend metal +``` + +### CPU Only +```bash +# CPU fallback (slower but works everywhere) +./claracore --models-folder /path/to/models --backend cpu +``` + +--- + +## 📁 Model Organization + +### Supported Formats +- **GGUF files** (`.gguf`) - Primary format +- **Quantized models** (Q4_K_M, Q5_K_S, Q8_0, etc.) +- **Full precision** (F16, F32) + +### Folder Structure Examples + +**Simple Structure:** +``` +/home/user/models/ +├── llama-3.2-3b-instruct.Q4_K_M.gguf +├── mistral-7b-v0.3.Q5_K_S.gguf +└── phi-3.5-mini-instruct.Q4_K_M.gguf +``` + +**Organized Structure:** +``` +/home/user/models/ +├── llama/ +│ ├── llama-3.2-3b-instruct.Q4_K_M.gguf +│ └── llama-3.2-1b-instruct.Q4_K_M.gguf +├── mistral/ +│ └── mistral-7b-v0.3.Q5_K_S.gguf +└── microsoft/ + └── phi-3.5-mini-instruct.Q4_K_M.gguf +``` + +### Model Naming Conventions + +ClaraCore automatically extracts model information from filenames: + +- **Model Name**: `llama-3.2-3b-instruct` +- **Quantization**: `Q4_K_M`, `Q5_K_S`, `F16` +- **Type Detection**: `instruct`, `chat`, `base` +- **Draft Models**: Automatically pairs larger models with smaller ones for speculative decoding + +--- + +## ⚙️ Configuration + +### Web Interface Setup + +1. **Start ClaraCore**: `./claracore` +2. **Open Browser**: `http://localhost:5800/ui/setup` +3. **Follow Wizard**: + - **Step 1**: Add model folders + - **Step 2**: System detection + - **Step 3**: Backend and memory configuration + - **Step 4**: Generate and apply configuration + +### Manual Configuration + +**System Settings** (`settings.json`): +```json +{ + "gpuType": "nvidia", + "backend": "cuda", + "vramGB": 10.0, + "ramGB": 32.0, + "preferredContext": 8192, + "throughputFirst": true, + "enableJinja": true, + "requireApiKey": false +} +``` + +**Model Folders** (`model_folders.json`): +```json +{ + "folders": [ + { + "path": "/home/user/models", + "enabled": true, + "recursive": true, + "addedAt": "2025-01-01T12:00:00Z" + } + ] +} +``` + +--- + +## 🔧 Advanced Configuration + +### Performance Optimization + +**High Memory System (32GB+ RAM):** +```bash +./claracore --models-folder /path/to/models \ + --backend cuda \ + --context 32768 \ + --ram 32.0 +``` + +**Low Memory System (8GB RAM):** +```bash +./claracore --models-folder /path/to/models \ + --backend vulkan \ + --context 4096 \ + --ram 8.0 +``` + +### Multiple Model Folders + +```bash +# Add multiple folders via API +curl -X POST http://localhost:5800/api/config/folders \ + -H "Content-Type: application/json" \ + -d '{ + "folderPaths": [ + "/home/user/models/llama", + "/home/user/models/mistral", + "/mnt/storage/models" + ], + "recursive": true + }' +``` + +### Custom Binary Path + +```bash +# Use custom llama-server binary +./claracore --binary-path /custom/path/to/llama-server +``` + +--- + +## 🌐 Web Interface Features + +### Available Pages + +- **`/ui/setup`** - Initial setup wizard +- **`/ui/models`** - Model management and chat interface +- **`/ui/configuration`** - Edit model parameters and settings +- **`/ui/downloads`** - Download models from Hugging Face +- **`/ui/settings`** - System preferences +- **`/ui/activity`** - Logs and system activity + +### Key Features + +- **Real-time Progress**: Setup operations show live progress +- **Restart Prompts**: Automatic prompts when configuration changes +- **Hardware Detection**: Visual system information display +- **Model Chat**: Test models directly in the interface +- **Download Manager**: Queue and manage model downloads + +--- + +## 🔍 Troubleshooting + +### Windows Security Issues + +**Error: "An Application Control policy has blocked this file"** + +This is Windows security protection, not malware. ClaraCore is safe! Solutions: + +1. **Run the troubleshooter first:** + ```powershell + .\scripts\troubleshoot.ps1 -UnblockFile + ``` + +2. **Manual unblock**: + ```powershell + Unblock-File "$env:LOCALAPPDATA\ClaraCore\claracore.exe" + ``` + +3. **If still blocked, disable Windows Defender Application Control:** + - Open Windows Security (search "Windows Security" in Start menu) + - Go to "App & browser control" + - Click "Reputation-based protection settings" + - Turn OFF "Check apps and files" + - Re-enable after installation for security + +4. **Alternative - Build from source**: + ```powershell + git clone https://github.com/claraverse-space/ClaraCore.git + cd ClaraCore + python build.py + .\claracore.exe + ``` + +5. **Service issues**: + ```powershell + .\scripts\troubleshoot.ps1 -FixService + ``` + +3. **Run as Administrator**: + ```powershell + Start-Process -Verb RunAs -FilePath "$env:LOCALAPPDATA\ClaraCore\claracore.exe" + ``` + +4. **Add to Windows Defender exclusions**: + - Windows Security > Virus & threat protection > Exclusions + - Add folder: `%LOCALAPPDATA%\ClaraCore` + +### Common Issues + +**1. "claracore: command not found"** +```bash +# Quick fix - add to current session: +export PATH="$HOME/.local/bin:$PATH" + +# Automatic fix script: +curl -fsSL https://raw.githubusercontent.com/claraverse-space/ClaraCore/main/scripts/fix-path.sh | bash + +# Manual fix - restart terminal or run: +source ~/.bashrc + +# Verify it works: +claracore --version +``` + +**2. Models Not Detected** +```bash +# Check folder permissions +ls -la /path/to/models + +# Verify GGUF files exist +find /path/to/models -name "*.gguf" + +# Scan folder manually +curl -X POST http://localhost:5800/api/config/scan-folder \ + -H "Content-Type: application/json" \ + -d '{"folderPath": "/path/to/models", "recursive": true}' +``` + +**2. Backend Issues** +```bash +# Check system detection +curl http://localhost:5800/api/system/detection + +# Force different backend +./claracore --models-folder /path/to/models --backend cpu +``` + +**3. Memory Problems** +```bash +# Reduce context length +curl -X POST http://localhost:5800/api/settings/system \ + -H "Content-Type: application/json" \ + -d '{"preferredContext": 2048, "vramGB": 4.0}' +``` + +**4. Binary Download Failures** +```bash +# Check binary status +ls -la binaries/llama-server/ + +# Manual binary download +curl -X POST http://localhost:5800/api/models/download \ + -H "Content-Type: application/json" \ + -d '{"url": "https://github.com/ggml-org/llama.cpp/releases/latest"}' +``` + +### Debug Mode + +```bash +# Enable verbose logging +./claracore --debug --models-folder /path/to/models + +# Check logs in real-time +curl -N http://localhost:5800/api/events +``` + +### Reset Configuration + +```bash +# Backup current config +cp config.yaml config.yaml.backup + +# Reset to defaults +rm config.yaml settings.json model_folders.json + +# Restart and reconfigure +./claracore --models-folder /path/to/models +``` + +--- + +## 🔧 Service Management + +If you installed using the automated installer, ClaraCore runs as a system service. + +### Linux/macOS Service Commands + +**Using systemctl (Linux):** +```bash +# Check status +sudo systemctl status claracore + +# Start/stop/restart +sudo systemctl start claracore +sudo systemctl stop claracore +sudo systemctl restart claracore + +# Enable/disable auto-start +sudo systemctl enable claracore +sudo systemctl disable claracore + +# View logs +sudo journalctl -u claracore -f +``` + +**Using launchctl (macOS):** +```bash +# Check status +sudo launchctl list | grep claracore + +# Start/stop +sudo launchctl load /Library/LaunchDaemons/com.claracore.server.plist +sudo launchctl unload /Library/LaunchDaemons/com.claracore.server.plist + +# View logs +tail -f /var/log/system.log | grep claracore +``` + +**Cross-platform service script:** +```bash +# Download service management script +wget https://raw.githubusercontent.com/claraverse-space/ClaraCore/main/scripts/claracore-service.sh +chmod +x claracore-service.sh + +# Use the service script +sudo ./claracore-service.sh status +sudo ./claracore-service.sh start +sudo ./claracore-service.sh stop +sudo ./claracore-service.sh restart +sudo ./claracore-service.sh logs +``` + +### Windows Service Commands + +**Using PowerShell:** +```powershell +# Check status +Get-Service ClaraCore + +# Start/stop/restart +Start-Service ClaraCore +Stop-Service ClaraCore +Restart-Service ClaraCore + +# View logs +Get-EventLog -LogName Application -Source ClaraCore -Newest 50 +``` + +**Using Services Manager:** +1. Press `Win + R`, type `services.msc` +2. Find "ClaraCore AI Inference Server" +3. Right-click for options + +### Uninstallation + +**Linux/macOS:** +```bash +# Download uninstall script +wget https://raw.githubusercontent.com/claraverse-space/ClaraCore/main/scripts/uninstall.sh +chmod +x uninstall.sh + +# Uninstall (keeps config) +sudo ./uninstall.sh + +# Uninstall and remove config +sudo ./uninstall.sh --remove-config +``` + +**Windows:** +```powershell +# Download uninstall script +Invoke-WebRequest -Uri "https://raw.githubusercontent.com/claraverse-space/ClaraCore/main/scripts/uninstall.ps1" -OutFile "uninstall.ps1" + +# Uninstall (keeps config) +.\uninstall.ps1 + +# Uninstall and remove config +.\uninstall.ps1 -RemoveConfig + +# Force uninstall without prompts +.\uninstall.ps1 -RemoveConfig -Force +``` + +--- + +## 📊 Performance Tips + +### Model Selection +- **For Chat**: Use instruct/chat models (Llama, Mistral, Phi) +- **For Speed**: Q4_K_M quantization offers good speed/quality balance +- **For Quality**: Q8_0 or F16 for highest quality (slower) +- **For Memory**: Q3_K_S or Q4_0 for lower memory usage + +### Hardware Optimization +- **NVIDIA**: Use CUDA backend with high GPU layers +- **AMD**: Use ROCm (Linux) or Vulkan +- **Intel**: Use Vulkan or CPU backend +- **Apple**: Use Metal backend on M1/M2/M3 + +### Context Length +- **Interactive Chat**: 4096-8192 tokens +- **Document Analysis**: 16384-32768 tokens +- **Code Generation**: 8192-16384 tokens + +--- + +## 🔗 Next Steps + +1. **Explore the API**: Check out the [Complete API Documentation](API_COMPREHENSIVE.md) +2. **Join the Community**: [GitHub Discussions](https://github.com/badboysm890/ClaraCore/discussions) +3. **Report Issues**: [GitHub Issues](https://github.com/badboysm890/ClaraCore/issues) +4. **Contribute**: See [Contributing Guidelines](../CONTRIBUTING.md) + +--- + +**Need help?** Join our community or open an issue on GitHub! \ No newline at end of file diff --git a/electron/claracore/checksums.txt b/electron/claracore/checksums.txt new file mode 100644 index 00000000..ed68c5a1 --- /dev/null +++ b/electron/claracore/checksums.txt @@ -0,0 +1,7 @@ +# SHA256 Checksums for ClaraCore v0.4.0 + +019ab72d81cd3555530eafac47687619aa98cd431bfbe125c2636156a2897466 claracore-linux-amd64 +00528e02ea300c11f850ddcf89610a4474c7d1bbcddcfb727239941062b6919d claracore-linux-arm64 +463212db89d89b327ceb6fda11e16a1d1ab6e914b602e804086085eb42bdc708 claracore-darwin-amd64 +c86c77d6a15edda6b730e78af2ab2185e661f97f8f211be43e49d0557c87ab07 claracore-darwin-arm64 +7367af82484f64e4b94b91e567702042a136eb57536868a7f63bf23c984a1017 claracore-windows-amd64.exe diff --git a/electron/claracore/progress_state.json b/electron/claracore/progress_state.json new file mode 100644 index 00000000..a6e97f66 --- /dev/null +++ b/electron/claracore/progress_state.json @@ -0,0 +1,14 @@ +{ + "setup_progress": { + "status": "completed", + "current_step": "Saving configuration file...", + "progress": 100, + "total_models": 3, + "processed_models": 3, + "current_model": "nsfw-panda-7b.Q5_K_M", + "error": "", + "completed": true, + "started_at": "2025-11-09T13:05:56.3563889+05:30", + "updated_at": "2025-11-09T13:06:06.5683856+05:30" + } +} \ No newline at end of file diff --git a/electron/comfyUIModelService.cjs b/electron/comfyUIModelService.cjs new file mode 100644 index 00000000..168f06f1 --- /dev/null +++ b/electron/comfyUIModelService.cjs @@ -0,0 +1,859 @@ +const { EventEmitter } = require('events'); +const Docker = require('dockerode'); +const path = require('path'); +const fs = require('fs'); +const https = require('https'); +const http = require('http'); +const crypto = require('crypto'); +const { promisify } = require('util'); +const { pipeline } = require('stream'); +const pipelineAsync = promisify(pipeline); + +class ComfyUIModelService extends EventEmitter { + constructor() { + super(); + + console.log('🎨 Initializing ComfyUIModelService...'); + + this.docker = new Docker(); + this.containerName = 'clara_comfyui'; + + // Local download and persistent storage directories + const os = require('os'); + this.localDownloadDir = path.join(os.homedir(), '.clara', 'model-downloads'); + this.persistentModelDir = path.join(os.homedir(), '.clara', 'comfyui-data', 'models'); + + console.log(`📁 Download dir: ${this.localDownloadDir}`); + console.log(`📁 Persistent dir: ${this.persistentModelDir}`); + + // Ensure local directories exist + if (!fs.existsSync(this.localDownloadDir)) { + fs.mkdirSync(this.localDownloadDir, { recursive: true }); + console.log(`✅ Created download directory: ${this.localDownloadDir}`); + } + if (!fs.existsSync(this.persistentModelDir)) { + fs.mkdirSync(this.persistentModelDir, { recursive: true }); + console.log(`✅ Created persistent directory: ${this.persistentModelDir}`); + } + + // Model categories and their paths (both local and container) + this.modelPaths = { + checkpoints: { + local: path.join(this.persistentModelDir, 'checkpoints'), + container: '/app/ComfyUI/models/checkpoints', + download: path.join(this.localDownloadDir, 'checkpoints') + }, + loras: { + local: path.join(this.persistentModelDir, 'loras'), + container: '/app/ComfyUI/models/loras', + download: path.join(this.localDownloadDir, 'loras') + }, + vae: { + local: path.join(this.persistentModelDir, 'vae'), + container: '/app/ComfyUI/models/vae', + download: path.join(this.localDownloadDir, 'vae') + }, + controlnet: { + local: path.join(this.persistentModelDir, 'controlnet'), + container: '/app/ComfyUI/models/controlnet', + download: path.join(this.localDownloadDir, 'controlnet') + }, + upscale_models: { + local: path.join(this.persistentModelDir, 'upscale_models'), + container: '/app/ComfyUI/models/upscale_models', + download: path.join(this.localDownloadDir, 'upscale_models') + }, + embeddings: { + local: path.join(this.persistentModelDir, 'embeddings'), + container: '/app/ComfyUI/models/embeddings', + download: path.join(this.localDownloadDir, 'embeddings') + }, + clip_vision: { + local: path.join(this.persistentModelDir, 'clip_vision'), + container: '/app/ComfyUI/models/clip_vision', + download: path.join(this.localDownloadDir, 'clip_vision') + } + }; + + // Ensure all local directories exist + Object.values(this.modelPaths).forEach(paths => { + [paths.local, paths.download].forEach(dir => { + if (!fs.existsSync(dir)) { + fs.mkdirSync(dir, { recursive: true }); + console.log(`✅ Created model directory: ${dir}`); + } + }); + }); + + console.log('🎨 ComfyUIModelService initialized successfully'); + + // Popular model repositories + this.repositories = { + huggingface: 'https://huggingface.co', + civitai: 'https://civitai.com/api/download/models', + openart: 'https://openart.ai' + }; + + // Download queue and status tracking + this.downloadQueue = new Map(); + this.activeDownloads = new Map(); + this.transferQueue = new Map(); + + console.log('🚀 ComfyUI Model Service initialized with local download support'); + console.log('📁 Local download directory:', this.localDownloadDir); + console.log('💾 Persistent model directory:', this.persistentModelDir); + } + + /** + * Get ComfyUI container instance + */ + async getContainer() { + try { + const container = this.docker.getContainer(this.containerName); + const info = await container.inspect(); + + if (info.State.Status !== 'running') { + throw new Error('ComfyUI container is not running'); + } + + return container; + } catch (error) { + throw new Error(`Failed to get ComfyUI container: ${error.message}`); + } + } + + /** + * List models currently installed in ComfyUI container + */ + async listInstalledModels(category = 'checkpoints') { + try { + const container = await this.getContainer(); + const modelPath = this.modelPaths[category]; + + if (!modelPath) { + throw new Error(`Unknown model category: ${category}`); + } + + const exec = await container.exec({ + Cmd: ['find', modelPath, '-type', 'f', '-name', '*.safetensors', '-o', '-name', '*.ckpt', '-o', '-name', '*.pth', '-o', '-name', '*.bin'], + AttachStdout: true, + AttachStderr: true + }); + + const stream = await exec.start({ hijack: true, stdin: false }); + + return new Promise((resolve, reject) => { + let output = ''; + let error = ''; + + stream.on('data', (data) => { + const chunk = data.toString(); + if (chunk.includes('Error') || chunk.includes('error')) { + error += chunk; + } else { + output += chunk; + } + }); + + stream.on('end', () => { + if (error) { + reject(new Error(error)); + } else { + const files = output.trim().split('\n') + .filter(line => line.trim()) + .map(filePath => ({ + name: path.basename(filePath), + path: filePath, + category: category, + size: null // We could get size with additional exec if needed + })); + resolve(files); + } + }); + + stream.on('error', reject); + }); + } catch (error) { + throw new Error(`Failed to list models: ${error.message}`); + } + } + + /** + * Get container storage info + */ + async getStorageInfo() { + try { + const container = await this.getContainer(); + + const exec = await container.exec({ + Cmd: ['df', '-h', '/app/ComfyUI/models'], + AttachStdout: true, + AttachStderr: true + }); + + const stream = await exec.start({ hijack: true, stdin: false }); + + return new Promise((resolve, reject) => { + let output = ''; + + stream.on('data', (data) => { + output += data.toString(); + }); + + stream.on('end', () => { + const lines = output.trim().split('\n'); + if (lines.length > 1) { + const [filesystem, size, used, available, percent, mountpoint] = lines[1].split(/\s+/); + resolve({ + filesystem, + size, + used, + available, + percent, + mountpoint + }); + } else { + resolve({ error: 'Could not parse storage info' }); + } + }); + + stream.on('error', reject); + }); + } catch (error) { + throw new Error(`Failed to get storage info: ${error.message}`); + } + } + + /** + * Download model from URL to local storage, then transfer to container + */ + async downloadModel(url, filename, category = 'checkpoints', onProgress = null, redirectCount = 0, options = {}) { + try { + // Prevent infinite redirect loops + if (redirectCount > 10) { + throw new Error('Too many redirects (maximum 10 allowed)'); + } + + this.emit('download:start', { filename, category, url }); + + // Download to local directory first for better performance + const downloadPath = path.join(this.modelPaths[category].download, filename); + const fileStream = fs.createWriteStream(downloadPath); + + // Parse URL to determine protocol + const urlObj = new URL(url); + const client = urlObj.protocol === 'https:' ? https : http; + + console.log(`📥 Downloading ${filename} to local storage...`); + + // Prepare headers with API key if provided + const headers = { + 'User-Agent': 'Clara-AI-Assistant/1.0', + 'Accept': '*/*', + 'Accept-Encoding': 'gzip, deflate, br', + 'Connection': 'keep-alive' + }; + + // Add authentication headers based on source and API key + if (options.source === 'civitai' || url.includes('civitai.com')) { + headers['Referer'] = 'https://civitai.com/'; + if (options.apiKey) { + headers['Authorization'] = `Bearer ${options.apiKey}`; + console.log(`🔑 Using CivitAI API key for authentication`); + } + } else if (options.source === 'huggingface' || url.includes('huggingface.co')) { + if (options.apiKey) { + headers['Authorization'] = `Bearer ${options.apiKey}`; + console.log(`🔑 Using HuggingFace API key for authentication`); + } + } + + console.log(`🌐 Request headers prepared, API key provided: ${!!options.apiKey}`); + + return new Promise((resolve, reject) => { + const request = client.get(url, { headers }, (response) => { + // Handle all redirect status codes (301, 302, 307, 308) + if (response.statusCode === 301 || response.statusCode === 302 || + response.statusCode === 307 || response.statusCode === 308) { + const redirectUrl = response.headers.location; + if (!redirectUrl) { + reject(new Error(`Redirect response missing location header`)); + return; + } + console.log(`📍 Following redirect (${response.statusCode}): ${redirectUrl} (redirect ${redirectCount + 1}/10)`); + + // For redirects, check if we're being redirected to a CDN URL + // CDN URLs typically don't need authentication headers and can fail with them + const redirectUrlObj = new URL(redirectUrl); + const isCdnUrl = redirectUrlObj.hostname.includes('cloudflarestorage.com') || + redirectUrlObj.hostname.includes('amazonaws.com') || + redirectUrlObj.hostname.includes('s3.') || + redirectUrlObj.hostname.includes('cdn.') || + redirectUrlObj.hostname.includes('storage.googleapis.com'); + + // If redirecting to CDN, strip authentication headers + const redirectOptions = isCdnUrl ? {} : options; + if (isCdnUrl) { + console.log(`🌐 Detected CDN redirect, removing auth headers for: ${redirectUrlObj.hostname}`); + } + + // Recursively follow the redirect with incremented count + return this.downloadModel(redirectUrl, filename, category, onProgress, redirectCount + 1, redirectOptions) + .then(resolve) + .catch(reject); + } + + if (response.statusCode !== 200) { + reject(new Error(`HTTP ${response.statusCode}: ${response.statusMessage}`)); + return; + } + + const totalSize = parseInt(response.headers['content-length'], 10); + let downloadedSize = 0; + + response.on('data', (chunk) => { + downloadedSize += chunk.length; + + if (onProgress && totalSize) { + const progress = (downloadedSize / totalSize) * 100; + onProgress(progress, downloadedSize, totalSize); + this.emit('download:progress', { + filename, + progress, + downloadedSize, + totalSize, + speed: this.calculateSpeed(downloadedSize) + }); + } + }); + + response.pipe(fileStream); + + fileStream.on('finish', async () => { + try { + // Verify file integrity + const stats = fs.statSync(downloadPath); + if (stats.size === 0) { + throw new Error('Downloaded file is empty'); + } + + console.log(`✅ Download completed: ${filename} (${stats.size} bytes)`); + + // Move to persistent storage + const persistentPath = path.join(this.modelPaths[category].local, filename); + await this.moveFile(downloadPath, persistentPath); + + console.log(`📁 Moved to persistent storage: ${persistentPath}`); + + this.emit('download:complete', { + filename, + category, + localPath: persistentPath, + containerPath: path.join(this.modelPaths[category].container, filename), + size: stats.size + }); + + resolve({ + filename, + category, + localPath: persistentPath, + containerPath: path.join(this.modelPaths[category].container, filename), + size: stats.size + }); + } catch (error) { + reject(new Error(`Download completed but file move failed: ${error.message}`)); + } + }); + + fileStream.on('error', (error) => { + console.error(`File stream error for ${filename}:`, error); + // Clean up partial download + try { + fs.unlinkSync(downloadPath); + } catch (cleanupError) { + console.warn('Could not clean up partial download:', cleanupError.message); + } + reject(error); + }); + }); + + request.on('error', (error) => { + console.error(`Request error for ${filename}:`, error); + // Clean up partial download + try { + fs.unlinkSync(downloadPath); + } catch (cleanupError) { + console.warn('Could not clean up partial download:', cleanupError.message); + } + reject(error); + }); + request.setTimeout(300000, () => { // 5 minute timeout + console.error(`Download timeout for ${filename}`); + request.destroy(); + // Clean up partial download + try { + fs.unlinkSync(downloadPath); + } catch (cleanupError) { + console.warn('Could not clean up partial download:', cleanupError.message); + } + reject(new Error('Download timeout (5 minutes)')); + }); + }); + } catch (error) { + this.emit('download:error', { filename, error: error.message }); + throw error; + } + } + + /** + * Transfer downloaded model file into ComfyUI container + */ + async installModelToContainer(localFilePath, filename, category = 'checkpoints') { + try { + this.emit('install:start', { filename, category }); + + const container = await this.getContainer(); + const targetPath = `${this.modelPaths[category]}/${filename}`; + + // Create the target directory if it doesn't exist + const mkdirExec = await container.exec({ + Cmd: ['mkdir', '-p', this.modelPaths[category]], + AttachStdout: true, + AttachStderr: true + }); + await mkdirExec.start({ hijack: true, stdin: false }); + + // Read the local file + const fileData = fs.readFileSync(localFilePath); + + // Copy file into container using docker cp equivalent + const tarStream = require('tar-stream'); + const pack = tarStream.pack(); + + pack.entry({ name: filename }, fileData); + pack.finalize(); + + // Use container.putArchive to copy the file + await container.putArchive(pack, { path: this.modelPaths[category] }); + + // Verify the file was copied successfully + const verifyExec = await container.exec({ + Cmd: ['ls', '-la', targetPath], + AttachStdout: true, + AttachStderr: true + }); + + const verifyStream = await verifyExec.start({ hijack: true, stdin: false }); + + return new Promise((resolve, reject) => { + let output = ''; + + verifyStream.on('data', (data) => { + output += data.toString(); + }); + + verifyStream.on('end', () => { + if (output.includes(filename)) { + // Clean up temporary file + fs.unlinkSync(localFilePath); + this.emit('install:complete', { filename, category, path: targetPath }); + resolve(targetPath); + } else { + reject(new Error('File verification failed')); + } + }); + + verifyStream.on('error', reject); + }); + } catch (error) { + this.emit('install:error', { filename, category, error: error.message }); + throw error; + } + } + + /** + * Download and install model in one operation + */ + async downloadAndInstallModel(url, filename, category = 'checkpoints', onProgress = null, options = {}) { + try { + // Download the model + const tempFilePath = await this.downloadModel(url, filename, category, onProgress, 0, options); + + // Install it to the container + const installedPath = await this.installModelToContainer(tempFilePath, filename, category); + + return { + success: true, + filename, + category, + installedPath, + message: `Successfully installed ${filename} to ${category}` + }; + } catch (error) { + return { + success: false, + filename, + category, + error: error.message + }; + } + } + + /** + * Remove model from ComfyUI container + */ + async removeModel(filename, category = 'checkpoints') { + try { + const container = await this.getContainer(); + const targetPath = `${this.modelPaths[category]}/${filename}`; + + const exec = await container.exec({ + Cmd: ['rm', '-f', targetPath], + AttachStdout: true, + AttachStderr: true + }); + + await exec.start({ hijack: true, stdin: false }); + + this.emit('remove:complete', { filename, category }); + return { success: true, message: `Removed ${filename} from ${category}` }; + } catch (error) { + this.emit('remove:error', { filename, category, error: error.message }); + throw error; + } + } + + /** + * Search popular models from various sources + */ + async searchModels(query, source = 'huggingface', category = 'checkpoints') { + // This would integrate with various APIs to search for models + // For now, return a mock response structure + return { + source, + category, + query, + results: [ + { + name: `${query} Model`, + description: 'AI Generated model search result', + downloadUrl: `https://example.com/models/${query}.safetensors`, + size: '2.3 GB', + author: 'Community', + downloads: 12345, + rating: 4.5 + } + ] + }; + } + + /** + * Calculate download speed + */ + calculateSpeed(downloadedBytes) { + if (!this.downloadStartTime) { + this.downloadStartTime = Date.now(); + return '0 MB/s'; + } + + const elapsed = (Date.now() - this.downloadStartTime) / 1000; + const speed = downloadedBytes / elapsed; + + if (speed > 1024 * 1024) { + return `${(speed / (1024 * 1024)).toFixed(1)} MB/s`; + } else { + return `${(speed / 1024).toFixed(1)} KB/s`; + } + } + + /** + * Get model management status + */ + async getStatus() { + try { + const storageInfo = await this.getStorageInfo(); + const checkpoints = await this.listInstalledModels('checkpoints'); + const loras = await this.listInstalledModels('loras'); + const vaes = await this.listInstalledModels('vae'); + + return { + containerStatus: 'running', + storage: storageInfo, + modelCounts: { + checkpoints: checkpoints.length, + loras: loras.length, + vaes: vaes.length + }, + totalModels: checkpoints.length + loras.length + vaes.length + }; + } catch (error) { + return { + containerStatus: 'error', + error: error.message + }; + } + } + + /** + * List locally stored models (persistent storage) + */ + async listLocalModels(category = 'checkpoints') { + try { + const localPath = this.modelPaths[category]?.local; + if (!localPath || !fs.existsSync(localPath)) { + return []; + } + + const files = fs.readdirSync(localPath); + const modelFiles = files.filter(file => { + const ext = path.extname(file).toLowerCase(); + return ['.safetensors', '.ckpt', '.pth', '.bin', '.pt'].includes(ext); + }); + + return modelFiles.map(filename => { + const filePath = path.join(localPath, filename); + const stats = fs.statSync(filePath); + + return { + name: filename, + category: category, + localPath: filePath, + containerPath: path.join(this.modelPaths[category].container, filename), + size: stats.size, + modified: stats.mtime, + isLocal: true, + isPersistent: true + }; + }); + } catch (error) { + throw new Error(`Failed to list local models: ${error.message}`); + } + } + + /** + * Move file from download to persistent storage + */ + async moveFile(sourcePath, destinationPath) { + return new Promise((resolve, reject) => { + // Ensure destination directory exists + const destDir = path.dirname(destinationPath); + if (!fs.existsSync(destDir)) { + fs.mkdirSync(destDir, { recursive: true }); + } + + // Use rename for same filesystem, copy+delete for cross-filesystem + fs.rename(sourcePath, destinationPath, (renameError) => { + if (renameError && renameError.code === 'EXDEV') { + // Cross-filesystem move - copy then delete + const readStream = fs.createReadStream(sourcePath); + const writeStream = fs.createWriteStream(destinationPath); + + readStream.pipe(writeStream); + + writeStream.on('finish', () => { + fs.unlink(sourcePath, (unlinkError) => { + if (unlinkError) { + console.warn('Warning: Could not delete source file:', unlinkError); + } + resolve(); + }); + }); + + writeStream.on('error', reject); + readStream.on('error', reject); + } else if (renameError) { + reject(renameError); + } else { + resolve(); + } + }); + }); + } + + /** + * Delete a local model from persistent storage + */ + async deleteLocalModel(filename, category = 'checkpoints') { + try { + const localPath = path.join(this.modelPaths[category].local, filename); + + if (!fs.existsSync(localPath)) { + throw new Error(`Model file not found: ${filename}`); + } + + fs.unlinkSync(localPath); + + this.emit('model:deleted', { filename, category, localPath }); + + console.log(`🗑️ Deleted local model: ${filename}`); + + return { success: true, filename, category }; + } catch (error) { + throw new Error(`Failed to delete model: ${error.message}`); + } + } + + /** + * Get comprehensive storage information + */ + async getEnhancedStorageInfo() { + try { + const info = { + local: {}, + persistent: {}, + container: {}, + summary: { + totalLocalModels: 0, + totalLocalSize: 0, + totalPersistentModels: 0, + totalPersistentSize: 0 + } + }; + + // Get local storage info for each category + for (const [category, paths] of Object.entries(this.modelPaths)) { + try { + const localModels = await this.listLocalModels(category); + const totalSize = localModels.reduce((sum, model) => sum + model.size, 0); + + info.persistent[category] = { + count: localModels.length, + totalSize: totalSize, + path: paths.local, + models: localModels + }; + + info.summary.totalPersistentModels += localModels.length; + info.summary.totalPersistentSize += totalSize; + } catch (error) { + info.persistent[category] = { error: error.message }; + } + } + + // Get container storage info (if container is running) + try { + const container = await this.getContainer(); + const exec = await container.exec({ + Cmd: ['df', '-h', '/app/ComfyUI/models'], + AttachStdout: true, + AttachStderr: true + }); + + const stream = await exec.start({ hijack: true, stdin: false }); + + const containerInfo = await new Promise((resolve, reject) => { + let output = ''; + + stream.on('data', (data) => { + output += data.toString(); + }); + + stream.on('end', () => { + const lines = output.trim().split('\n'); + if (lines.length > 1) { + const [filesystem, size, used, available, percent, mountpoint] = lines[1].split(/\s+/); + resolve({ + filesystem, + size, + used, + available, + percent, + mountpoint, + isAccessible: true + }); + } else { + resolve({ error: 'Could not parse container storage info' }); + } + }); + + stream.on('error', reject); + }); + + info.container = containerInfo; + } catch (error) { + info.container = { + error: `Container not accessible: ${error.message}`, + isAccessible: false + }; + } + + return info; + } catch (error) { + throw new Error(`Failed to get enhanced storage info: ${error.message}`); + } + } + + /** + * Transfer external model file to persistent storage + */ + async importExternalModel(externalPath, filename, category = 'checkpoints') { + try { + if (!fs.existsSync(externalPath)) { + throw new Error(`External model file not found: ${externalPath}`); + } + + const persistentPath = path.join(this.modelPaths[category].local, filename); + + // Copy file to persistent storage + await this.moveFile(externalPath, persistentPath); + + console.log(`📥 Imported external model: ${filename} to ${category}`); + + this.emit('model:imported', { + filename, + category, + localPath: persistentPath, + containerPath: path.join(this.modelPaths[category].container, filename) + }); + + return { + success: true, + filename, + category, + localPath: persistentPath, + containerPath: path.join(this.modelPaths[category].container, filename) + }; + } catch (error) { + throw new Error(`Failed to import external model: ${error.message}`); + } + } + + /** + * Backup models from container to host + */ + async backupModels(category = 'checkpoints', backupPath) { + try { + const container = await this.getContainer(); + const sourcePath = this.modelPaths[category].container; + + // Create tar archive of the model directory + const exec = await container.exec({ + Cmd: ['tar', '-czf', `/tmp/${category}_backup.tar.gz`, '-C', sourcePath, '.'], + AttachStdout: true, + AttachStderr: true + }); + + await exec.start({ hijack: true, stdin: false }); + + // Copy the backup file out of the container + const stream = await container.getArchive({ path: `/tmp/${category}_backup.tar.gz` }); + const backupFile = path.join(backupPath, `${category}_backup_${Date.now()}.tar.gz`); + + await pipelineAsync(stream, fs.createWriteStream(backupFile)); + + // Clean up temporary file in container + const cleanupExec = await container.exec({ + Cmd: ['rm', '-f', `/tmp/${category}_backup.tar.gz`], + AttachStdout: true, + AttachStderr: true + }); + await cleanupExec.start({ hijack: true, stdin: false }); + + return { success: true, backupFile }; + } catch (error) { + throw new Error(`Backup failed: ${error.message}`); + } + } +} + +module.exports = ComfyUIModelService; \ No newline at end of file diff --git a/electron/debug-paths.cjs b/electron/debug-paths.cjs new file mode 100644 index 00000000..2abc645f --- /dev/null +++ b/electron/debug-paths.cjs @@ -0,0 +1,101 @@ +/** + * Debug script to test binary paths in production builds + * This can be invoked via IPC to help diagnose path issues + */ + +const path = require('path'); +const fs = require('fs'); +const { app } = require('electron'); +const log = require('electron-log'); + +function debugPaths() { + const info = { + nodeEnv: process.env.NODE_ENV, + isDev: process.env.NODE_ENV === 'development', + __dirname: __dirname, + appPath: app.getAppPath(), + resourcesPath: process.resourcesPath, + userDataPath: app.getPath('userData'), + exePath: app.getPath('exe'), + possibleBinaryPaths: [], + existingPaths: [], + directoryContents: {} + }; + + // Test all possible binary locations + const possiblePaths = [ + path.join(__dirname, 'llamacpp-binaries'), + path.join(process.resourcesPath, 'electron', 'llamacpp-binaries'), + path.join(app.getAppPath(), 'electron', 'llamacpp-binaries'), + path.join(app.getPath('userData'), 'llamacpp-binaries') + ]; + + possiblePaths.forEach(testPath => { + info.possibleBinaryPaths.push(testPath); + + if (fs.existsSync(testPath)) { + info.existingPaths.push(testPath); + + try { + const contents = fs.readdirSync(testPath); + info.directoryContents[testPath] = contents; + + // Check for platform-specific directories + contents.forEach(item => { + const itemPath = path.join(testPath, item); + if (fs.statSync(itemPath).isDirectory() && item.includes('-')) { + try { + const platformContents = fs.readdirSync(itemPath); + info.directoryContents[`${testPath}/${item}`] = platformContents; + } catch (e) { + info.directoryContents[`${testPath}/${item}`] = `Error: ${e.message}`; + } + } + }); + } catch (e) { + info.directoryContents[testPath] = `Error reading directory: ${e.message}`; + } + } + }); + + return info; +} + +function logDebugInfo() { + const info = debugPaths(); + + log.info('=== BINARY PATH DEBUG INFO ==='); + log.info('Environment:', info.nodeEnv); + log.info('__dirname:', info.__dirname); + log.info('app.getAppPath():', info.appPath); + log.info('process.resourcesPath:', info.resourcesPath); + log.info('app.getPath("userData"):', info.userDataPath); + log.info('app.getPath("exe"):', info.exePath); + + log.info('\n=== POSSIBLE BINARY PATHS ==='); + info.possibleBinaryPaths.forEach(p => log.info(`- ${p}`)); + + log.info('\n=== EXISTING PATHS ==='); + info.existingPaths.forEach(p => log.info(`✅ ${p}`)); + + if (info.existingPaths.length === 0) { + log.error('❌ No binary paths found!'); + } + + log.info('\n=== DIRECTORY CONTENTS ==='); + Object.entries(info.directoryContents).forEach(([path, contents]) => { + log.info(`${path}:`); + if (Array.isArray(contents)) { + contents.forEach(file => log.info(` - ${file}`)); + } else { + log.info(` ${contents}`); + } + }); + + return info; +} + +module.exports = { + debugPaths, + logDebugInfo +}; \ No newline at end of file diff --git a/electron/debug-service-paths.cjs b/electron/debug-service-paths.cjs new file mode 100644 index 00000000..ad6e9c27 --- /dev/null +++ b/electron/debug-service-paths.cjs @@ -0,0 +1,59 @@ +const path = require('path'); +const fs = require('fs'); + +/** + * Debug script to check service paths in both development and production + */ + +console.log('=== Service Path Debug ==='); + +const binaryName = 'llama-optimizer-windows.exe'; + +// Production path (when built with electron-builder) +const resourcesPath = process.resourcesPath + ? path.join(process.resourcesPath, 'electron', 'services', binaryName) + : null; + +// Development path +const devPath = path.join(__dirname, 'services', binaryName); + +console.log('Environment:', process.env.NODE_ENV || 'development'); +console.log('process.resourcesPath:', process.resourcesPath || 'undefined'); +console.log('__dirname:', __dirname); +console.log(''); + +console.log('Production path:', resourcesPath || 'N/A'); +console.log('Production exists:', resourcesPath ? fs.existsSync(resourcesPath) : 'N/A'); +console.log(''); + +console.log('Development path:', devPath); +console.log('Development exists:', fs.existsSync(devPath)); +console.log(''); + +// Check what path would be selected +let selectedPath; +if (resourcesPath && fs.existsSync(resourcesPath)) { + selectedPath = resourcesPath; + console.log('Selected: Production path'); +} else if (fs.existsSync(devPath)) { + selectedPath = devPath; + console.log('Selected: Development path'); +} else { + console.log('Selected: NONE - Error would occur'); +} + +console.log('Final path:', selectedPath || 'ERROR'); + +// List services directory contents if it exists +const servicesDir = resourcesPath ? path.dirname(resourcesPath) : path.join(__dirname, 'services'); +if (fs.existsSync(servicesDir)) { + console.log(''); + console.log('Services directory contents:'); + const files = fs.readdirSync(servicesDir); + files.forEach(file => { + console.log(' -', file); + }); +} else { + console.log(''); + console.log('Services directory not found:', servicesDir); +} diff --git a/electron/debug-volume-mounting.cjs b/electron/debug-volume-mounting.cjs new file mode 100644 index 00000000..d7adf77b --- /dev/null +++ b/electron/debug-volume-mounting.cjs @@ -0,0 +1,339 @@ +#!/usr/bin/env node + +const Docker = require('dockerode'); +const fs = require('fs'); +const path = require('path'); +const os = require('os'); + +class VolumeDebugger { + constructor() { + this.docker = new Docker(); + this.appDataPath = path.join(os.homedir(), '.clara'); + this.pythonBackendDataPath = path.join(this.appDataPath, 'python_backend_data'); + } + + async debugVolumeMounting() { + console.log('🔍 Debugging Clara Python Backend Volume Mounting\n'); + + // 1. Check host directory structure + await this.checkHostDirectories(); + + // 2. Check container existence and configuration + await this.checkContainerConfiguration(); + + // 3. Check container volume mounts + await this.checkContainerVolumeMounts(); + + // 4. Verify file permissions + await this.checkFilePermissions(); + + // 5. Check container logs + await this.checkContainerLogs(); + + // 6. Test container file system access + await this.testContainerFileAccess(); + + console.log('\n🏁 Volume mounting diagnostic complete!'); + } + + async checkHostDirectories() { + console.log('📁 Checking Host Directory Structure:'); + console.log('====================================='); + + const directories = [ + this.appDataPath, + this.pythonBackendDataPath, + path.join(this.pythonBackendDataPath, '.clara'), + path.join(this.pythonBackendDataPath, '.clara', 'lightrag_storage'), + path.join(this.pythonBackendDataPath, '.clara', 'lightrag_storage', 'metadata'), + ]; + + const files = [ + path.join(this.pythonBackendDataPath, '.clara', 'lightrag_storage', 'metadata', 'notebooks.json'), + path.join(this.pythonBackendDataPath, '.clara', 'lightrag_storage', 'metadata', 'documents.json'), + ]; + + directories.forEach(dir => { + const exists = fs.existsSync(dir); + console.log(` ${exists ? '✅' : '❌'} ${dir} ${exists ? '(exists)' : '(missing)'}`); + + if (exists) { + const stats = fs.statSync(dir); + console.log(` Permissions: ${stats.mode.toString(8)} | Owner: ${stats.uid}:${stats.gid}`); + } + }); + + console.log('\n📄 Checking Metadata Files:'); + files.forEach(file => { + const exists = fs.existsSync(file); + console.log(` ${exists ? '✅' : '❌'} ${file} ${exists ? '(exists)' : '(missing)'}`); + + if (exists) { + const stats = fs.statSync(file); + const content = fs.readFileSync(file, 'utf8'); + console.log(` Size: ${stats.size} bytes | Content: ${content.substring(0, 100)}${content.length > 100 ? '...' : ''}`); + } + }); + console.log(''); + } + + async checkContainerConfiguration() { + console.log('🐳 Checking Container Configuration:'); + console.log('==================================='); + + try { + const container = this.docker.getContainer('clara_python'); + const containerInfo = await container.inspect(); + + console.log(` ✅ Container exists: ${containerInfo.Name}`); + console.log(` 📊 State: ${containerInfo.State.Status} (Running: ${containerInfo.State.Running})`); + console.log(` 🖼️ Image: ${containerInfo.Config.Image}`); + console.log(` 🔄 Restart Count: ${containerInfo.RestartCount}`); + + // Check mounts + console.log('\n 📂 Configured Mounts:'); + containerInfo.Mounts.forEach((mount, index) => { + console.log(` ${index + 1}. Type: ${mount.Type}`); + console.log(` Source: ${mount.Source}`); + console.log(` Destination: ${mount.Destination}`); + console.log(` Mode: ${mount.Mode || 'default'}`); + console.log(` RW: ${mount.RW}`); + console.log(''); + }); + + } catch (error) { + console.log(` ❌ Container not found or error: ${error.message}`); + } + console.log(''); + } + + async checkContainerVolumeMounts() { + console.log('🔗 Checking Container Volume Mounts:'); + console.log('==================================='); + + try { + const container = this.docker.getContainer('clara_python'); + const containerInfo = await container.inspect(); + + // Look for our specific mount + const claraMounts = containerInfo.Mounts.filter(mount => + mount.Destination === '/home/clara' || + mount.Source.includes('python_backend_data') + ); + + if (claraMounts.length === 0) { + console.log(' ❌ No Clara home directory mount found!'); + console.log(' 🚨 ISSUE DETECTED: python_backend_data is not mounted to /home/clara'); + } else { + claraMounts.forEach(mount => { + console.log(` ✅ Found Clara mount:`); + console.log(` Source: ${mount.Source}`); + console.log(` Destination: ${mount.Destination}`); + console.log(` Type: ${mount.Type}`); + console.log(` RW: ${mount.RW}`); + + // Verify source exists + const sourceExists = fs.existsSync(mount.Source); + console.log(` Source exists on host: ${sourceExists ? '✅' : '❌'}`); + }); + } + + } catch (error) { + console.log(` ❌ Error checking mounts: ${error.message}`); + } + console.log(''); + } + + async checkFilePermissions() { + console.log('🔐 Checking File Permissions:'); + console.log('============================='); + + const importantPaths = [ + this.pythonBackendDataPath, + path.join(this.pythonBackendDataPath, '.clara'), + path.join(this.pythonBackendDataPath, '.clara', 'lightrag_storage'), + path.join(this.pythonBackendDataPath, '.clara', 'lightrag_storage', 'metadata'), + ]; + + importantPaths.forEach(dir => { + if (fs.existsSync(dir)) { + const stats = fs.statSync(dir); + const mode = stats.mode.toString(8); + const readable = fs.constants.R_OK; + const writable = fs.constants.W_OK; + + try { + fs.accessSync(dir, readable | writable); + console.log(` ✅ ${dir} - Mode: ${mode} (readable & writable)`); + } catch (error) { + console.log(` ❌ ${dir} - Mode: ${mode} (permission error: ${error.code})`); + } + } else { + console.log(` ❌ ${dir} - Does not exist`); + } + }); + console.log(''); + } + + async checkContainerLogs() { + console.log('📜 Checking Container Logs (last 50 lines):'); + console.log('==========================================='); + + try { + const container = this.docker.getContainer('clara_python'); + const logs = await container.logs({ + stdout: true, + stderr: true, + tail: 50, + timestamps: true + }); + + const logText = logs.toString(); + console.log(logText); + + } catch (error) { + console.log(` ❌ Error getting logs: ${error.message}`); + } + console.log(''); + } + + async testContainerFileAccess() { + console.log('🧪 Testing Container File System Access:'); + console.log('======================================='); + + try { + const container = this.docker.getContainer('clara_python'); + + // Test if the container can see the mounted files + const commands = [ + 'ls -la /home/clara', + 'ls -la /home/clara/.clara', + 'ls -la /home/clara/.clara/lightrag_storage', + 'ls -la /home/clara/.clara/lightrag_storage/metadata', + 'cat /home/clara/.clara/lightrag_storage/metadata/notebooks.json', + 'cat /home/clara/.clara/lightrag_storage/metadata/documents.json', + 'whoami', + 'id', + 'pwd' + ]; + + for (const command of commands) { + try { + console.log(`\n 🔧 Running: ${command}`); + const exec = await container.exec({ + Cmd: ['sh', '-c', command], + AttachStdout: true, + AttachStderr: true + }); + + const stream = await exec.start(); + const result = await this.streamToString(stream); + console.log(` 📤 Output: ${result.trim()}`); + + } catch (error) { + console.log(` ❌ Error: ${error.message}`); + } + } + + } catch (error) { + console.log(` ❌ Container access error: ${error.message}`); + } + console.log(''); + } + + async streamToString(stream) { + return new Promise((resolve, reject) => { + let data = ''; + stream.on('data', chunk => { + data += chunk.toString(); + }); + stream.on('end', () => resolve(data)); + stream.on('error', reject); + }); + } + + async fixVolumeMounting() { + console.log('🔧 Attempting to Fix Volume Mounting Issues:'); + console.log('============================================'); + + try { + // 1. Stop the container + console.log(' 1️⃣ Stopping container...'); + const container = this.docker.getContainer('clara_python'); + await container.stop(); + console.log(' ✅ Container stopped'); + + // 2. Remove the container + console.log(' 2️⃣ Removing container...'); + await container.remove({ force: true }); + console.log(' ✅ Container removed'); + + // 3. Ensure directories exist + console.log(' 3️⃣ Ensuring directories exist...'); + await this.ensureDirectoryStructure(); + console.log(' ✅ Directories verified'); + + // 4. Restart using Docker setup + console.log(' 4️⃣ Restarting container with proper volume mounting...'); + console.log(' ℹ️ Please restart Clara to recreate the container with proper volumes'); + + } catch (error) { + console.log(` ❌ Fix attempt failed: ${error.message}`); + } + } + + async ensureDirectoryStructure() { + const directories = [ + this.pythonBackendDataPath, + path.join(this.pythonBackendDataPath, '.clara'), + path.join(this.pythonBackendDataPath, '.clara', 'lightrag_storage'), + path.join(this.pythonBackendDataPath, '.clara', 'lightrag_storage', 'metadata'), + path.join(this.pythonBackendDataPath, '.cache'), + path.join(this.pythonBackendDataPath, 'uploads'), + path.join(this.pythonBackendDataPath, 'temp') + ]; + + directories.forEach(dir => { + if (!fs.existsSync(dir)) { + fs.mkdirSync(dir, { recursive: true }); + console.log(` ✅ Created: ${dir}`); + } + }); + + // Create metadata files + const metadataFiles = [ + { file: '.clara/lightrag_storage/metadata/notebooks.json', content: '{}' }, + { file: '.clara/lightrag_storage/metadata/documents.json', content: '{}' } + ]; + + metadataFiles.forEach(({ file, content }) => { + const filePath = path.join(this.pythonBackendDataPath, file); + if (!fs.existsSync(filePath)) { + fs.writeFileSync(filePath, content, 'utf8'); + console.log(` ✅ Created: ${file}`); + } + }); + } +} + +// Run the debugger +async function main() { + const debugger = new VolumeDebugger(); + + const args = process.argv.slice(2); + + if (args.includes('--fix')) { + await debugger.fixVolumeMounting(); + } else { + await debugger.debugVolumeMounting(); + + console.log('\n💡 Suggestions:'); + console.log('=============='); + console.log('1. If the python_backend_data mount is missing, run: node debug-volume-mounting.cjs --fix'); + console.log('2. If files are missing, the container will recreate them on next startup'); + console.log('3. If permissions are wrong, check that Clara has read/write access to ~/.clara/'); + console.log('4. Make sure Docker has permission to access the ~/.clara directory'); + } +} + +main().catch(console.error); \ No newline at end of file diff --git a/electron/dockerSetup.cjs b/electron/dockerSetup.cjs new file mode 100644 index 00000000..3f119515 --- /dev/null +++ b/electron/dockerSetup.cjs @@ -0,0 +1,3523 @@ +const { EventEmitter } = require('events'); +const Docker = require('dockerode'); +const path = require('path'); +const fs = require('fs'); +const os = require('os'); +const { app, dialog } = require('electron'); +const tar = require('tar-fs'); +const http = require('http'); +const { spawn, exec } = require('child_process'); +const { promisify } = require('util'); + +const execAsync = promisify(exec); + +class DockerSetup extends EventEmitter { + constructor(connectionConfig = null) { + super(); + this.isDevMode = process.env.NODE_ENV === 'development'; + this.appDataPath = path.join(os.homedir(), '.clara'); + + // Docker binary paths - using Docker CLI path for both docker and compose commands + this.dockerPath = '/usr/local/bin/docker'; + + // Store connection configuration + this.connectionConfig = connectionConfig || this.loadConnectionConfig(); + this.connectionMode = this.connectionConfig?.mode || 'local'; // 'local' or 'remote' + + // Initialize Docker client with the first working socket or remote connection + this.docker = this.initializeDockerClient(); + + // SSH tunnel management for remote connections + this.sshTunnels = {}; + this.activeTunnels = []; + + // Path for storing pull timestamps + this.pullTimestampsPath = path.join(this.appDataPath, 'pull_timestamps.json'); + + // Get system architecture + this.systemArch = this.getSystemArchitecture(); + console.log(`Detected system architecture: ${this.systemArch}`); + + // Ensure app data directory exists + if (!fs.existsSync(this.appDataPath)) { + fs.mkdirSync(this.appDataPath, { recursive: true }); + } + + // Create python_backend_data directory for explicit Python container persistence + // NOTE: This MUST be defined before the containers object to avoid undefined reference + this.pythonBackendDataPath = path.join(this.appDataPath, 'python_backend_data'); + if (!fs.existsSync(this.pythonBackendDataPath)) { + fs.mkdirSync(this.pythonBackendDataPath, { recursive: true }); + console.log(`📁 Created Python backend data directory: ${this.pythonBackendDataPath}`); + } + + // Container configuration - Removed Ollama container + this.containers = { + python: { + name: 'clara_python', + image: this.getArchSpecificImage('clara17verse/clara-backend', 'latest'), + // On Linux, use host network mode so Python backend runs on port 5000 + // On Windows/Mac, use bridge mode with port mapping 5001->5000 + port: process.platform === 'linux' ? 5000 : 5001, + internalPort: 5000, + healthCheck: this.isPythonRunning.bind(this), + volumes: [ + // Mount the python_backend_data folder as the clara user's home directory + `${this.pythonBackendDataPath}:/home/clara`, + // Keep backward compatibility for existing data paths + 'clara_python_models:/app/models' + ], + volumeNames: ['clara_python_models'], + // Base environment variables (always applied) + environment: [ + 'TOKENIZERS_PARALLELISM=false', // Prevent tokenizer warnings + 'OMP_NUM_THREADS=1', // Optimize for containerized environment + ], + // GPU-specific environment variables (will be conditionally added in startContainer) + gpuEnvironment: [ + 'NVIDIA_VISIBLE_DEVICES=all', + 'CUDA_VISIBLE_DEVICES=0', + // PyTorch/CUDA optimizations for AI inference + 'PYTORCH_CUDA_ALLOC_CONF=max_split_size_mb:512,expandable_segments:True', + 'CUDA_LAUNCH_BLOCKING=0', + 'TORCH_CUDNN_V8_API_ENABLED=1', + 'CUDA_MODULE_LOADING=LAZY', + 'CUDA_CACHE_DISABLE=0', // Enable CUDA caching for faster inference + // Whisper/Speech processing optimizations + 'WHISPER_CUDA=1', + 'FASTER_WHISPER_DEVICE=cuda' + ], + runtime: 'nvidia', // Enable GPU support if available + restartPolicy: 'unless-stopped' + }, + n8n: { + name: 'clara_n8n', + image: this.getArchSpecificImage('n8nio/n8n', 'latest'), + port: 5678, + internalPort: 5678, + healthCheck: this.checkN8NHealth.bind(this), + volumes: [ + `${path.join(this.appDataPath, 'n8n')}:/home/node/.n8n` + ] + }, + comfyui: { + name: 'clara_comfyui', + image: this.getArchSpecificImage('clara17verse/clara-comfyui', 'with-custom-nodes'), + port: 8188, + internalPort: 8188, + healthCheck: this.isComfyUIRunning.bind(this), + volumes: this.getComfyUIVolumes(), + environment: [ + 'NVIDIA_VISIBLE_DEVICES=all', + 'CUDA_VISIBLE_DEVICES=0', + // RTX 4090 optimizations (24GB VRAM) + 'PYTORCH_CUDA_ALLOC_CONF=max_split_size_mb:2048,expandable_segments:True', + 'CUDA_LAUNCH_BLOCKING=0', + 'TORCH_CUDNN_V8_API_ENABLED=1', + 'CUDA_MODULE_LOADING=LAZY', + // Disable xFormers warnings temporarily + 'XFORMERS_MORE_DETAILS=0', + // ComfyUI optimizations for RTX 4090 + 'COMFYUI_FORCE_FP16=1', + 'COMFYUI_DISABLE_XFORMERS_WARNING=1', + 'COMFYUI_HIGHVRAM=1', + 'COMFYUI_DISABLE_MODEL_OFFLOAD=1', + // Keep models in VRAM (don't offload to CPU) + 'COMFYUI_VRAM_USAGE=gpu-only' + ], + runtime: 'nvidia', // Enable GPU support if available + restartPolicy: 'unless-stopped' + } + }; + + // Initialize Python backend directory structure + this.initializePythonBackendDirectories(); + + // Initialize pull timestamps if not exists + this.initializePullTimestamps(); + + // Docker Compose file path + this.composeFilePath = path.join(this.appDataPath, 'docker-compose.yml'); + + // Docker Desktop app paths + this.dockerAppPaths = { + darwin: '/Applications/Docker.app' + }; + + // Get the app root directory + this.appRoot = path.resolve(__dirname, '..'); + + // Default ports with fallbacks + // On Linux, Python backend uses host network mode and runs on port 5000 + // On Windows/Mac, it uses bridge mode with port mapping 5001->5000 + this.ports = { + python: process.platform === 'linux' ? 5000 : 5001, + n8n: 5678, + ollama: 11434, + comfyui: 8188 + }; + + // Maximum retry attempts for service health checks + this.maxRetries = 3; + this.retryDelay = 5000; // 5 seconds + + // Clara container names + this.containerNames = ['clara_python', 'clara_n8n', 'clara_comfyui']; + + // Create subdirectories for each service + Object.keys(this.containers).forEach(service => { + const servicePath = path.join(this.appDataPath, service); + if (!fs.existsSync(servicePath)) { + fs.mkdirSync(servicePath, { recursive: true }); + } + }); + + // Create ComfyUI specific directories + const comfyuiDirs = [ + 'comfyui_models', + 'comfyui_output', + 'comfyui_input', + 'comfyui_custom_nodes', + 'comfyui_temp' + ]; + + comfyuiDirs.forEach(dir => { + const dirPath = path.join(this.appDataPath, dir); + if (!fs.existsSync(dirPath)) { + fs.mkdirSync(dirPath, { recursive: true }); + } + }); + } + + /** + * Initialize Python backend directory structure + * This creates the necessary subdirectories within the python_backend_data folder + * to ensure proper data organization and persistence + */ + initializePythonBackendDirectories() { + try { + console.log('🔧 Initializing Python backend directory structure...'); + + // Create essential directories for Python backend data + const pythonDirectories = [ + '.clara', // Clara configuration directory + '.clara/lightrag_storage', // RAG storage directory + '.clara/lightrag_storage/metadata', // Metadata for notebooks and documents + '.cache', // Python cache directory + 'uploads', // File uploads directory + 'temp' // Temporary files directory + ]; + + pythonDirectories.forEach(dir => { + const dirPath = path.join(this.pythonBackendDataPath, dir); + if (!fs.existsSync(dirPath)) { + fs.mkdirSync(dirPath, { recursive: true }); + console.log(` ✓ Created: ${dir}`); + } + }); + + // Create initial metadata files if they don't exist + const metadataFiles = [ + { file: '.clara/lightrag_storage/metadata/notebooks.json', content: '{}' }, + { file: '.clara/lightrag_storage/metadata/documents.json', content: '{}' } + ]; + + metadataFiles.forEach(({ file, content }) => { + const filePath = path.join(this.pythonBackendDataPath, file); + if (!fs.existsSync(filePath)) { + fs.writeFileSync(filePath, content, 'utf8'); + console.log(` ✓ Created: ${file}`); + } + }); + + console.log('✅ Python backend directory structure initialized successfully'); + + } catch (error) { + console.error('❌ Error initializing Python backend directories:', error.message); + // Don't throw here - the container can still work without perfect directory structure + } + } + + /** + * Get information about the Python backend data directory + * This provides transparency about where data is stored + */ + getPythonBackendInfo() { + return { + dataPath: this.pythonBackendDataPath, + mountPoint: '/home/clara', + description: 'All Python backend data is stored in a dedicated folder and mounted as the clara user home directory', + structure: { + '.clara/': 'Clara configuration and storage', + '.clara/lightrag_storage/': 'RAG system data and embeddings', + '.clara/lightrag_storage/metadata/': 'Notebook and document metadata', + '.cache/': 'Python package cache and temporary files', + 'uploads/': 'User uploaded files', + 'temp/': 'Temporary processing files' + }, + benefits: [ + 'Complete data persistence across container restarts', + 'Easy backup - just copy the python_backend_data folder', + 'Transparent data location on host system', + 'No data loss when updating containers' + ] + }; + } + + /** + * Detect if NVIDIA GPU and Docker runtime are available + */ + async detectNvidiaGPU() { + try { + // Check if nvidia-smi is available + const { stdout } = await execAsync('nvidia-smi --query-gpu=name --format=csv,noheader,nounits'); + const gpus = stdout.trim().split('\n').filter(line => line.trim()); + + if (gpus.length > 0) { + console.log(`🎮 Detected NVIDIA GPU(s): ${gpus.join(', ')}`); + + // Check if nvidia-container-runtime is available in Docker + try { + const { stdout: dockerInfo } = await execAsync('docker info'); + if (dockerInfo.includes('nvidia') || dockerInfo.includes('Nvidia')) { + console.log('✅ NVIDIA Container Runtime detected in Docker'); + return true; + } else { + console.log('⚠️ NVIDIA GPU detected but nvidia-container-runtime not available in Docker'); + this.getGPUSetupInstructions(); + return false; + } + } catch (runtimeError) { + console.log('⚠️ Could not check Docker runtime support'); + // Try to test GPU access directly + return await this.testNvidiaDockerAccess(); + } + } + + return false; + } catch (error) { + console.log('ℹ️ No NVIDIA GPU detected or nvidia-smi not available'); + return false; + } + } + + /** + * Test NVIDIA Docker access by running a simple GPU container + */ + async testNvidiaDockerAccess() { + try { + console.log('🧪 Testing NVIDIA Docker access...'); + const { stdout } = await execAsync('docker run --rm --gpus all nvidia/cuda:11.0-base nvidia-smi', { timeout: 30000 }); + if (stdout.includes('NVIDIA-SMI')) { + console.log('✅ NVIDIA Docker access confirmed'); + return true; + } + return false; + } catch (error) { + console.log('❌ NVIDIA Docker access test failed:', error.message); + return false; + } + } + + /** + * Provide GPU setup instructions for the user + */ + getGPUSetupInstructions() { + const platform = process.platform; + + console.log('\n🔧 GPU Setup Instructions:'); + console.log('=========================================='); + + if (platform === 'linux') { + console.log('For Linux (Ubuntu/Debian):'); + console.log('1. Install NVIDIA Container Toolkit:'); + console.log(' curl -fsSL https://nvidia.github.io/libnvidia-container/gpgkey | sudo gpg --dearmor -o /usr/share/keyrings/nvidia-container-toolkit-keyring.gpg'); + console.log(' curl -s -L https://nvidia.github.io/libnvidia-container/stable/deb/nvidia-container-toolkit.list | sed \'s#deb https://#deb [signed-by=/usr/share/keyrings/nvidia-container-toolkit-keyring.gpg] https://#g\' | sudo tee /etc/apt/sources.list.d/nvidia-container-toolkit.list'); + console.log(' sudo apt-get update && sudo apt-get install -y nvidia-container-toolkit'); + console.log('2. Configure Docker:'); + console.log(' sudo nvidia-ctk runtime configure --runtime=docker'); + console.log(' sudo systemctl restart docker'); + } else if (platform === 'win32') { + console.log('For Windows with Docker Desktop:'); + console.log('1. Ensure you have NVIDIA drivers installed'); + console.log('2. Enable WSL2 integration in Docker Desktop'); + console.log('3. Install nvidia-container-toolkit in WSL2:'); + console.log(' Follow Linux instructions above in your WSL2 distribution'); + } else if (platform === 'darwin') { + console.log('For macOS:'); + console.log('NVIDIA GPU support is not available on macOS with Docker Desktop'); + console.log('Consider using Metal Performance Shaders for GPU acceleration'); + } + + console.log('\n📖 Full documentation:'); + console.log('https://docs.nvidia.com/datacenter/cloud-native/container-toolkit/install-guide.html'); + console.log('==========================================\n'); + } + + /** + * Check if ComfyUI optimizations have already been run + */ + async checkComfyUIOptimizationStatus() { + try { + const optimizationFlagPath = path.join(this.appDataPath, 'comfyui_optimized.flag'); + return fs.existsSync(optimizationFlagPath); + } catch (error) { + return false; + } + } + + /** + * Mark ComfyUI as optimized + */ + async markComfyUIOptimized() { + try { + const optimizationFlagPath = path.join(this.appDataPath, 'comfyui_optimized.flag'); + fs.writeFileSync(optimizationFlagPath, new Date().toISOString()); + } catch (error) { + console.error('Error marking ComfyUI as optimized:', error); + } + } + + /** + * Optimize ComfyUI container for better GPU performance + */ + async optimizeComfyUIContainer() { + try { + // Check if already optimized + if (await this.checkComfyUIOptimizationStatus()) { + console.log('✅ ComfyUI already optimized, skipping...'); + return; + } + + console.log('🚀 Optimizing ComfyUI container for GPU performance...'); + + const container = this.docker.getContainer('clara_comfyui'); + + // Check if container is running + const containerInfo = await container.inspect(); + if (containerInfo.State.Status !== 'running') { + console.log('⚠️ ComfyUI container is not running, skipping optimization'); + return; + } + + // Run optimization commands inside the container + const optimizationCommands = [ + // Fix xFormers compatibility + 'pip install --force-reinstall xformers --index-url https://download.pytorch.org/whl/cu118', + // Install optimized ONNX runtime for ControlNet + 'pip install onnxruntime-gpu --force-reinstall', + // Clear PyTorch cache + 'python -c "import torch; torch.cuda.empty_cache()"' + ]; + + for (const command of optimizationCommands) { + try { + console.log(`Running: ${command}`); + const exec = await container.exec({ + Cmd: ['bash', '-c', command], + AttachStdout: true, + AttachStderr: true + }); + + const stream = await exec.start({ hijack: true, stdin: false }); + + // Wait for command to complete + await new Promise((resolve, reject) => { + let output = ''; + stream.on('data', (data) => { + output += data.toString(); + }); + stream.on('end', () => { + console.log(`✅ Command completed: ${command.substring(0, 50)}...`); + resolve(output); + }); + stream.on('error', reject); + + // Timeout after 5 minutes + setTimeout(() => reject(new Error('Command timeout')), 300000); + }); + } catch (error) { + console.log(`⚠️ Optimization command failed: ${command} - ${error.message}`); + } + } + + console.log('✅ ComfyUI optimization completed'); + await this.markComfyUIOptimized(); + } catch (error) { + console.error('❌ Error optimizing ComfyUI container:', error.message); + } + } + + /** + * Get ComfyUI volumes - hybrid approach with persistent storage and local model management + */ + getComfyUIVolumes() { + const os = require('os'); + + // Create persistent data directory for ComfyUI + const comfyUIDataDir = path.join(os.homedir(), '.clara', 'comfyui-data'); + + // Ensure directory exists + if (!fs.existsSync(comfyUIDataDir)) { + fs.mkdirSync(comfyUIDataDir, { recursive: true }); + } + + // Create subdirectories for different types of persistent data + const subdirs = ['models', 'outputs', 'temp', 'custom_nodes', 'user', 'config']; + subdirs.forEach(subdir => { + const subdirPath = path.join(comfyUIDataDir, subdir); + if (!fs.existsSync(subdirPath)) { + fs.mkdirSync(subdirPath, { recursive: true }); + } + + // Create model type subdirectories + if (subdir === 'models') { + const modelTypes = ['checkpoints', 'loras', 'vae', 'controlnet', 'upscale_models', 'embeddings', 'clip_vision']; + modelTypes.forEach(modelType => { + const modelTypePath = path.join(subdirPath, modelType); + if (!fs.existsSync(modelTypePath)) { + fs.mkdirSync(modelTypePath, { recursive: true }); + } + }); + } + }); + + console.log('🚀 ComfyUI persistent data directory:', comfyUIDataDir); + console.log('📁 Models will be stored persistently and managed locally'); + + // Mount persistent volumes for data that should survive container restarts + return [ + // Persistent model storage - allows local downloads to be transferred + `${path.join(comfyUIDataDir, 'models')}:/app/ComfyUI/models:rw`, + + // Output directory for generated images + `${path.join(comfyUIDataDir, 'outputs')}:/app/ComfyUI/output:rw`, + + // Temp directory for processing + `${path.join(comfyUIDataDir, 'temp')}:/app/ComfyUI/temp:rw`, + + // Custom nodes for extensions + `${path.join(comfyUIDataDir, 'custom_nodes')}:/app/ComfyUI/custom_nodes:rw`, + + // User directory for personal settings + `${path.join(comfyUIDataDir, 'user')}:/app/ComfyUI/user:rw`, + + // Config directory for ComfyUI settings + `${path.join(comfyUIDataDir, 'config')}:/app/ComfyUI/config:rw`, + + // Legacy support for existing paths + `${path.join(this.appDataPath, 'comfyui_input')}:/app/ComfyUI/input:rw` + ]; + } + + /** + * Enhanced Docker detection with comprehensive support for all Docker variants + */ + async detectDockerInstallations() { + const detectedInstallations = []; + + console.log('🔍 Starting comprehensive Docker detection...'); + + // 1. Socket-based detection (existing + enhanced) + const socketResults = await this.detectSocketBasedDocker(); + detectedInstallations.push(...socketResults); + + // 2. TCP/HTTP Docker detection + const tcpResults = await this.detectTcpDocker(); + detectedInstallations.push(...tcpResults); + + // 3. Docker Context detection + const contextResults = await this.detectDockerContexts(); + detectedInstallations.push(...contextResults); + + // 4. Docker Machine detection + const machineResults = await this.detectDockerMachine(); + detectedInstallations.push(...machineResults); + + // 5. Alternative container runtimes (Podman, etc.) + const alternativeResults = await this.detectAlternativeRuntimes(); + detectedInstallations.push(...alternativeResults); + + // 6. Process-based detection (fallback) + const processResults = await this.detectDockerProcesses(); + detectedInstallations.push(...processResults); + + // Remove duplicates and sort by priority + const uniqueInstallations = this.deduplicateAndPrioritize(detectedInstallations); + + console.log(`✅ Docker detection complete. Found ${uniqueInstallations.length} installation(s):`, + uniqueInstallations.map(i => `${i.type} (${i.method})`)); + + return uniqueInstallations; + } + + /** + * Enhanced socket-based Docker detection + */ + async detectSocketBasedDocker() { + const results = []; + + // Comprehensive list of socket locations + const socketLocations = [ + // Docker Desktop locations + { path: path.join(os.homedir(), '.docker', 'desktop', 'docker.sock'), type: 'Docker Desktop', priority: 1 }, + { path: path.join(os.homedir(), '.docker', 'docker.sock'), type: 'Docker Desktop', priority: 2 }, + + // Traditional Linux socket locations + { path: '/var/run/docker.sock', type: 'Docker Engine', priority: 3 }, + { path: '/run/docker.sock', type: 'Docker Engine', priority: 4 }, + + // WSL2 and Windows locations + { path: '/mnt/wsl/docker-desktop/docker.sock', type: 'Docker Desktop (WSL2)', priority: 5 }, + { path: '/mnt/wsl/shared-docker/docker.sock', type: 'Docker (WSL2 Shared)', priority: 6 }, + { path: '/mnt/c/Users/Public/.docker/docker.sock', type: 'Docker (Windows Shared)', priority: 7 }, + + // Alternative Docker implementations + { path: path.join(os.homedir(), '.colima', 'docker.sock'), type: 'Colima', priority: 8 }, + { path: path.join(os.homedir(), '.colima', 'default', 'docker.sock'), type: 'Colima (Default)', priority: 9 }, + { path: path.join(os.homedir(), '.rd', 'docker.sock'), type: 'Rancher Desktop', priority: 10 }, + { path: path.join(os.homedir(), '.lima', 'docker', 'sock', 'docker.sock'), type: 'Lima Docker', priority: 11 }, + { path: path.join(os.homedir(), '.lima', 'default', 'sock', 'docker.sock'), type: 'Lima Docker (Default)', priority: 12 }, + + // OrbStack (macOS Docker alternative) + { path: path.join(os.homedir(), '.orbstack', 'run', 'docker.sock'), type: 'OrbStack', priority: 13 }, + + // Snap Docker (Linux) + { path: '/var/snap/docker/common/var-lib-docker.sock', type: 'Docker (Snap)', priority: 14 }, + { path: '/run/snap.docker.dockerd.socket', type: 'Docker (Snap)', priority: 15 }, + + // Flatpak Docker (Linux) + { path: path.join(os.homedir(), '.var', 'app', 'io.docker.Docker', 'docker.sock'), type: 'Docker (Flatpak)', priority: 16 }, + + // Rootless Docker locations + { path: path.join(os.homedir(), '.docker', 'run', 'docker.sock'), type: 'Docker (Rootless)', priority: 17 }, + { path: `/run/user/${process.getuid ? process.getuid() : '1000'}/docker.sock`, type: 'Docker (Rootless User)', priority: 18 }, + + // Podman socket locations + { path: path.join(os.homedir(), '.local', 'share', 'containers', 'podman', 'machine', 'podman.sock'), type: 'Podman', priority: 19 }, + { path: `/run/user/${process.getuid ? process.getuid() : '1000'}/podman/podman.sock`, type: 'Podman (User)', priority: 20 }, + { path: '/run/podman/podman.sock', type: 'Podman (System)', priority: 21 } + ]; + + // Windows named pipe + if (process.platform === 'win32') { + socketLocations.unshift({ + path: '//./pipe/docker_engine', + type: 'Docker Desktop (Windows)', + priority: 0, + isNamedPipe: true + }); + + // Additional Windows pipes + socketLocations.push( + { path: '//./pipe/podman-machine-default', type: 'Podman (Windows)', priority: 22, isNamedPipe: true }, + { path: '//./pipe/docker_wsl', type: 'Docker (WSL)', priority: 23, isNamedPipe: true } + ); + } + + // Test each socket location + for (const location of socketLocations) { + try { + let canConnect = false; + + if (location.isNamedPipe) { + // For Windows named pipes, just try to create the client + const testClient = new Docker({ socketPath: location.path }); + await testClient.ping(); + canConnect = true; + } else { + // For Unix sockets, check file existence first + if (fs.existsSync(location.path)) { + const testClient = new Docker({ socketPath: location.path }); + await testClient.ping(); + canConnect = true; + } + } + + if (canConnect) { + results.push({ + type: location.type, + method: 'socket', + path: location.path, + priority: location.priority, + client: new Docker({ socketPath: location.path }), + isNamedPipe: location.isNamedPipe || false + }); + + console.log(`✅ Found working ${location.type} at: ${location.path}`); + } + } catch (error) { + // Silent fail for socket detection + continue; + } + } + + return results; + } + + /** + * Detect TCP/HTTP Docker connections + */ + async detectTcpDocker() { + const results = []; + const tcpHosts = [ + { host: 'localhost', port: 2375, tls: false, type: 'Docker (TCP)' }, + { host: 'localhost', port: 2376, tls: true, type: 'Docker (TLS)' }, + { host: '127.0.0.1', port: 2375, tls: false, type: 'Docker (TCP)' }, + { host: '127.0.0.1', port: 2376, tls: true, type: 'Docker (TLS)' } + ]; + + // Check environment variables for custom TCP hosts + if (process.env.DOCKER_HOST && process.env.DOCKER_HOST.startsWith('tcp://')) { + const url = new URL(process.env.DOCKER_HOST); + tcpHosts.unshift({ + host: url.hostname, + port: parseInt(url.port) || 2376, + tls: process.env.DOCKER_TLS_VERIFY === '1', + type: 'Docker (TCP from DOCKER_HOST)' + }); + } + + for (const tcpHost of tcpHosts) { + try { + const dockerOptions = { + host: tcpHost.host, + port: tcpHost.port + }; + + if (tcpHost.tls) { + dockerOptions.protocol = 'https'; + if (process.env.DOCKER_CERT_PATH) { + dockerOptions.ca = fs.readFileSync(path.join(process.env.DOCKER_CERT_PATH, 'ca.pem')); + dockerOptions.cert = fs.readFileSync(path.join(process.env.DOCKER_CERT_PATH, 'cert.pem')); + dockerOptions.key = fs.readFileSync(path.join(process.env.DOCKER_CERT_PATH, 'key.pem')); + } + } + + const testClient = new Docker(dockerOptions); + await testClient.ping(); + + results.push({ + type: tcpHost.type, + method: 'tcp', + host: tcpHost.host, + port: tcpHost.port, + tls: tcpHost.tls, + priority: 50, + client: testClient + }); + + console.log(`✅ Found working ${tcpHost.type} at: ${tcpHost.host}:${tcpHost.port}`); + } catch (error) { + // Silent fail for TCP detection + continue; + } + } + + return results; + } + + /** + * Detect Docker contexts + */ + async detectDockerContexts() { + const results = []; + + try { + const { stdout } = await execAsync('docker context ls --format json', { timeout: 5000 }); + const contexts = stdout.trim().split('\n').map(line => JSON.parse(line)); + + for (const context of contexts) { + if (context.Current) { + // This is the current context, try to use it + try { + const testClient = new Docker(); // Uses current context + await testClient.ping(); + + results.push({ + type: `Docker Context (${context.Name})`, + method: 'context', + contextName: context.Name, + endpoint: context.DockerEndpoint, + priority: 25, + client: testClient + }); + + console.log(`✅ Found working Docker context: ${context.Name} (${context.DockerEndpoint})`); + } catch (error) { + continue; + } + } + } + } catch (error) { + // Docker CLI not available or contexts not supported + } + + return results; + } + + /** + * Detect Docker Machine installations + */ + async detectDockerMachine() { + const results = []; + + try { + const { stdout } = await execAsync('docker-machine ls --format "{{.Name}},{{.State}},{{.URL}}"', { timeout: 5000 }); + const machines = stdout.trim().split('\n').filter(Boolean); + + for (const machineInfo of machines) { + const [name, state, url] = machineInfo.split(','); + + if (state === 'Running' && url) { + try { + const machineUrl = new URL(url); + const dockerOptions = { + host: machineUrl.hostname, + port: parseInt(machineUrl.port) || 2376, + protocol: 'https' + }; + + // Try to get machine environment + const { stdout: envOutput } = await execAsync(`docker-machine env ${name}`, { timeout: 3000 }); + const certPath = envOutput.match(/DOCKER_CERT_PATH="([^"]+)"/)?.[1]; + + if (certPath && fs.existsSync(certPath)) { + dockerOptions.ca = fs.readFileSync(path.join(certPath, 'ca.pem')); + dockerOptions.cert = fs.readFileSync(path.join(certPath, 'cert.pem')); + dockerOptions.key = fs.readFileSync(path.join(certPath, 'key.pem')); + } + + const testClient = new Docker(dockerOptions); + await testClient.ping(); + + results.push({ + type: `Docker Machine (${name})`, + method: 'machine', + machineName: name, + url: url, + priority: 30, + client: testClient + }); + + console.log(`✅ Found working Docker Machine: ${name} (${url})`); + } catch (error) { + continue; + } + } + } + } catch (error) { + // Docker Machine not available + } + + return results; + } + + /** + * Detect alternative container runtimes + */ + async detectAlternativeRuntimes() { + const results = []; + + // Test Podman compatibility + try { + const { stdout } = await execAsync('podman version --format json', { timeout: 3000 }); + const podmanInfo = JSON.parse(stdout); + + // Podman can be used as Docker replacement + const podmanSockets = [ + path.join(os.homedir(), '.local', 'share', 'containers', 'podman', 'machine', 'podman.sock'), + `/run/user/${process.getuid ? process.getuid() : '1000'}/podman/podman.sock`, + '/run/podman/podman.sock' + ]; + + for (const socketPath of podmanSockets) { + if (fs.existsSync(socketPath)) { + try { + const testClient = new Docker({ socketPath }); + await testClient.ping(); + + results.push({ + type: `Podman v${podmanInfo.Client.Version}`, + method: 'podman', + path: socketPath, + priority: 40, + client: testClient, + isPodman: true + }); + + console.log(`✅ Found working Podman at: ${socketPath}`); + break; // Only add one Podman instance + } catch (error) { + continue; + } + } + } + } catch (error) { + // Podman not available + } + + return results; + } + + /** + * Process-based Docker detection (fallback method) + */ + async detectDockerProcesses() { + const results = []; + + try { + let psCommand; + if (process.platform === 'win32') { + psCommand = 'wmic process where "name=\'dockerd.exe\' or name=\'docker.exe\'" get ProcessId,CommandLine /format:csv'; + } else { + psCommand = 'ps aux | grep -E "(dockerd|docker|podman)" | grep -v grep'; + } + + const { stdout } = await execAsync(psCommand, { timeout: 5000 }); + + if (stdout.trim()) { + // Found Docker processes, try default connection + try { + const testClient = new Docker(); + await testClient.ping(); + + results.push({ + type: 'Docker (Process Detection)', + method: 'process', + priority: 60, + client: testClient + }); + + console.log('✅ Found Docker via process detection'); + } catch (error) { + // Process exists but can't connect + } + } + } catch (error) { + // Process detection failed + } + + return results; + } + + /** + * Remove duplicates and prioritize Docker installations + */ + deduplicateAndPrioritize(installations) { + // Remove duplicates based on connection details + const unique = installations.filter((installation, index, self) => { + return index === self.findIndex(i => + i.path === installation.path && + i.host === installation.host && + i.port === installation.port + ); + }); + + // Sort by priority (lower number = higher priority) + return unique.sort((a, b) => a.priority - b.priority); + } + + /** + * Get the best Docker client from detected installations + */ + async getBestDockerClient() { + const installations = await this.detectDockerInstallations(); + + if (installations.length === 0) { + throw new Error('No Docker installations found'); + } + + // Return the highest priority (first) installation + const best = installations[0]; + console.log(`🎯 Using ${best.type} via ${best.method}`); + + return { + client: best.client, + info: best + }; + } + + /** + * Get detailed Docker detection report for debugging + */ + async getDockerDetectionReport() { + try { + const installations = await this.detectDockerInstallations(); + + const report = { + timestamp: new Date().toISOString(), + platform: process.platform, + architecture: os.arch(), + systemArch: this.systemArch, + totalFound: installations.length, + installations: installations.map(install => ({ + type: install.type, + method: install.method, + priority: install.priority, + path: install.path, + host: install.host, + port: install.port, + contextName: install.contextName, + machineName: install.machineName, + isPodman: install.isPodman || false, + isNamedPipe: install.isNamedPipe || false + })), + environment: { + DOCKER_HOST: process.env.DOCKER_HOST || 'not set', + DOCKER_TLS_VERIFY: process.env.DOCKER_TLS_VERIFY || 'not set', + DOCKER_CERT_PATH: process.env.DOCKER_CERT_PATH || 'not set', + DOCKER_MACHINE_NAME: process.env.DOCKER_MACHINE_NAME || 'not set' + } + }; + + console.log('📊 Docker Detection Report:', JSON.stringify(report, null, 2)); + return report; + } catch (error) { + console.error('Failed to generate Docker detection report:', error); + return { + error: error.message, + timestamp: new Date().toISOString(), + platform: process.platform, + architecture: os.arch() + }; + } + } + + /** + * Test all detected Docker installations + */ + async testAllDockerInstallations() { + const installations = await this.detectDockerInstallations(); + const results = []; + + console.log(`🧪 Testing ${installations.length} Docker installation(s)...`); + + for (const installation of installations) { + const testResult = { + type: installation.type, + method: installation.method, + working: false, + error: null, + responseTime: null + }; + + try { + const startTime = Date.now(); + await installation.client.ping(); + testResult.working = true; + testResult.responseTime = Date.now() - startTime; + console.log(`✅ ${installation.type} (${installation.method}): Working (${testResult.responseTime}ms)`); + } catch (error) { + testResult.error = error.message; + console.log(`❌ ${installation.type} (${installation.method}): Failed - ${error.message}`); + } + + results.push(testResult); + } + + return results; + } + + /** + * Get system architecture and map to Docker platform + */ + getSystemArchitecture() { + const arch = os.arch(); + const platform = os.platform(); + + console.log(`System info - Platform: ${platform}, Arch: ${arch}`); + + // Map Node.js arch to Docker platform + const archMap = { + 'x64': 'amd64', + 'arm64': 'arm64', + 'arm': 'arm/v7', + 'ia32': '386' + }; + + const dockerArch = archMap[arch] || arch; + + // For Docker platform specification, use linux as the OS part + // This works for Windows containers running Linux containers via WSL2 + return `linux/${dockerArch}`; + } + + /** + * Get just the Docker architecture without OS prefix + */ + getDockerArchitecture() { + const arch = os.arch(); + + // Map Node.js arch to Docker arch + const archMap = { + 'x64': 'amd64', + 'arm64': 'arm64', + 'arm': 'arm/v7', + 'ia32': '386' + }; + + return archMap[arch] || arch; + } + + /** + * Get architecture-specific image name + */ + getArchSpecificImage(baseImage, tag) { + // Special handling for clara-backend images which have architecture-specific tags + if (baseImage === 'clara17verse/clara-backend') { + const arch = os.arch(); + const platform = os.platform(); + + console.log(`Getting clara-backend image for platform: ${platform}, arch: ${arch}`); + + // For ARM64 systems (Mac ARM64 and Linux ARM64), use the default tag without suffix + if (arch === 'arm64') { + const imageName = `${baseImage}:${tag}`; + console.log(`Using ARM64 image: ${imageName}`); + return imageName; + } + + // For x64/AMD64 systems (Windows x64, Linux x64, Mac x64), use the -amd64 suffix + if (arch === 'x64') { + const imageName = `${baseImage}:${tag}-amd64`; + console.log(`Using AMD64 image: ${imageName}`); + return imageName; + } + + // Fallback to AMD64 for other architectures (ia32, etc.) + const imageName = `${baseImage}:${tag}-amd64`; + console.log(`Using fallback AMD64 image for arch ${arch}: ${imageName}`); + return imageName; + } + + // For other images, use the original approach (multi-arch images) + const imageName = `${baseImage}:${tag}`; + console.log(`Using standard multi-arch image: ${imageName}`); + return imageName; + } + + /** + * Check if container image has updates available + */ + async checkForImageUpdates(imageName, statusCallback) { + try { + // Validate imageName parameter + if (!imageName || typeof imageName !== 'string') { + console.error('Invalid imageName provided to checkForImageUpdates:', imageName); + return { + hasUpdate: false, + reason: 'Invalid image name provided', + imageName: imageName || 'undefined', + error: 'Invalid image name' + }; + } + + statusCallback(`Checking for updates to ${imageName}...`); + + // Get local image info + let localImage = null; + let localDigest = null; + try { + localImage = await this.docker.getImage(imageName).inspect(); + localDigest = localImage.RepoDigests && localImage.RepoDigests[0] + ? localImage.RepoDigests[0].split('@')[1] + : null; + console.log(`Local image digest for ${imageName}:`, localDigest); + } catch (error) { + if (error.statusCode === 404) { + statusCallback(`${imageName} not found locally, will download...`); + return { hasUpdate: true, reason: 'Image not found locally', imageName }; + } + throw error; + } + + // Enhanced update check: try to pull with no-cache option and compare digests + return new Promise((resolve, reject) => { + // Force a fresh check by pulling without cache + const pullOptions = { + platform: this.systemArch, + // Note: Docker API doesn't support --no-cache for pulls, but we'll check more thoroughly + }; + + console.log(`Starting update check for ${imageName} with platform ${this.systemArch}`); + + this.docker.pull(imageName, pullOptions, (err, stream) => { + if (err) { + console.error('Error checking for updates with platform specification:', err); + + // If platform-specific check fails, try without platform specification + console.log(`Retrying ${imageName} update check without platform specification...`); + + this.docker.pull(imageName, {}, (fallbackErr, fallbackStream) => { + if (fallbackErr) { + console.error('Error checking for updates (fallback):', fallbackErr); + resolve({ hasUpdate: false, reason: 'Failed to check for updates', error: fallbackErr.message, imageName }); + return; + } + + this.handleUpdateCheckStreamEnhanced(fallbackStream, imageName, statusCallback, resolve, localDigest); + }); + return; + } + + this.handleUpdateCheckStreamEnhanced(stream, imageName, statusCallback, resolve, localDigest); + }); + }); + } catch (error) { + console.error('Error checking for image updates:', error); + return { + hasUpdate: false, + reason: 'Error checking for updates', + error: error.message, + imageName: imageName || 'undefined' + }; + } + } + + /** + * Enhanced handle for the update check stream with better detection + */ + handleUpdateCheckStreamEnhanced(stream, imageName, statusCallback, resolve, localDigest) { + let hasUpdate = false; + let updateReason = ''; + let downloadingDetected = false; + let layersDownloaded = 0; + let totalLayers = 0; + let newDigest = null; + + stream.on('data', (data) => { + const lines = data.toString().split('\n').filter(Boolean); + lines.forEach(line => { + try { + const parsed = JSON.parse(line); + + // Log all status messages for debugging + if (parsed.status) { + console.log(`Docker pull status for ${imageName}:`, parsed.status); + } + + // Check for various update indicators + if (parsed.status) { + if (parsed.status.includes('Image is up to date')) { + // Even if Docker says "up to date", we'll do additional checks + updateReason = updateReason || 'Image appears up to date'; + } else if (parsed.status.includes('Downloading') || + parsed.status.includes('Extracting') || + parsed.status.includes('Pulling fs layer') || + parsed.status.includes('Waiting') || + parsed.status.includes('Verifying Checksum') || + parsed.status.includes('Download complete')) { + hasUpdate = true; + downloadingDetected = true; + updateReason = 'New layers detected - update available'; + layersDownloaded++; + } else if (parsed.status.includes('Pull complete')) { + if (downloadingDetected) { + hasUpdate = true; + updateReason = 'Update layers downloaded'; + } + } else if (parsed.status.includes('Already exists')) { + // Layer already exists - this is normal + totalLayers++; + } else if (parsed.status.includes('Status: Downloaded newer image')) { + hasUpdate = true; + updateReason = 'Downloaded newer image'; + } else if (parsed.status.includes('digest: sha256:')) { + // Extract the new digest for comparison + const digestMatch = parsed.status.match(/digest: (sha256:[a-f0-9]+)/); + if (digestMatch) { + newDigest = digestMatch[1]; + console.log(`Remote digest for ${imageName}:`, newDigest); + + // Compare digests if we have both + if (localDigest && newDigest && localDigest !== newDigest) { + hasUpdate = true; + updateReason = 'Image digest changed - update available'; + console.log(`Digest mismatch: local=${localDigest}, remote=${newDigest}`); + } + } + } + } + + // Check progress info for layer downloads + if (parsed.progressDetail && (parsed.progressDetail.current || parsed.progressDetail.total)) { + // This indicates actual download progress, meaning there's an update + hasUpdate = true; + downloadingDetected = true; + updateReason = 'Download progress detected - update in progress'; + } + } catch (e) { + // Ignore parse errors but log them for debugging + console.log('Parse error in Docker stream:', e.message); + } + }); + }); + + stream.on('end', () => { + // Final decision logic + if (!hasUpdate && layersDownloaded === 0 && totalLayers === 0) { + // If no layers were processed at all, assume we need to check more thoroughly + console.log(`No layer information detected for ${imageName}, might need update`); + hasUpdate = true; + updateReason = 'Unable to verify current status - recommend update check'; + } + + console.log(`Update check complete for ${imageName}: hasUpdate=${hasUpdate}, reason=${updateReason}`); + statusCallback(`Update check complete for ${imageName}: ${updateReason}`); + + resolve({ + hasUpdate, + reason: updateReason || 'No updates detected', + imageName, + layersDownloaded, + totalLayers, + localDigest, + remoteDigest: newDigest + }); + }); + + stream.on('error', (error) => { + console.error('Stream error during update check:', error); + resolve({ + hasUpdate: false, + reason: 'Error checking for updates', + error: error.message, + imageName + }); + }); + } + + /** + * Handle the update check stream + */ + handleUpdateCheckStream(stream, imageName, statusCallback, resolve) { + let hasUpdate = false; + let updateReason = ''; + let downloadingDetected = false; + + stream.on('data', (data) => { + const lines = data.toString().split('\n').filter(Boolean); + lines.forEach(line => { + try { + const parsed = JSON.parse(line); + + // Check for various update indicators + if (parsed.status) { + if (parsed.status.includes('Image is up to date')) { + hasUpdate = false; + updateReason = 'Image is up to date'; + } else if (parsed.status.includes('Downloading') || + parsed.status.includes('Extracting') || + parsed.status.includes('Pulling fs layer')) { + hasUpdate = true; + downloadingDetected = true; + updateReason = 'New version available'; + } else if (parsed.status.includes('Pull complete')) { + if (downloadingDetected) { + hasUpdate = true; + updateReason = 'Update downloaded'; + } + } + } + } catch (e) { + // Ignore parse errors + } + }); + }); + + stream.on('end', () => { + statusCallback(`Update check complete for ${imageName}`); + resolve({ + hasUpdate, + reason: updateReason || 'No updates available', + imageName + }); + }); + + stream.on('error', (error) => { + console.error('Stream error during update check:', error); + resolve({ + hasUpdate: false, + reason: 'Error checking for updates', + error: error.message, + imageName // Add imageName here to fix the undefined issue + }); + }); + } + + /** + * Automatically update containers without user prompts + */ + async autoUpdateContainers(statusCallback) { + try { + console.log('🔍 Starting autoUpdateContainers...'); + statusCallback('Starting container update check...'); + + // Check if we've already checked for updates recently (within the last hour) + const lastUpdateCheckFile = path.join(this.appDataPath, 'last_update_check.json'); + const now = Date.now(); + const oneHour = 60 * 60 * 1000; // 1 hour in milliseconds + + let shouldCheckForUpdates = true; + try { + if (fs.existsSync(lastUpdateCheckFile)) { + const lastCheck = JSON.parse(fs.readFileSync(lastUpdateCheckFile, 'utf8')); + if (now - lastCheck.timestamp < oneHour) { + console.log('⏭️ Skipping update check (checked recently)'); + statusCallback('Skipping update check (checked recently)'); + shouldCheckForUpdates = false; + } + } + } catch (error) { + console.log('Error reading last update check file:', error.message); + // Continue with update check if we can't read the file + } + + if (!shouldCheckForUpdates) { + console.log('✅ autoUpdateContainers completed (skipped)'); + return true; + } + + console.log('🔍 Checking for container updates...'); + statusCallback(`Checking for container updates (${this.systemArch})...`); + + // Check for updates for all containers in parallel + const updateChecks = []; + for (const [name, config] of Object.entries(this.containers)) { + // Skip ComfyUI container on macOS and Linux as it's not supported + if (name === 'comfyui' && (process.platform === 'darwin' || process.platform === 'linux')) { + console.log(`⏭️ Skipping ComfyUI container update check on ${process.platform} (not supported)`); + continue; + } + + console.log(`📦 Adding update check for ${name}: ${config.image}`); + updateChecks.push( + this.checkForImageUpdates(config.image, statusCallback) + .then(result => { + console.log(`✅ Update check completed for ${name}: hasUpdate=${result.hasUpdate}`); + return { ...result, containerName: name }; + }) + .catch(error => { + console.error(`❌ Update check failed for ${name}:`, error.message); + return { + hasUpdate: false, + error: error.message, + containerName: name, + imageName: config.image + }; + }) + ); + } + + console.log('⏳ Waiting for all update checks to complete...'); + const updateResults = await Promise.all(updateChecks); + console.log('✅ All update checks completed'); + + const updatesAvailable = updateResults.filter(result => result.hasUpdate); + + // Save the timestamp of this update check + try { + fs.writeFileSync(lastUpdateCheckFile, JSON.stringify({ timestamp: now })); + console.log('📝 Saved update check timestamp'); + } catch (error) { + console.log('Error saving last update check timestamp:', error.message); + } + + if (updatesAvailable.length > 0) { + console.log(`📥 Found ${updatesAvailable.length} container update(s) available`); + statusCallback(`Found ${updatesAvailable.length} container update(s) available - updating automatically...`); + + // Update containers in parallel for faster startup + const updatePromises = updatesAvailable.map(async (update) => { + try { + console.log(`📥 Starting update for ${update.imageName}...`); + await this.pullImageWithProgress(update.imageName, statusCallback); + console.log(`✅ Update completed for ${update.imageName}`); + return { success: true, imageName: update.imageName }; + } catch (error) { + console.error(`❌ Update failed for ${update.imageName}:`, error.message); + statusCallback(`Failed to update ${update.imageName}: ${error.message}`, 'warning'); + return { success: false, imageName: update.imageName, error: error.message }; + } + }); + + console.log('⏳ Waiting for all updates to complete...'); + const updateResults = await Promise.all(updatePromises); + console.log('✅ All updates completed'); + + const successCount = updateResults.filter(r => r.success).length; + const failCount = updateResults.filter(r => !r.success).length; + + if (successCount > 0) { + console.log(`✅ Updated ${successCount} container(s) successfully`); + statusCallback(`✓ Updated ${successCount} container(s) successfully`); + } + if (failCount > 0) { + console.log(`⚠️ ${failCount} container(s) failed to update`); + statusCallback(`⚠️ ${failCount} container(s) failed to update`, 'warning'); + } + } else { + console.log('✅ All containers are up to date'); + statusCallback('All containers are up to date'); + } + + console.log('✅ autoUpdateContainers completed successfully'); + return true; + } catch (error) { + console.error('❌ autoUpdateContainers failed:', error); + statusCallback(`Update check failed: ${error.message}`, 'warning'); + return false; + } + } + + /** + * Show update dialog and handle user choice + */ + async showUpdateDialog(updateInfo, parentWindow = null) { + const updatesAvailable = updateInfo.filter(info => info.hasUpdate); + + if (updatesAvailable.length === 0) { + return { updateAll: false, updates: [] }; + } + + const updateList = updatesAvailable.map(info => + `• ${info.imageName}: ${info.reason}` + ).join('\n'); + + const dialogOptions = { + type: 'question', + buttons: ['Update Now', 'Skip Updates', 'Cancel'], + defaultId: 0, + title: 'Container Updates Available', + message: `Updates are available for the following containers:\n\n${updateList}\n\nWould you like to update them now?`, + detail: `Architecture: ${this.systemArch}\n\nUpdating will ensure you have the latest features and security fixes.`, + alwaysOnTop: true, + modal: true + }; + + // If a parent window is provided, show dialog relative to it, otherwise show as standalone dialog + const response = parentWindow + ? await dialog.showMessageBox(parentWindow, dialogOptions) + : await dialog.showMessageBox(dialogOptions); + + return { + updateAll: response.response === 0, + skip: response.response === 1, + cancel: response.response === 2, + updates: updatesAvailable + }; + } + + /** + * Pull image with progress tracking and architecture specification + */ + async pullImageWithProgress(imageName, statusCallback) { + return new Promise((resolve, reject) => { + statusCallback(`Pulling ${imageName} for ${this.systemArch}...`); + + // Try pulling with platform specification first + this.docker.pull(imageName, { platform: this.systemArch }, (err, stream) => { + if (err) { + console.error('Error pulling image with platform specification:', err); + + // If platform-specific pull fails, try without platform specification + console.log(`Retrying ${imageName} pull without platform specification...`); + statusCallback(`Retrying ${imageName} pull without platform specification...`); + + this.docker.pull(imageName, {}, (fallbackErr, fallbackStream) => { + if (fallbackErr) { + console.error('Error pulling image (fallback):', fallbackErr); + + // Special fallback for clara-backend images: try base image without -amd64 suffix + if (imageName.includes('clara17verse/clara-backend') && imageName.includes('-amd64')) { + const baseImageName = imageName.replace('-amd64', ''); + console.log(`Trying base clara-backend image: ${baseImageName}`); + statusCallback(`Trying base clara-backend image: ${baseImageName}`); + + this.docker.pull(baseImageName, {}, (baseErr, baseStream) => { + if (baseErr) { + console.error('Error pulling base clara-backend image:', baseErr); + reject(baseErr); + return; + } + + this.handlePullStream(baseStream, baseImageName, statusCallback, resolve, reject); + }); + return; + } + + reject(fallbackErr); + return; + } + + this.handlePullStream(fallbackStream, imageName, statusCallback, resolve, reject); + }); + return; + } + + this.handlePullStream(stream, imageName, statusCallback, resolve, reject); + }); + }); + } + + /** + * Handle the Docker pull stream + */ + handlePullStream(stream, imageName, statusCallback, resolve, reject) { + let lastStatus = ''; + let progress = {}; + let isFirstTimePull = false; + + // Check if this is a first-time pull by looking at pull timestamps + const timestamps = this.getPullTimestamps(); + const lastPull = timestamps[imageName] || 0; + if (lastPull === 0) { + isFirstTimePull = true; + } + + // Show first-time setup message + if (isFirstTimePull) { + statusCallback(`🚀 First-time setup: Downloading AI services...`, 'info', { percentage: 0 }); + statusCallback(`This may take 5-15 minutes depending on your internet speed. Clara is downloading essential AI components - please wait...`, 'info', { percentage: 5 }); + } + + stream.on('data', (data) => { + const lines = data.toString().split('\n').filter(Boolean); + lines.forEach(line => { + try { + const parsed = JSON.parse(line); + + if (parsed.error) { + console.error('Pull error:', parsed.error); + reject(new Error(parsed.error)); + return; + } + + if (parsed.status && parsed.status !== lastStatus) { + lastStatus = parsed.status; + + // Track progress for different layers + if (parsed.id && parsed.progressDetail) { + progress[parsed.id] = parsed.progressDetail; + + // Calculate overall progress + const layers = Object.values(progress); + const totalCurrent = layers.reduce((sum, layer) => sum + (layer.current || 0), 0); + const totalTotal = layers.reduce((sum, layer) => sum + (layer.total || 0), 0); + + if (totalTotal > 0) { + const percentage = Math.round((totalCurrent / totalTotal) * 100); + const progressMessage = isFirstTimePull + ? `First-time download: ${imageName} (${percentage}%) - ${parsed.status}` + : `Pulling ${imageName}: ${parsed.status} (${percentage}%)`; + statusCallback(progressMessage, 'info', { percentage }); + } else { + const progressMessage = isFirstTimePull + ? `First-time setup: ${imageName} - ${parsed.status}` + : `Pulling ${imageName}: ${parsed.status}`; + statusCallback(progressMessage, 'info', { percentage: 10 }); + } + } else { + // Handle status messages without progress details + let percentage = 10; + if (parsed.status.includes('Downloading')) { + percentage = 30; + } else if (parsed.status.includes('Extracting')) { + percentage = 70; + } else if (parsed.status.includes('Pull complete')) { + percentage = 90; + } + + const progressMessage = isFirstTimePull + ? `First-time setup: ${imageName} - ${parsed.status}` + : `Pulling ${imageName}: ${parsed.status}`; + statusCallback(progressMessage, 'info', { percentage }); + } + } + } catch (e) { + // Ignore parse errors + } + }); + }); + + stream.on('end', () => { + const successMessage = isFirstTimePull + ? `✓ First-time setup complete: ${imageName} downloaded successfully!` + : `✓ Successfully pulled ${imageName}`; + statusCallback(successMessage, 'success', { percentage: 100 }); + this.updatePullTimestamp(imageName); + resolve(); + }); + + stream.on('error', (error) => { + console.error('Stream error:', error); + reject(error); + }); + } + + async execAsync(command, timeout = 60000) { + // Replace docker-compose with docker compose + command = command + .replace(/^docker-compose\s/, `"${this.dockerPath}" compose `) + .replace(/^docker\s/, `"${this.dockerPath}" `); + + return new Promise((resolve, reject) => { + exec(command, { + timeout, + env: { + ...process.env, + PATH: '/usr/local/bin:/usr/bin:/bin:/usr/sbin:/sbin' + } + }, (error, stdout, stderr) => { + if (error) { + reject(error); + } else { + resolve(stdout.trim()); + } + }); + }); + } + + async findAvailablePort(startPort, endPort = startPort + 100) { + for (let port = startPort; port <= endPort; port++) { + try { + await this.execAsync(`lsof -i :${port}`); + } catch (error) { + // If lsof fails, it means the port is available + return port; + } + } + throw new Error(`No available ports found between ${startPort} and ${endPort}`); + } + + async isPortInUse(port) { + try { + await this.execAsync(`lsof -i :${port}`); + return true; + } catch (error) { + return false; + } + } + + async findWorkingDockerSocket() { + try { + const { client, info } = await this.getBestDockerClient(); + + // Return the socket path or connection info + if (info.path) { + return info.path; + } else if (info.host && info.port) { + return `tcp://${info.host}:${info.port}`; + } else { + // Fallback to default socket + return process.platform === 'win32' ? '//./pipe/docker_engine' : '/var/run/docker.sock'; + } + } catch (error) { + console.error('Enhanced Docker detection failed:', error.message); + + // Fallback to original simple detection + const possibleSockets = [ + // Docker Desktop locations + path.join(os.homedir(), '.docker', 'desktop', 'docker.sock'), + path.join(os.homedir(), '.docker', 'docker.sock'), + // Traditional Linux socket locations + '/var/run/docker.sock', + '/run/docker.sock', + // WSL2 socket location + '/mnt/wsl/docker-desktop/docker.sock', + // Colima socket location (for macOS/Linux) + path.join(os.homedir(), '.colima', 'docker.sock'), + // Rancher Desktop socket location + path.join(os.homedir(), '.rd', 'docker.sock') + ]; + + // Windows pipe + if (process.platform === 'win32') { + return '//./pipe/docker_engine'; + } + + // Check environment variable first + if (process.env.DOCKER_HOST) { + const match = process.env.DOCKER_HOST.match(/unix:\/\/(.*)/); + if (match && match[1]) { + try { + const docker = new Docker({ socketPath: match[1] }); + await docker.ping(); + console.log('Using Docker socket from DOCKER_HOST:', match[1]); + return match[1]; + } catch (error) { + console.log('DOCKER_HOST socket not working:', error.message); + } + } + } + + // Try each socket location + for (const socketPath of possibleSockets) { + try { + if (fs.existsSync(socketPath)) { + const docker = new Docker({ socketPath }); + await docker.ping(); + console.log('Found working Docker socket at:', socketPath); + return socketPath; + } + } catch (error) { + console.log('Socket not working at:', socketPath, error.message); + continue; + } + } + + throw new Error('No working Docker socket found'); + } + } + + initializeDockerClient() { + try { + // Check if we're using remote Docker connection + if (this.connectionMode === 'remote' && this.connectionConfig) { + console.log('🌐 Initializing remote Docker connection...'); + return this.createRemoteDockerClient(this.connectionConfig); + } + + // Local Docker connection logic (existing) + // Try enhanced detection first (but don't await since this is sync) + // We'll use the enhanced detection in isDockerRunning() instead + + // For Windows, always use the named pipe as default + if (process.platform === 'win32') { + return new Docker({ socketPath: '//./pipe/docker_engine' }); + } + + // For other platforms, try to find a working socket synchronously + const socketPaths = [ + process.env.DOCKER_HOST ? process.env.DOCKER_HOST.replace('unix://', '') : null, + path.join(os.homedir(), '.docker', 'desktop', 'docker.sock'), + path.join(os.homedir(), '.docker', 'docker.sock'), + '/var/run/docker.sock', + '/run/docker.sock', + '/mnt/wsl/docker-desktop/docker.sock', + path.join(os.homedir(), '.colima', 'docker.sock'), + path.join(os.homedir(), '.rd', 'docker.sock'), + // Additional enhanced locations + path.join(os.homedir(), '.colima', 'default', 'docker.sock'), + path.join(os.homedir(), '.lima', 'docker', 'sock', 'docker.sock'), + path.join(os.homedir(), '.lima', 'default', 'sock', 'docker.sock'), + path.join(os.homedir(), '.orbstack', 'run', 'docker.sock'), + path.join(os.homedir(), '.docker', 'run', 'docker.sock') + ].filter(Boolean); + + for (const socketPath of socketPaths) { + if (fs.existsSync(socketPath)) { + try { + return new Docker({ socketPath }); + } catch (error) { + console.log(`Failed to initialize Docker with socket ${socketPath}:`, error.message); + } + } + } + + // If no socket works, fall back to default + return new Docker({ socketPath: '/var/run/docker.sock' }); + } catch (error) { + console.error('Error initializing Docker client:', error); + // Return a default client - the isDockerRunning check will handle the error case + return new Docker({ socketPath: '/var/run/docker.sock' }); + } + } + + /** + * Create a Docker client for remote connections + * Supports both direct TCP and SSH-based connections + */ + createRemoteDockerClient(config) { + try { + if (config.protocol === 'ssh') { + // SSH-based connection using docker context + // This requires SSH key authentication + console.log(`🔐 Creating SSH Docker connection to ${config.username}@${config.host}`); + + // For SSH, we use DOCKER_HOST environment variable approach + // Set up SSH tunnel to forward Docker socket + const dockerHost = `ssh://${config.username}@${config.host}`; + + // Create Docker client that will use SSH + return new Docker({ + host: config.host, + port: config.port || 22, + protocol: 'ssh', + username: config.username, + // Note: For proper SSH support, we'll need to set up SSH tunnel separately + // or use docker context CLI commands + }); + } else if (config.protocol === 'tcp') { + // Direct TCP connection (Docker daemon exposed on TCP port) + console.log(`🌐 Creating TCP Docker connection to ${config.host}:${config.port}`); + + return new Docker({ + host: config.host, + port: config.port || 2375, + protocol: 'http' + }); + } else { + throw new Error(`Unsupported remote protocol: ${config.protocol}`); + } + } catch (error) { + console.error('Failed to create remote Docker client:', error); + throw error; + } + } + + async isDockerRunning() { + try { + // If current client isn't working, try to find a working socket + try { + await this.docker.ping(); + return true; + } catch (error) { + console.log('Current Docker client not working, trying enhanced detection...'); + + // Use enhanced detection to find the best Docker client + try { + const { client, info } = await this.getBestDockerClient(); + this.docker = client; + await this.docker.ping(); + console.log(`✅ Successfully connected to ${info.type} via ${info.method}`); + return true; + } catch (enhancedError) { + console.log('Enhanced detection failed, trying fallback method...'); + + // Fallback to original socket detection + const workingSocket = await this.findWorkingDockerSocket(); + this.docker = new Docker({ socketPath: workingSocket }); + await this.docker.ping(); + return true; + } + } + } catch (error) { + console.error('Docker is not running or not accessible:', error.message); + return false; + } + } + + async createNetwork() { + try { + // First check if the network already exists + const networks = await this.docker.listNetworks(); + const networkExists = networks.some(network => network.Name === 'clara_network'); + + if (networkExists) { + console.log('Network clara_network already exists, skipping creation'); + return; + } + + // Create the network if it doesn't exist + try { + console.log('Creating clara_network...'); + await this.docker.createNetwork({ + Name: 'clara_network', + Driver: 'bridge', + Attachable: true, + Internal: false, + Scope: 'local' + }); + console.log('Successfully created clara_network'); + } catch (error) { + // Special handling for conflict error (network created between our check and creation) + if (error.statusCode === 409) { + console.log('Network already exists (409 error), continuing...'); + return; + } + + // Log details for other errors to help troubleshooting + console.error('Error creating network:', error.message); + console.error('Error details:', error); + + // For Mac-specific issues, provide more guidance + if (process.platform === 'darwin') { + console.log('On macOS, make sure Docker Desktop is running and properly configured'); + console.log('Try restarting Docker Desktop if issues persist'); + } + + // For Windows-specific issues + if (process.platform === 'win32') { + console.log('On Windows, ensure Docker Desktop is running and WSL2 backend is enabled'); + console.log('Try restarting Docker Desktop if issues persist'); + } + + throw new Error(`Failed to create network: ${error.message}`); + } + } catch (error) { + console.error('Error in createNetwork:', error.message); + // Re-throw the error so the caller can handle it appropriately + throw error; + } + } + + async createDockerVolumes() { + try { + console.log('Creating Docker volumes for persistent storage...'); + + // Get list of existing volumes + const volumes = await this.docker.listVolumes(); + const existingVolumeNames = volumes.Volumes ? volumes.Volumes.map(vol => vol.Name) : []; + + // Collect all volume names from all containers that specify them + const volumesToCreate = []; + for (const [serviceName, config] of Object.entries(this.containers)) { + // Skip ComfyUI container on macOS and Linux as it's not supported + if (serviceName === 'comfyui' && (process.platform === 'darwin' || process.platform === 'linux')) { + console.log(`⏭️ Skipping ComfyUI volume creation on ${process.platform} (not supported)`); + continue; + } + + if (config.volumeNames) { + for (const volumeName of config.volumeNames) { + if (!existingVolumeNames.includes(volumeName)) { + volumesToCreate.push({ + name: volumeName, + service: serviceName + }); + } + } + } + } + + // Create volumes that don't exist + for (const volumeInfo of volumesToCreate) { + try { + await this.docker.createVolume({ + Name: volumeInfo.name, + Driver: 'local', + Labels: { + 'clara.service': volumeInfo.service, + 'clara.managed': 'true' + } + }); + console.log(`✓ Created Docker volume: ${volumeInfo.name} for ${volumeInfo.service}`); + } catch (error) { + // Handle conflict error (volume created between our check and creation) + if (error.statusCode === 409) { + console.log(`Volume ${volumeInfo.name} already exists, continuing...`); + } else { + console.error(`Error creating volume ${volumeInfo.name}:`, error.message); + // Don't throw here, continue with other volumes + } + } + } + + if (volumesToCreate.length === 0) { + console.log('All required Docker volumes already exist'); + } + + } catch (error) { + console.error('Error in createDockerVolumes:', error.message); + // Don't throw here to allow the application to continue + // Containers will still work with the bind mounts if volumes fail + } + } + + async pullImage(imageName, statusCallback) { + // Use the new architecture-aware pull method + return this.pullImageWithProgress(imageName, statusCallback); + } + + async startContainer(config) { + try { + // Validate config parameter + if (!config) { + throw new Error('Container configuration is required but was not provided'); + } + + if (!config.name) { + throw new Error('Container configuration is missing required "name" property'); + } + + if (!config.image) { + throw new Error(`Container configuration for "${config.name}" is missing required "image" property`); + } + + // First, verify Docker daemon is responsive + try { + if (config.statusCallback) { + config.statusCallback('Verifying Docker daemon...', 'info', { percentage: 0 }); + } + await this.docker.ping(); + } catch (pingError) { + throw new Error(`Docker daemon is not responsive: ${pingError.message}. Please ensure Docker Desktop is running.`); + } + + // Check if container exists and is running + try { + const existingContainer = await this.docker.getContainer(config.name); + const containerInfo = await existingContainer.inspect(); + + if (containerInfo.State.Running) { + console.log(`Container ${config.name} is already running, checking health...`); + + // Check if the running container is healthy + const isHealthy = await config.healthCheck(); + if (isHealthy) { + console.log(`Container ${config.name} is running and healthy, skipping recreation`); + if (config.statusCallback) { + config.statusCallback(`${config.name} is already running and healthy`, 'success', { percentage: 100 }); + } + return { + success: true, + alreadyRunning: true, + message: `${config.name} is already running and healthy` + }; + } + + console.log(`Container ${config.name} is running but not healthy, will recreate`); + if (config.statusCallback) { + config.statusCallback(`${config.name} is unhealthy, restarting...`, 'warning', { percentage: 5 }); + } + await existingContainer.stop(); + await existingContainer.remove({ force: true }); + } else { + console.log(`Container ${config.name} exists but is not running, will recreate`); + if (config.statusCallback) { + config.statusCallback(`${config.name} is stopped, restarting...`, 'info', { percentage: 5 }); + } + await existingContainer.remove({ force: true }); + } + } catch (error) { + if (error.statusCode !== 404) { + console.error(`Error checking container ${config.name}:`, error); + } else { + console.log(`No existing container ${config.name}, will create new one`); + } + } + + // First ensure we have the image + try { + await this.docker.getImage(config.image).inspect(); + } catch (error) { + if (error.statusCode === 404) { + console.log(`Image ${config.image} not found locally, pulling for ${this.systemArch}...`); + + // Use the statusCallback if provided, otherwise fallback to console.log + const statusCallback = config.statusCallback || ((status) => console.log(status)); + await this.pullImageWithProgress(config.image, statusCallback); + } else { + throw error; + } + } + + // Ensure the Clara network exists before creating the container + try { + console.log('Ensuring clara_network exists...'); + + // Send progress update if callback is available + if (config.statusCallback) { + config.statusCallback('Setting up container network...', 'info', { percentage: 0 }); + } + + await this.createNetwork(); + + if (config.statusCallback) { + config.statusCallback('Network setup complete', 'success', { percentage: 10 }); + } + } catch (networkError) { + console.error('Failed to create/verify network:', networkError); + if (config.statusCallback) { + config.statusCallback('Network setup failed, attempting to continue...', 'warning', { percentage: 5 }); + } + // Don't throw immediately - try to continue as the network might exist + console.log('Attempting to continue despite network setup issues...'); + } + + console.log(`Creating container ${config.name} with port mapping ${config.internalPort} -> ${config.port}`); + + // Check for GPU availability if this container requests GPU runtime + let useGPURuntime = false; + if (config.runtime === 'nvidia') { + useGPURuntime = await this.detectNvidiaGPU(); + if (useGPURuntime) { + console.log(`🚀 GPU support enabled for ${config.name}`); + } else { + console.log(`⚠️ GPU requested but not available for ${config.name}, falling back to CPU`); + } + } + + // Create and start container + let networkMode = 'clara_network'; + + // Check if clara_network actually exists before using it + try { + const networks = await this.docker.listNetworks(); + const networkExists = networks.some(network => network.Name === 'clara_network'); + if (!networkExists) { + console.warn('clara_network not found, falling back to bridge network'); + networkMode = 'bridge'; + if (config.statusCallback) { + config.statusCallback('Using default bridge network (some features may be limited)', 'warning', { percentage: 15 }); + } + } + } catch (error) { + console.warn('Failed to check network status, using bridge network:', error.message); + networkMode = 'bridge'; + } + + // Add host.docker.internal mapping for Linux + const isLinux = process.platform === 'linux'; + const extraHosts = []; + + // Special handling for Python backend on Linux: use host network mode + // This allows it to access host services (like ClaraCore on port 8091) without firewall issues + const isPythonOnLinux = isLinux && config.name === 'clara_python'; + + if (isPythonOnLinux) { + console.log('🐧 Linux detected: Using host network mode for Python backend (can access localhost:8091 directly)'); + networkMode = 'host'; + } else if (isLinux) { + // For other containers on Linux, add host.docker.internal mapping + // On Linux, we need to get the gateway IP of clara_network + // host-gateway resolves to default bridge (172.17.0.1), but we use custom network + try { + const networks = await this.docker.listNetworks({ filters: { name: ['clara_network'] } }); + if (networks && networks.length > 0) { + const network = await this.docker.getNetwork(networks[0].Id); + const networkInfo = await network.inspect(); + const gatewayIP = networkInfo.IPAM?.Config?.[0]?.Gateway; + + if (gatewayIP) { + extraHosts.push(`host.docker.internal:${gatewayIP}`); + console.log(`🐧 Linux detected: Adding host.docker.internal:${gatewayIP} mapping (clara_network gateway)`); + } else { + // Fallback to host-gateway if we can't detect gateway IP + extraHosts.push('host.docker.internal:host-gateway'); + console.log('🐧 Linux detected: Adding host.docker.internal:host-gateway mapping (fallback)'); + } + } else { + extraHosts.push('host.docker.internal:host-gateway'); + console.log('🐧 Linux detected: Adding host.docker.internal:host-gateway mapping (network not found)'); + } + } catch (err) { + console.error('❌ Failed to detect clara_network gateway:', err.message); + extraHosts.push('host.docker.internal:host-gateway'); + console.log('🐧 Linux detected: Adding host.docker.internal:host-gateway mapping (error fallback)'); + } + } + + const containerConfig = { + Image: config.image, + name: config.name, + // Host network mode doesn't use port mappings + ...(isPythonOnLinux ? {} : { + ExposedPorts: { + [`${config.internalPort}/tcp`]: {} + } + }), + HostConfig: { + // Host network mode doesn't use port bindings + ...(isPythonOnLinux ? {} : { + PortBindings: { + [`${config.internalPort}/tcp`]: [{ HostPort: config.port.toString() }] + } + }), + Binds: config.volumes, + NetworkMode: networkMode, + // Add extra hosts for host.docker.internal on Linux (not needed for host mode) + ...(!isPythonOnLinux && extraHosts.length > 0 && { ExtraHosts: extraHosts }), + // Add restart policy if specified + ...(config.restartPolicy && { RestartPolicy: { Name: config.restartPolicy } }), + // Add GPU device access using modern DeviceRequests API (works on both Linux and Windows) + ...(useGPURuntime && { + DeviceRequests: [{ + Driver: 'nvidia', + Count: -1, // All GPUs + Capabilities: [['gpu', 'compute', 'utility']] + }] + }) + }, + Env: [ + 'PYTHONUNBUFFERED=1', + 'OLLAMA_BASE_URL=http://clara_ollama:11434', + // Add any environment variables from the container config + ...(config.environment || []), + // Add GPU-specific environment variables if GPU is available + ...(useGPURuntime ? [ + 'NVIDIA_VISIBLE_DEVICES=all', + 'NVIDIA_DRIVER_CAPABILITIES=compute,utility', + // Add container-specific GPU environment variables + ...(config.gpuEnvironment || []) + ] : [ + // CPU fallback environment variables + ...(config.name === 'clara_python' ? [ + 'WHISPER_CUDA=0', + 'FASTER_WHISPER_DEVICE=cpu' + ] : []) + ]) + ] + }; + + try { + if (config.statusCallback) { + config.statusCallback(`Creating container ${config.name}...`, 'info', { percentage: 20 }); + } + + const newContainer = await this.docker.createContainer(containerConfig); + console.log(`Container ${config.name} created, starting...`); + + if (config.statusCallback) { + config.statusCallback(`Starting container ${config.name}...`, 'info', { percentage: 30 }); + } + + await newContainer.start(); + console.log(`Container ${config.name} started, waiting for health check...`); + + if (config.statusCallback) { + config.statusCallback(`Container started, performing health checks...`, 'info', { percentage: 40 }); + } + } catch (containerError) { + console.error(`Failed to create/start container ${config.name}:`, containerError); + + // Provide more specific error messages + if (containerError.message.includes('network')) { + throw new Error(`Network error: ${containerError.message}. Try restarting Docker Desktop.`); + } else if (containerError.message.includes('port')) { + throw new Error(`Port binding error: ${containerError.message}. Port ${config.port} may be in use.`); + } else { + throw new Error(`Container creation failed: ${containerError.message}`); + } + } + + // Initial delay to give the container time to fully start + await new Promise(resolve => setTimeout(resolve, 5000)); + + // Wait for health check + let healthy = false; + const maxHealthChecks = 5; + for (let i = 0; i < maxHealthChecks; i++) { + console.log(`Health check attempt ${i + 1} for ${config.name}...`); + + if (config.statusCallback) { + const healthProgress = 40 + (i / maxHealthChecks) * 50; // Progress from 40% to 90% + config.statusCallback(`Health check ${i + 1}/${maxHealthChecks} for ${config.name}...`, 'info', { percentage: Math.round(healthProgress) }); + } + + try { + healthy = await config.healthCheck(); + console.log(`Health check result for ${config.name}: ${healthy}`); + if (healthy) { + if (config.statusCallback) { + config.statusCallback(`${config.name} is healthy and ready!`, 'success', { percentage: 100 }); + } + break; + } + } catch (error) { + console.error(`Health check error for ${config.name}:`, error); + } + // Increased delay between attempts to 5 seconds + await new Promise(resolve => setTimeout(resolve, 5000)); + } + + if (!healthy) { + // Get container logs to help diagnose the issue + const container = await this.docker.getContainer(config.name); + const logs = await container.logs({ + stdout: true, + stderr: true, + tail: 50 + }); + console.error(`Container logs for ${config.name}:`, logs.toString()); + throw new Error(`Container ${config.name} failed health check after 5 attempts`); + } + + // Run optimization for ComfyUI container after it's healthy + if (config.name === 'clara_comfyui' && useGPURuntime) { + console.log('🚀 Running GPU optimizations for ComfyUI...'); + // Run optimization in background to not block startup + setTimeout(() => { + this.optimizeComfyUIContainer().catch(error => { + console.error('Optimization failed:', error.message); + }); + }, 10000); // Wait 10 seconds after health check + } + } catch (error) { + console.error(`Error starting ${config.name}:`, error); + throw error; + } + } + + async initializePullTimestamps() { + try { + if (!fs.existsSync(this.pullTimestampsPath)) { + const initialTimestamps = {}; + Object.keys(this.containers).forEach(key => { + initialTimestamps[this.containers[key].image] = 0; + }); + fs.writeFileSync(this.pullTimestampsPath, JSON.stringify(initialTimestamps, null, 2)); + } + } catch (error) { + console.error('Error initializing pull timestamps:', error); + } + } + + getPullTimestamps() { + try { + if (fs.existsSync(this.pullTimestampsPath)) { + return JSON.parse(fs.readFileSync(this.pullTimestampsPath, 'utf8')); + } + } catch (error) { + console.error('Error reading pull timestamps:', error); + } + return {}; + } + + updatePullTimestamp(imageName) { + try { + const timestamps = this.getPullTimestamps(); + timestamps[imageName] = Date.now(); + fs.writeFileSync(this.pullTimestampsPath, JSON.stringify(timestamps, null, 2)); + } catch (error) { + console.error('Error updating pull timestamp:', error); + } + } + + async checkImageUpdate(imageName) { + try { + // First try to inspect the local image + try { + await this.docker.getImage(imageName).inspect(); + } catch (error) { + // If image doesn't exist locally, we need to pull it + if (error.statusCode === 404) { + return true; + } + } + + // Try to pull the image to check for updates + return new Promise((resolve, reject) => { + this.docker.pull(imageName, (err, stream) => { + if (err) { + // If we can't pull, but have local image, use local + if (err.statusCode === 404) { + resolve(false); + return; + } + reject(err); + return; + } + + let needsUpdate = false; + + stream.on('data', (data) => { + const lines = data.toString().split('\n').filter(Boolean); + lines.forEach(line => { + try { + const parsed = JSON.parse(line); + // Check for "up to date" message + if (parsed.status && parsed.status.includes('up to date')) { + needsUpdate = false; + } + // Check for "downloading" or "extracting" which indicates an update + if (parsed.status && (parsed.status.includes('Downloading') || parsed.status.includes('Extracting'))) { + needsUpdate = true; + } + } catch (e) { + // Ignore parse errors + } + }); + }); + + stream.on('end', () => { + resolve(needsUpdate); + }); + + stream.on('error', (error) => { + console.error('Stream error during pull:', error); + resolve(true); // If we can't determine, assume update needed + }); + }); + }); + } catch (error) { + console.error('Error checking image update:', error); + return true; // If we can't determine, assume update needed + } + } + + shouldPullImage(imageName, forceCheck = false) { + try { + if (forceCheck) { + return this.checkImageUpdate(imageName); + } + + const timestamps = this.getPullTimestamps(); + const lastPull = timestamps[imageName] || 0; + const daysSinceLastPull = (Date.now() - lastPull) / (1000 * 60 * 60 * 24); + return daysSinceLastPull >= 10; + } catch (error) { + console.error('Error checking pull timestamp:', error); + return true; // Pull if there's an error reading timestamps + } + } + + /** + * Wrapper function with timeout to prevent indefinite hangs + */ + async withTimeout(promise, timeoutMs, operationName) { + return new Promise((resolve, reject) => { + const timeout = setTimeout(() => { + reject(new Error(`Operation '${operationName}' timed out after ${timeoutMs}ms`)); + }, timeoutMs); + + promise + .then(result => { + clearTimeout(timeout); + resolve(result); + }) + .catch(error => { + clearTimeout(timeout); + reject(error); + }); + }); + } + + /** + * Enhanced setup with timeout protection and better error handling + */ + async setup(selectedFeatures, statusCallback) { + const setupStartTime = Date.now(); + console.log('🚀 Starting Docker setup with timeout protection...'); + + try { + // Set overall timeout for setup (10 minutes) + return await this.withTimeout( + this.performSetup(selectedFeatures, statusCallback), + 10 * 60 * 1000, // 10 minutes + 'Docker Setup' + ); + } catch (error) { + const setupDuration = ((Date.now() - setupStartTime) / 1000).toFixed(2); + console.error(`❌ Docker setup failed after ${setupDuration}s:`, error); + + if (error.message.includes('timed out')) { + statusCallback(`Setup timed out after ${setupDuration}s. This may indicate a network or Docker issue.`, 'error'); + + // Provide recovery suggestions + statusCallback('💡 Try restarting Docker Desktop and Clara, or check your internet connection.', 'info'); + + // Attempt graceful cleanup + try { + console.log('🧹 Attempting cleanup after timeout...'); + await this.cleanupAfterTimeout(); + } catch (cleanupError) { + console.error('Cleanup after timeout failed:', cleanupError); + } + } else { + statusCallback(`Setup failed: ${error.message}`, 'error'); + } + + return false; + } + } + + /** + * Cleanup operations after a timeout + */ + async cleanupAfterTimeout() { + try { + // Stop any partially started containers + for (const [name, config] of Object.entries(this.containers)) { + // Skip ComfyUI container on macOS and Linux as it's not supported + if (name === 'comfyui' && (process.platform === 'darwin' || process.platform === 'linux')) { + console.log(`⏭️ Skipping ComfyUI container cleanup on ${process.platform} (not supported)`); + continue; + } + + try { + const container = await this.docker.getContainer(config.name); + const containerInfo = await container.inspect(); + if (containerInfo.State.Running) { + console.log(`🛑 Stopping partially started container: ${config.name}`); + await container.stop({ t: 10 }); // 10 second graceful stop + } + } catch (error) { + // Ignore errors during cleanup + console.log(`Cleanup: Could not stop ${config.name}:`, error.message); + } + } + } catch (error) { + console.error('Error during timeout cleanup:', error); + } + } + + /** + * Main setup logic (extracted to allow timeout wrapping) + */ + async performSetup(selectedFeatures, statusCallback) { + try { + if (!await this.isDockerRunning()) { + let dockerDownloadLink; + let installMessage; + switch (process.platform) { + case 'darwin': + dockerDownloadLink = 'https://desktop.docker.com/mac/main/arm64/Docker.dmg'; + installMessage = 'download Docker Desktop'; + break; + case 'win32': + dockerDownloadLink = 'https://desktop.docker.com/win/main/amd64/Docker%20Desktop%20Installer.exe'; + installMessage = 'download Docker Desktop'; + break; + case 'linux': + default: + dockerDownloadLink = 'https://docs.docker.com/engine/install/'; + installMessage = 'install Docker Engine'; + break; + } + const errorMessage = `Docker is not running. Please ${installMessage} for better experience with workflows and automation.\n\nDownload from: ${dockerDownloadLink}`; + statusCallback(errorMessage, 'warning'); + + // Return false but don't throw error - let the app continue without Docker + return false; + } + + // Get user's feature selections from global scope - conservative defaults during onboarding + const selectedFeatures = global.selectedFeatures || { + comfyUI: false, // Conservative default - only start if explicitly selected + n8n: false, // Conservative default - only start if explicitly selected + ragAndTts: false, // Conservative default - prevent unwanted Python backend downloads + claraCore: true // Always enable core functionality + }; + + statusCallback('Creating Docker network...'); + await this.createNetwork(); + + statusCallback('Creating Docker volumes for persistent storage...'); + await this.createDockerVolumes(); + + // Check if Ollama is running on the system (no container management) + const ollamaRunning = await this.checkOllamaAvailability(); + if (ollamaRunning) { + statusCallback('✓ Ollama detected and available at http://localhost:11434', 'success'); + } else { + statusCallback('⚠️ Ollama not detected. Please install Ollama manually if you want local AI models. Visit: https://ollama.com', 'warning'); + } + + // Check if this is first-time setup + const timestamps = this.getPullTimestamps(); + const isFirstTimeSetup = Object.values(timestamps).every(timestamp => timestamp === 0); + + if (isFirstTimeSetup) { + statusCallback(`🎉 Welcome to Clara! Setting up your AI environment for the first time...`, 'info', { percentage: 5 }); + statusCallback(`This initial setup will download several AI services (may require 1-3 GB). Subsequent startups will be much faster!`, 'info', { percentage: 10 }); + } + + // Automatically check for and install container updates + console.log('🔄 Calling autoUpdateContainers...'); + await this.autoUpdateContainers(statusCallback); + console.log('✅ autoUpdateContainers completed, continuing with setup...'); + + // Filter containers based on user selections + console.log('🔍 Filtering containers based on user selections...'); + const enabledContainers = {}; + for (const [name, config] of Object.entries(this.containers)) { + let shouldEnable = false; + + switch (name) { + case 'python': + // Python backend only if user selected TTS/RAG features + // During onboarding, respect user's explicit choice to prevent unwanted downloads + shouldEnable = selectedFeatures.ragAndTts || false; + if (!shouldEnable) { + console.log(`⏭️ Python backend disabled (TTS/RAG not selected by user)`); + statusCallback(`⏭️ Python backend disabled (TTS/RAG not selected by user)`, 'info'); + } + break; + case 'n8n': + // N8N only if user selected it + shouldEnable = selectedFeatures.n8n; + break; + case 'comfyui': + // ComfyUI only if user selected it AND platform supports it (Windows only) + shouldEnable = selectedFeatures.comfyUI && process.platform === 'win32'; + if (selectedFeatures.comfyUI && process.platform !== 'win32') { + console.log(`⚠️ ComfyUI is not supported on ${process.platform} - requires Windows with NVIDIA GPU`); + statusCallback(`⚠️ ComfyUI is not supported on ${process.platform} - requires Windows with NVIDIA GPU`, 'warning'); + } + break; + default: + // Unknown services disabled by default + shouldEnable = false; + break; + } + + if (shouldEnable) { + enabledContainers[name] = config; + console.log(`✓ ${name} service enabled (selected by user)`); + statusCallback(`✓ ${name} service enabled (selected by user)`); + } else { + console.log(`⏭️ ${name} service disabled (not selected by user)`); + statusCallback(`⏭️ ${name} service disabled (not selected by user)`, 'info'); + } + } + console.log(`📦 Enabled containers: ${Object.keys(enabledContainers).join(', ')}`); + + // Resolve actual available image names and ensure all images are available locally + console.log('🔍 Resolving container images...'); + const resolvedContainers = {}; + for (const [name, config] of Object.entries(enabledContainers)) { + console.log(`🔍 Resolving image for ${name}...`); + + // Check if image is already architecture-specific (has -amd64 or -arm64 suffix) + const isAlreadyArchSpecific = config.image.includes('-amd64') || config.image.includes('-arm64'); + + let resolvedImageName; + if (isAlreadyArchSpecific) { + // Image is already architecture-specific, use as-is + resolvedImageName = config.image; + console.log(`✓ Using architecture-specific ${name} image: ${resolvedImageName}`); + statusCallback(`Using architecture-specific ${name} image: ${resolvedImageName}`); + } else { + // Parse base image and tag and resolve the actual available image name + const [baseImage, tag] = config.image.split(':'); + console.log(`🔍 Resolving ${name} image from base: ${baseImage}:${tag || 'latest'}`); + statusCallback(`Resolving ${name} image...`); + resolvedImageName = await this.resolveImageName(baseImage, tag || 'latest'); + console.log(`✓ Resolved ${name} image: ${resolvedImageName}`); + } + + // Create updated config with resolved image name + resolvedContainers[name] = { + ...config, + image: resolvedImageName + }; + + console.log(`🔍 Checking if ${name} image is available locally...`); + try { + await this.docker.getImage(resolvedImageName).inspect(); + console.log(`✓ ${name} image ready (${resolvedImageName})`); + statusCallback(`✓ ${name} image ready (${resolvedImageName})`); + } catch (error) { + if (error.statusCode === 404) { + console.log(`📥 ${name} image not found locally, downloading...`); + statusCallback(`Downloading ${name} image (${resolvedImageName})...`); + await this.pullImageWithProgress(resolvedImageName, statusCallback); + console.log(`✅ ${name} image downloaded successfully`); + } else { + console.error(`❌ Error checking ${name} image:`, error); + throw error; + } + } + } + + // Update containers configuration with resolved image names + console.log('📝 Updating containers configuration...'); + this.containers = resolvedContainers; + console.log(`✅ Container configuration updated with ${Object.keys(resolvedContainers).length} containers`); + + // Start containers in sequence + console.log('🚀 Starting containers...'); + for (const [name, config] of Object.entries(this.containers)) { + // Skip ComfyUI container on macOS and Linux as it's not supported + if (name === 'comfyui' && (process.platform === 'darwin' || process.platform === 'linux')) { + console.log(`⏭️ Skipping ComfyUI container startup on ${process.platform} (not supported)`); + continue; + } + + console.log(`🚀 Starting ${name} service...`); + statusCallback(`Starting ${name} service...`); + + try { + await this.startContainer(config); + console.log(`✅ ${name} service started successfully`); + statusCallback(`✅ ${name} service started successfully`); + } catch (error) { + console.error(`❌ Failed to start ${name} service:`, error); + statusCallback(`❌ Failed to start ${name} service: ${error.message}`, 'error'); + throw error; + } + } + + console.log('✅ All services started successfully'); + statusCallback('All services started successfully'); + return true; + } catch (error) { + statusCallback(`Setup failed: ${error.message}`, 'error'); + // Return false instead of throwing to allow app to continue + return false; + } + } + + async stop() { + try { + for (const [name, config] of Object.entries(this.containers)) { + // Skip ComfyUI container on macOS and Linux as it's not supported + if (name === 'comfyui' && (process.platform === 'darwin' || process.platform === 'linux')) { + console.log(`⏭️ Skipping ComfyUI container stop on ${process.platform} (not supported)`); + continue; + } + + try { + const container = await this.docker.getContainer(config.name); + await container.stop(); + await container.remove(); + } catch (error) { + // Ignore errors if container doesn't exist + } + } + + // Clean up network + try { + const network = await this.docker.getNetwork('clara_network'); + await network.remove(); + } catch (error) { + // Ignore network removal errors + } + } catch (error) { + console.error('Error stopping services:', error); + throw error; + } + } + + async cleanupDockerVolumes() { + try { + console.log('Cleaning up Clara-managed Docker volumes...'); + + // Get list of existing volumes + const volumes = await this.docker.listVolumes(); + if (!volumes.Volumes) { + console.log('No volumes to clean up'); + return; + } + + // Find Clara-managed volumes + const claraVolumes = volumes.Volumes.filter(vol => + vol.Labels && vol.Labels['clara.managed'] === 'true' + ); + + // Remove Clara-managed volumes + for (const volume of claraVolumes) { + try { + const dockerVolume = this.docker.getVolume(volume.Name); + await dockerVolume.remove(); + console.log(`✓ Removed Docker volume: ${volume.Name}`); + } catch (error) { + console.error(`Error removing volume ${volume.Name}:`, error.message); + // Don't throw here, continue with other volumes + } + } + + if (claraVolumes.length === 0) { + console.log('No Clara-managed volumes found to clean up'); + } + + } catch (error) { + console.error('Error in cleanupDockerVolumes:', error.message); + // Don't throw here to allow the application to continue + } + } + + async isPythonRunning() { + try { + if (!this.ports.python) { + return false; + } + + const response = await new Promise((resolve, reject) => { + const req = http.get(`http://localhost:${this.ports.python}/health`, (res) => { + if (res.statusCode === 200) { + let data = ''; + res.on('data', chunk => { + data += chunk; + }); + res.on('end', () => { + try { + const jsonResponse = JSON.parse(data); + const isHealthy = jsonResponse.status === 'healthy' || jsonResponse.status === 'ok'; + resolve(isHealthy); + } catch (e) { + console.error('Failed to parse health check JSON:', e); + resolve(false); + } + }); + } else { + reject(new Error(`Python health check failed with status ${res.statusCode}`)); + } + }); + + req.on('error', (error) => { + // Only log non-connection errors (ECONNREFUSED is expected when service is down) + if (error.code !== 'ECONNREFUSED') { + console.error('Python health check request error:', error); + } + resolve(false); + }); + + req.setTimeout(5000, () => { + // Timeout is expected when service is not responding + req.destroy(); + resolve(false); + }); + }); + + return response; + } catch (error) { + // Only log unexpected errors + if (error.code !== 'ECONNREFUSED') { + console.error('Python health check error:', error); + } + return false; + } + } + + async checkN8NHealth() { + try { + const response = await new Promise((resolve, reject) => { + http.get(`http://localhost:${this.ports.n8n}/healthz`, (res) => { + if (res.statusCode === 200) { + resolve({ success: true }); + } else { + reject(new Error(`N8N health check failed with status ${res.statusCode}`)); + } + }).on('error', (error) => { + // Only log non-connection errors (ECONNREFUSED is expected when service is down) + if (error.code !== 'ECONNREFUSED') { + reject(error); + } else { + resolve({ success: false, error: 'Service not running' }); + } + }); + }); + return response; + } catch (error) { + return { success: false, error: error.message }; + } + } + + async checkOllamaAvailability() { + try { + const response = await new Promise((resolve, reject) => { + const req = http.get('http://localhost:11434/api/tags', (res) => { + if (res.statusCode === 200) { + resolve(true); + } else { + resolve(false); + } + }); + + req.on('error', () => resolve(false)); + + // Add timeout to avoid hanging + req.setTimeout(3000, () => { + req.destroy(); + resolve(false); + }); + }); + + return response; + } catch (error) { + return false; + } + } + + /** + * Manual update check that can be called from the main process + */ + async checkForUpdates(statusCallback) { + try { + if (!await this.isDockerRunning()) { + throw new Error('Docker is not running'); + } + + statusCallback(`Checking for container updates (${this.systemArch})...`); + const updateChecks = []; + + for (const [name, config] of Object.entries(this.containers)) { + // Skip ComfyUI container on macOS and Linux as it's not supported + if (name === 'comfyui' && (process.platform === 'darwin' || process.platform === 'linux')) { + console.log(`⏭️ Skipping ComfyUI container update check on ${process.platform} (not supported)`); + continue; + } + + // Use enhanced update checking + updateChecks.push( + this.checkForImageUpdatesForced(config.image, statusCallback) + .then(result => ({ ...result, containerName: name })) + ); + } + + const updateResults = await Promise.all(updateChecks); + const updatesAvailable = updateResults.filter(result => result.hasUpdate); + + return { + updatesAvailable: updatesAvailable.length > 0, + updates: updateResults, + architecture: this.systemArch + }; + } catch (error) { + console.error('Error checking for updates:', error); + throw error; + } + } + + /** + * Forced update check that's more aggressive in detecting updates + */ + async checkForImageUpdatesForced(imageName, statusCallback) { + try { + statusCallback(`Force checking updates for ${imageName}...`); + + // First, try the regular update check + const regularCheck = await this.checkForImageUpdates(imageName, statusCallback); + + // If the regular check says no update, but we want to be more thorough, + // let's try a different approach + if (!regularCheck.hasUpdate) { + console.log(`Regular check found no updates for ${imageName}, trying forced check...`); + + // Try to get remote manifest information + try { + const manifestResult = await this.checkImageManifestForced(imageName); + if (manifestResult.hasUpdate) { + return { + ...regularCheck, + hasUpdate: true, + reason: 'Forced check detected newer manifest', + forcedCheck: true + }; + } + } catch (error) { + console.log('Forced manifest check failed:', error.message); + } + } + + return regularCheck; + } catch (error) { + console.error('Error in forced update check:', error); + return { + hasUpdate: false, + reason: 'Error in forced update check', + error: error.message, + imageName + }; + } + } + + /** + * Check image manifest for updates more aggressively + */ + async checkImageManifestForced(imageName) { + return new Promise((resolve) => { + // Force pull with explicit latest tag to check for updates + const latestImageName = imageName.includes(':') ? imageName : `${imageName}:latest`; + + console.log(`Forced manifest check for ${latestImageName}`); + + // Try pulling to see if there are any new layers + this.docker.pull(latestImageName, { + platform: this.systemArch, + // Try to bypass cache by using different options + }, (err, stream) => { + if (err) { + console.log('Forced pull failed:', err.message); + resolve({ hasUpdate: false, reason: 'Forced pull failed' }); + return; + } + + let downloadActivity = false; + let layerActivity = false; + + stream.on('data', (data) => { + const lines = data.toString().split('\n').filter(Boolean); + lines.forEach(line => { + try { + const parsed = JSON.parse(line); + if (parsed.status) { + // Look for any kind of activity that suggests there might be updates + if (parsed.status.includes('Pulling') || + parsed.status.includes('Downloading') || + parsed.status.includes('Extracting') || + parsed.progressDetail) { + downloadActivity = true; + } + if (parsed.status.includes('Already exists') || + parsed.status.includes('Pull complete')) { + layerActivity = true; + } + } + } catch (e) { + // Ignore + } + }); + }); + + stream.on('end', () => { + const hasUpdate = downloadActivity || (!layerActivity && !downloadActivity); + resolve({ + hasUpdate, + reason: hasUpdate ? 'Forced check suggests update available' : 'Forced check confirms up to date' + }); + }); + + stream.on('error', () => { + resolve({ hasUpdate: false, reason: 'Forced check stream error' }); + }); + }); + }); + } + + /** + * Update specific containers + */ + async updateContainers(containerNames, statusCallback) { + try { + if (!await this.isDockerRunning()) { + throw new Error('Docker is not running'); + } + + const containersToUpdate = containerNames || Object.keys(this.containers); + const results = []; + + for (const containerName of containersToUpdate) { + // Skip ComfyUI container on macOS and Linux as it's not supported + if (containerName === 'comfyui' && (process.platform === 'darwin' || process.platform === 'linux')) { + console.log(`⏭️ Skipping ComfyUI container update on ${process.platform} (not supported)`); + results.push({ + container: containerName, + success: false, + error: `ComfyUI is not supported on ${process.platform}` + }); + continue; + } + + const config = this.containers[containerName]; + if (!config) { + results.push({ + container: containerName, + success: false, + error: 'Container not found' + }); + continue; + } + + try { + statusCallback(`Updating ${containerName}...`); + + // Stop and remove existing container + try { + const existingContainer = await this.docker.getContainer(config.name); + const containerInfo = await existingContainer.inspect(); + + if (containerInfo.State.Running) { + await existingContainer.stop(); + } + await existingContainer.remove({ force: true }); + statusCallback(`Stopped and removed old ${containerName} container`); + } catch (error) { + // Container might not exist, which is fine + if (error.statusCode !== 404) { + console.warn(`Warning removing old container ${config.name}:`, error.message); + } + } + + // Pull latest image + await this.pullImageWithProgress(config.image, statusCallback); + + // Start new container + await this.startContainer(config); + + results.push({ + container: containerName, + success: true, + message: `Successfully updated ${containerName}` + }); + + statusCallback(`✓ ${containerName} updated successfully`); + } catch (error) { + results.push({ + container: containerName, + success: false, + error: error.message + }); + statusCallback(`✗ Failed to update ${containerName}: ${error.message}`, 'error'); + } + } + + return results; + } catch (error) { + console.error('Error updating containers:', error); + throw error; + } + } + + async isComfyUIRunning() { + try { + if (!this.ports.comfyui) { + return false; + } + + const response = await new Promise((resolve, _reject) => { + const req = http.get(`http://localhost:${this.ports.comfyui}/`, (res) => { + // ComfyUI returns 200 for the main page when running + if (res.statusCode === 200) { + resolve(true); + } else { + resolve(false); + } + }); + + req.on('error', (error) => { + // Only log non-connection errors (ECONNREFUSED is expected when service is down) + if (error.code !== 'ECONNREFUSED') { + console.error('ComfyUI health check request error:', error); + } + resolve(false); + }); + + req.setTimeout(5000, () => { + // Timeout is expected when service is not responding + req.destroy(); + resolve(false); + }); + }); + + return response; + } catch (error) { + // Only log unexpected errors + if (error.code !== 'ECONNREFUSED') { + console.error('ComfyUI health check error:', error); + } + return false; + } + } + + /** + * Resolve the actual available image name by testing different variants + */ + async resolveImageName(baseImage, tag) { + // For clara-backend, we need to test which variant is actually available + if (baseImage === 'clara17verse/clara-backend') { + const arch = os.arch(); + + // List of image variants to try in order of preference + const imageVariants = []; + + if (arch === 'arm64') { + // For ARM64, prefer base image first, then amd64 as fallback + imageVariants.push(`${baseImage}:${tag}`); + imageVariants.push(`${baseImage}:${tag}-amd64`); + } else { + // For x64/AMD64, prefer amd64 image first, then base as fallback + imageVariants.push(`${baseImage}:${tag}-amd64`); + imageVariants.push(`${baseImage}:${tag}`); + } + + // Test each variant to see which one exists + for (const imageName of imageVariants) { + try { + console.log(`Testing availability of image: ${imageName}`); + + // Try to inspect the image locally first + try { + await this.docker.getImage(imageName).inspect(); + console.log(`Found local image: ${imageName}`); + return imageName; + } catch (localError) { + // Image not local, try to pull manifest to check if it exists remotely + const manifestExists = await this.checkImageManifest(imageName); + if (manifestExists) { + console.log(`Remote image available: ${imageName}`); + return imageName; + } + } + } catch (error) { + console.log(`Image not available: ${imageName} - ${error.message}`); + continue; + } + } + + // If no specific variant works, return the original preference + const fallbackImage = arch === 'arm64' ? `${baseImage}:${tag}` : `${baseImage}:${tag}-amd64`; + console.log(`No variants found, using fallback: ${fallbackImage}`); + return fallbackImage; + } + + // For other images, return as-is + return `${baseImage}:${tag}`; + } + + /** + * Check if an image manifest exists without pulling the full image + */ + async checkImageManifest(imageName) { + return new Promise((resolve) => { + // Use a quick pull with dry-run-like behavior + this.docker.pull(imageName, {}, (err, stream) => { + if (err) { + resolve(false); + return; + } + + let manifestFound = false; + + stream.on('data', (data) => { + const lines = data.toString().split('\n').filter(Boolean); + lines.forEach(line => { + try { + const parsed = JSON.parse(line); + // If we get any valid status, the manifest exists + if (parsed.status && !parsed.error) { + manifestFound = true; + } + } catch (e) { + // Ignore parse errors + } + }); + }); + + stream.on('end', () => { + resolve(manifestFound); + }); + + stream.on('error', () => { + resolve(false); + }); + + // Stop the stream early since we just want to check manifest + setTimeout(() => { + try { + stream.destroy(); + } catch (e) { + // Ignore destroy errors + } + resolve(manifestFound); + }, 5000); + }); + }); + } + + /** + * Load connection configuration from disk + * Returns stored remote Docker configuration or null for local + */ + loadConnectionConfig() { + try { + const configPath = path.join(this.appDataPath, 'docker-connection.json'); + if (fs.existsSync(configPath)) { + const config = JSON.parse(fs.readFileSync(configPath, 'utf8')); + console.log(`📁 Loaded Docker connection config: ${config.mode} mode`); + return config; + } + } catch (error) { + console.error('Error loading connection config:', error); + } + return null; + } + + /** + * Save connection configuration to disk + */ + saveConnectionConfig(config) { + try { + const configPath = path.join(this.appDataPath, 'docker-connection.json'); + fs.writeFileSync(configPath, JSON.stringify(config, null, 2), 'utf8'); + console.log(`💾 Saved Docker connection config: ${config.mode} mode`); + return true; + } catch (error) { + console.error('Error saving connection config:', error); + return false; + } + } + + /** + * Switch between local and remote Docker connections + */ + async switchConnection(newConfig) { + try { + console.log(`🔄 Switching Docker connection to ${newConfig.mode} mode...`); + + // Close existing SSH tunnels if switching from remote + if (this.connectionMode === 'remote') { + await this.closeAllSSHTunnels(); + } + + // Update configuration + this.connectionConfig = newConfig; + this.connectionMode = newConfig.mode; + + // Reinitialize Docker client + this.docker = this.initializeDockerClient(); + + // Test connection + const isRunning = await this.isDockerRunning(); + if (!isRunning) { + throw new Error('Failed to connect to Docker with new configuration'); + } + + // Save configuration + this.saveConnectionConfig(newConfig); + + console.log(`✅ Successfully switched to ${newConfig.mode} mode`); + return { success: true, mode: newConfig.mode }; + } catch (error) { + console.error('Error switching Docker connection:', error); + return { success: false, error: error.message }; + } + } + + /** + * Create SSH tunnel for a specific port + * This allows remote services to be accessed as if they were local + */ + async createSSHTunnel(localPort, remotePort, serviceName) { + if (this.connectionMode !== 'remote' || !this.connectionConfig) { + console.log('SSH tunnels only needed for remote mode'); + return; + } + + return new Promise((resolve, reject) => { + try { + const { host, username, sshKeyPath, port: sshPort = 22 } = this.connectionConfig; + + console.log(`🔗 Creating SSH tunnel for ${serviceName}: localhost:${localPort} -> ${host}:${remotePort}`); + + // Build SSH command for port forwarding + const sshArgs = [ + '-N', // No remote command + '-L', `${localPort}:localhost:${remotePort}`, // Local port forwarding + '-p', sshPort.toString(), + '-o', 'StrictHostKeyChecking=no', + '-o', 'ServerAliveInterval=60', + '-o', 'ServerAliveCountMax=3' + ]; + + // Add SSH key if provided + if (sshKeyPath && fs.existsSync(sshKeyPath)) { + sshArgs.push('-i', sshKeyPath); + } + + sshArgs.push(`${username}@${host}`); + + // Spawn SSH tunnel process + const tunnel = spawn('ssh', sshArgs); + + tunnel.stdout.on('data', (data) => { + console.log(`SSH tunnel ${serviceName} stdout:`, data.toString()); + }); + + tunnel.stderr.on('data', (data) => { + const message = data.toString(); + // SSH outputs connection info to stderr, not always errors + if (!message.includes('Warning') && !message.includes('Authenticated')) { + console.error(`SSH tunnel ${serviceName} stderr:`, message); + } + }); + + tunnel.on('error', (error) => { + console.error(`SSH tunnel ${serviceName} error:`, error); + reject(error); + }); + + tunnel.on('close', (code) => { + console.log(`SSH tunnel ${serviceName} closed with code ${code}`); + delete this.sshTunnels[serviceName]; + this.activeTunnels = this.activeTunnels.filter(t => t.service !== serviceName); + }); + + // Store tunnel reference + this.sshTunnels[serviceName] = tunnel; + this.activeTunnels.push({ + service: serviceName, + localPort, + remotePort, + process: tunnel + }); + + // Give SSH a moment to establish the tunnel + setTimeout(() => { + console.log(`✅ SSH tunnel for ${serviceName} established`); + resolve(tunnel); + }, 2000); + + } catch (error) { + console.error(`Failed to create SSH tunnel for ${serviceName}:`, error); + reject(error); + } + }); + } + + /** + * Close a specific SSH tunnel + */ + async closeSSHTunnel(serviceName) { + const tunnel = this.sshTunnels[serviceName]; + if (tunnel) { + console.log(`🔌 Closing SSH tunnel for ${serviceName}...`); + tunnel.kill(); + delete this.sshTunnels[serviceName]; + this.activeTunnels = this.activeTunnels.filter(t => t.service !== serviceName); + } + } + + /** + * Close all SSH tunnels + */ + async closeAllSSHTunnels() { + console.log('🔌 Closing all SSH tunnels...'); + for (const serviceName of Object.keys(this.sshTunnels)) { + await this.closeSSHTunnel(serviceName); + } + } + + /** + * Get list of active SSH tunnels + */ + getActiveTunnels() { + return this.activeTunnels.map(t => ({ + service: t.service, + localPort: t.localPort, + remotePort: t.remotePort, + isActive: !t.process.killed + })); + } + + /** + * Test remote Docker connection + * Useful for validating configuration before saving + */ + async testRemoteConnection(config) { + try { + console.log(`🧪 Testing remote Docker connection to ${config.host}...`); + + // Create a temporary Docker client + const testClient = this.createRemoteDockerClient(config); + + // Try to ping the Docker daemon + await testClient.ping(); + + // Get Docker version info + const version = await testClient.version(); + + console.log(`✅ Remote Docker connection successful! Version: ${version.Version}`); + return { + success: true, + version: version.Version, + apiVersion: version.ApiVersion, + platform: version.Platform?.Name || 'Unknown' + }; + } catch (error) { + console.error('❌ Remote Docker connection test failed:', error); + return { + success: false, + error: error.message + }; + } + } + + /** + * Get connection info (for UI display) + */ + getConnectionInfo() { + return { + mode: this.connectionMode, + config: this.connectionConfig, + activeTunnels: this.getActiveTunnels(), + isRemote: this.connectionMode === 'remote' + }; + } +} + +module.exports = DockerSetup; \ No newline at end of file diff --git a/electron/featureSelection.cjs b/electron/featureSelection.cjs new file mode 100644 index 00000000..b5b5e8cf --- /dev/null +++ b/electron/featureSelection.cjs @@ -0,0 +1,275 @@ +const { BrowserWindow, app, screen, ipcMain } = require('electron'); +const path = require('path'); +const fs = require('fs'); +const yaml = require('js-yaml'); + +class FeatureSelectionScreen { + constructor() { + this.window = null; + this.configPath = path.join(app.getPath('userData'), 'clara-features.yaml'); + + // Default feature configuration + this.defaultConfig = { + version: '1.0.0', + firstTimeSetup: true, + selectedFeatures: { + comfyUI: true, + n8n: true, + ragAndTts: true, + claraCore: true // Always enabled + }, + setupTimestamp: null + }; + } + + /** + * Check if this is the first time launch + */ + isFirstTimeLaunch() { + try { + if (!fs.existsSync(this.configPath)) { + return true; + } + + const configContent = fs.readFileSync(this.configPath, 'utf8'); + const config = yaml.load(configContent); + + return config?.firstTimeSetup === true; + } catch (error) { + console.error('Error checking first time launch:', error); + return true; // Default to first time if we can't read config + } + } + + /** + * Load existing feature configuration + */ + loadConfig() { + try { + if (!fs.existsSync(this.configPath)) { + return this.defaultConfig; + } + + const configContent = fs.readFileSync(this.configPath, 'utf8'); + const config = yaml.load(configContent); + + // Merge with defaults to ensure all properties exist + return { + ...this.defaultConfig, + ...config, + selectedFeatures: { + ...this.defaultConfig.selectedFeatures, + ...config.selectedFeatures + } + }; + } catch (error) { + console.error('Error loading feature config:', error); + return this.defaultConfig; + } + } + + /** + * Save feature configuration to YAML file + */ + saveConfig(config) { + try { + const yamlContent = yaml.dump(config, { + indent: 2, + lineWidth: 120, + noRefs: true + }); + + fs.writeFileSync(this.configPath, yamlContent, 'utf8'); + console.log('✅ Feature configuration saved to:', this.configPath); + return true; + } catch (error) { + console.error('❌ Error saving feature configuration:', error); + return false; + } + } + + /** + * Create and show the feature selection window + */ + async show() { + return new Promise((resolve, reject) => { + try { + const isDev = process.env.NODE_ENV === 'development'; + + // Get primary display dimensions + const primaryDisplay = screen.getPrimaryDisplay(); + const { width, height } = primaryDisplay.workAreaSize; + + this.window = new BrowserWindow({ + width: Math.max(600, Math.min(900, width * 0.8)), + height: Math.max(500, Math.min(700, height * 0.85)), + minWidth: 500, + minHeight: 450, + center: true, + frame: false, + transparent: false, + resizable: true, + alwaysOnTop: true, + show: false, + webPreferences: { + nodeIntegration: true, + contextIsolation: false, + enableRemoteModule: true + }, + skipTaskbar: false + }); + + // Load current configuration + const currentConfig = this.loadConfig(); + + // Handle window events + this.window.once('ready-to-show', () => { + this.window.show(); + // Send current configuration to renderer + this.window.webContents.send('load-config', currentConfig); + }); + + // Track if we're closing intentionally + let intentionalClose = false; + + this.window.on('closed', () => { + this.window = null; + // Only reject if the window was closed without completing selection + if (!intentionalClose) { + reject(new Error('Feature selection window was closed')); + } + }); + + // Handle feature selection completion + ipcMain.once('feature-selection-complete', (event, selectedFeatures) => { + console.log('Received feature-selection-complete event:', selectedFeatures); + try { + // Create final configuration + const finalConfig = { + ...currentConfig, + firstTimeSetup: false, + selectedFeatures: { + claraCore: true, // Always enabled + ...selectedFeatures + }, + setupTimestamp: new Date().toISOString() + }; + + // Save configuration + const saved = this.saveConfig(finalConfig); + if (saved) { + console.log('✅ Feature selection completed:', finalConfig.selectedFeatures); + + // Mark as intentional close + intentionalClose = true; + + // Resolve the promise first + resolve(finalConfig.selectedFeatures); + + // Then close the window after a small delay + setTimeout(() => { + console.log('Closing feature selection window after successful save...'); + if (this.window && !this.window.isDestroyed()) { + this.window.destroy(); // Use destroy instead of close to force it + } + }, 100); + } else { + console.error('Failed to save feature configuration'); + intentionalClose = true; + reject(new Error('Failed to save feature configuration')); + + // Close window after rejection + setTimeout(() => { + if (this.window && !this.window.isDestroyed()) { + this.window.destroy(); + } + }, 100); + } + } catch (error) { + console.error('Error handling feature selection:', error); + intentionalClose = true; + reject(error); + + // Close window after error + setTimeout(() => { + if (this.window && !this.window.isDestroyed()) { + this.window.destroy(); + } + }, 100); + } + }); + + // Handle window close request + ipcMain.once('close-feature-selection', () => { + console.log('Received close-feature-selection request'); + intentionalClose = true; + reject(new Error('Feature selection was cancelled')); + + // Force close the window + setTimeout(() => { + if (this.window && !this.window.isDestroyed()) { + this.window.destroy(); + } + }, 100); + }); + + // Load the HTML file + const htmlPath = isDev + ? path.join(__dirname, 'featureSelection.html') + : path.join(app.getAppPath(), 'electron', 'featureSelection.html'); + + console.log('Loading feature selection from:', htmlPath); + this.window.loadFile(htmlPath); + + } catch (error) { + console.error('Error creating feature selection window:', error); + reject(error); + } + }); + } + + /** + * Close the feature selection window + */ + close() { + console.log('Attempting to close feature selection window...'); + if (this.window && !this.window.isDestroyed()) { + // Remove IPC listeners + ipcMain.removeAllListeners('feature-selection-complete'); + ipcMain.removeAllListeners('close-feature-selection'); + + try { + this.window.close(); + console.log('Feature selection window closed successfully'); + } catch (error) { + console.error('Error closing feature selection window:', error); + } + this.window = null; + } else { + console.log('Feature selection window already closed or destroyed'); + } + } + + /** + * Get the current feature configuration (for use by main process) + */ + static getCurrentConfig() { + const configPath = path.join(app.getPath('userData'), 'clara-features.yaml'); + + try { + if (!fs.existsSync(configPath)) { + return null; + } + + const configContent = fs.readFileSync(configPath, 'utf8'); + const config = yaml.load(configContent); + + return config?.selectedFeatures || null; + } catch (error) { + console.error('Error loading current feature config:', error); + return null; + } + } +} + +module.exports = FeatureSelectionScreen; \ No newline at end of file diff --git a/electron/featureSelection.html b/electron/featureSelection.html new file mode 100644 index 00000000..34f13d0c --- /dev/null +++ b/electron/featureSelection.html @@ -0,0 +1,513 @@ + + + + + + ClaraVerse - Feature Selection + + + + + + + + + + + + + + + + + + + + + +
+ +
+ +
+
+ +
+

+ ClaraVerse +

+
+ + +
+

+ Welcome to Your AI Universe +

+

+ Choose the features you want to enable. You can change these settings later in the application preferences. +

+
+
+ + +
+

+ Clara Core (AI Assistant & Chat) is always enabled and provides the foundation for all other features. +

+
+ + +
+
+ +
+ +
+ +
+ + +
+
+ +
+
+

+ ComfyUI +

+

+ AI Image Generation +

+
+
+

+ Advanced AI image generation with Stable Diffusion, ControlNet, and custom workflows. + Create stunning artwork, edit images, and experiment with AI-powered creativity. +

+
+ + +
+ +
+ +
+ + +
+
+ +
+
+

+ N8N +

+

+ Workflow Automation +

+
+
+

+ Powerful workflow automation platform with 400+ integrations. + Connect apps, automate tasks, and build complex workflows without coding. +

+
+ + +
+ +
+ +
+ + +
+
+ +
+
+

+ RAG & TTS +

+

+ Advanced AI Features +

+
+
+

+ Retrieval-Augmented Generation for document analysis and Text-to-Speech capabilities. + Chat with your documents and enjoy natural voice synthesis. +

+
+
+
+ + +
+ Click on features to enable or disable them. Selected features will be downloaded and configured during startup. +
+ + +
+ + +
+
+ + + + \ No newline at end of file diff --git a/electron/ipcLogger.cjs b/electron/ipcLogger.cjs new file mode 100644 index 00000000..b69cae28 --- /dev/null +++ b/electron/ipcLogger.cjs @@ -0,0 +1,247 @@ +const fs = require('fs'); +const path = require('path'); +const { app } = require('electron'); + +class IPCLogger { + constructor() { + // Create logs directory in userData + this.logsDir = path.join(app.getPath('userData'), 'logs'); + this.logFile = path.join(this.logsDir, 'ipc.log'); + this.maxFileSize = 100 * 1024 * 1024; // 100MB + this.maxBackups = 5; // Keep 5 backup files + + this.ensureLogsDirectory(); + this.initializeLogFile(); + } + + ensureLogsDirectory() { + if (!fs.existsSync(this.logsDir)) { + fs.mkdirSync(this.logsDir, { recursive: true }); + } + } + + initializeLogFile() { + if (!fs.existsSync(this.logFile)) { + fs.writeFileSync(this.logFile, ''); + } + } + + formatLogEntry(type, channel, data, direction = 'incoming') { + const timestamp = new Date().toISOString(); + const arrow = direction === 'incoming' ? '→' : '←'; + + let dataStr = ''; + if (data !== undefined) { + try { + dataStr = typeof data === 'object' ? JSON.stringify(data, null, 2) : String(data); + } catch (error) { + dataStr = '[Circular or Non-serializable Object]'; + } + } + + return `${timestamp} [${type}] ${arrow} ${channel}${dataStr ? `\n Data: ${dataStr}` : ''}\n`; + } + + checkFileSize() { + try { + const stats = fs.statSync(this.logFile); + if (stats.size >= this.maxFileSize) { + this.rotateLogFile(); + } + } catch (error) { + console.error('Error checking log file size:', error); + } + } + + rotateLogFile() { + try { + // Rotate existing backup files + for (let i = this.maxBackups - 1; i >= 1; i--) { + const oldFile = `${this.logFile}.${i}`; + const newFile = `${this.logFile}.${i + 1}`; + + if (fs.existsSync(oldFile)) { + if (i === this.maxBackups - 1) { + fs.unlinkSync(oldFile); // Delete oldest backup + } else { + fs.renameSync(oldFile, newFile); + } + } + } + + // Move current log to .1 + if (fs.existsSync(this.logFile)) { + fs.renameSync(this.logFile, `${this.logFile}.1`); + } + + // Create new empty log file + fs.writeFileSync(this.logFile, ''); + + this.log('SYSTEM', 'log-rotation', { message: 'Log file rotated due to size limit' }); + } catch (error) { + console.error('Error rotating log file:', error); + } + } + + log(type, channel, data, direction = 'incoming') { + try { + this.checkFileSize(); + const logEntry = this.formatLogEntry(type, channel, data, direction); + fs.appendFileSync(this.logFile, logEntry); + } catch (error) { + console.error('Error writing to log file:', error); + } + } + + logIPC(channel, data, direction = 'incoming') { + this.log('IPC', channel, data, direction); + } + + logError(channel, error, direction = 'incoming') { + this.log('ERROR', channel, { + message: error.message, + stack: error.stack, + code: error.code + }, direction); + } + + logSystem(message, data = null) { + this.log('SYSTEM', 'system-event', { message, data }); + } + + // Enhanced logging methods for different communication types + logServiceCall(serviceName, methodName, args = null, result = null) { + this.log('SERVICE', `${serviceName}.${methodName}`, { + args: args, + result: result + }, 'internal'); + } + + logProcessSpawn(command, args, options = {}) { + this.log('PROCESS', 'spawn', { + command: command, + args: args, + options: options + }, 'outgoing'); + } + + logProcessExit(command, exitCode, signal = null) { + this.log('PROCESS', 'exit', { + command: command, + exitCode: exitCode, + signal: signal + }, 'incoming'); + } + + logHttpRequest(method, url, options = {}) { + this.log('HTTP', `${method} ${url}`, { + options: options + }, 'outgoing'); + } + + logHttpResponse(method, url, statusCode, responseData = null) { + this.log('HTTP', `${method} ${url}`, { + statusCode: statusCode, + response: responseData + }, 'incoming'); + } + + logDockerOperation(operation, containerName, data = null) { + this.log('DOCKER', `${operation}:${containerName}`, data, 'outgoing'); + } + + logDockerResult(operation, containerName, result) { + this.log('DOCKER', `${operation}:${containerName}`, result, 'incoming'); + } + + logWatchdogEvent(eventType, serviceName, data = null) { + this.log('WATCHDOG', `${eventType}:${serviceName}`, data, 'internal'); + } + + logFileOperation(operation, filePath, data = null) { + this.log('FILE', `${operation}:${path.basename(filePath)}`, { + path: filePath, + data: data + }, 'internal'); + } + + async readLogs(lines = 1000) { + try { + if (!fs.existsSync(this.logFile)) { + return ''; + } + + const data = fs.readFileSync(this.logFile, 'utf8'); + const logLines = data.split('\n'); + + // Return last N lines (or all if fewer than N) + const startIndex = Math.max(0, logLines.length - lines); + return logLines.slice(startIndex).join('\n'); + } catch (error) { + console.error('Error reading log file:', error); + return `Error reading log file: ${error.message}`; + } + } + + async getLogFiles() { + try { + const files = fs.readdirSync(this.logsDir) + .filter(file => file.startsWith('ipc.log')) + .map(file => { + const fullPath = path.join(this.logsDir, file); + const stats = fs.statSync(fullPath); + return { + name: file, + path: fullPath, + size: stats.size, + modified: stats.mtime, + isActive: file === 'ipc.log' + }; + }) + .sort((a, b) => { + // Sort with active file first, then by modification time (newest first) + if (a.isActive) return -1; + if (b.isActive) return 1; + return b.modified - a.modified; + }); + + return files; + } catch (error) { + console.error('Error getting log files:', error); + return []; + } + } + + async clearLogs() { + try { + // Remove all log files + const files = await this.getLogFiles(); + for (const file of files) { + fs.unlinkSync(file.path); + } + + // Create new empty log file + this.initializeLogFile(); + this.logSystem('Logs cleared by user'); + + return { success: true }; + } catch (error) { + console.error('Error clearing logs:', error); + return { success: false, error: error.message }; + } + } + + formatBytes(bytes, decimals = 2) { + if (bytes === 0) return '0 Bytes'; + + const k = 1024; + const dm = decimals < 0 ? 0 : decimals; + const sizes = ['Bytes', 'KB', 'MB', 'GB']; + + const i = Math.floor(Math.log(bytes) / Math.log(k)); + + return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + ' ' + sizes[i]; + } +} + +module.exports = IPCLogger; \ No newline at end of file diff --git a/electron/loading.html b/electron/loading.html new file mode 100644 index 00000000..9795f3d0 --- /dev/null +++ b/electron/loading.html @@ -0,0 +1,975 @@ + + + + + + + + +
+ + +
+ + +
+ + +
+ + +
+ +
+
+
+
+ +
+ +
Clara
+
Your AI Assistant
+ +
+
+
+
+
+ +
+
+
+ +
Initializing...
+
+ + + + + + \ No newline at end of file diff --git a/electron/loadingScreen.cjs b/electron/loadingScreen.cjs new file mode 100644 index 00000000..17dcfe33 --- /dev/null +++ b/electron/loadingScreen.cjs @@ -0,0 +1,118 @@ +const { BrowserWindow, app } = require('electron'); +const path = require('path'); +const fs = require('fs'); + +class LoadingScreen { + constructor() { + const isDev = process.env.NODE_ENV === 'development'; + + // Check fullscreen startup preference + let shouldStartFullscreen = false; + try { + const userDataPath = app.getPath('userData'); + const startupSettingsPath = path.join(userDataPath, 'clara-startup-settings.json'); + const legacySettingsPath = path.join(userDataPath, 'clara-settings.json'); + const veryLegacySettingsPath = path.join(userDataPath, 'settings.json'); + + if (fs.existsSync(startupSettingsPath)) { + const startupSettings = JSON.parse(fs.readFileSync(startupSettingsPath, 'utf8')); + shouldStartFullscreen = !!startupSettings.startFullscreen; + } else if (fs.existsSync(legacySettingsPath)) { + const legacySettings = JSON.parse(fs.readFileSync(legacySettingsPath, 'utf8')); + const legacyStartup = legacySettings.startup || {}; + shouldStartFullscreen = legacyStartup.startFullscreen ?? legacySettings.fullscreen_startup ?? false; + } else if (fs.existsSync(veryLegacySettingsPath)) { + const veryLegacySettings = JSON.parse(fs.readFileSync(veryLegacySettingsPath, 'utf8')); + shouldStartFullscreen = veryLegacySettings.startup?.startFullscreen ?? veryLegacySettings.fullscreen_startup ?? false; + } + } catch (error) { + console.error('Error reading fullscreen startup preference:', error); + } + + this.window = new BrowserWindow({ + fullscreen: shouldStartFullscreen, + frame: false, + transparent: false, + webPreferences: { + nodeIntegration: true, + contextIsolation: false + }, + skipTaskbar: true, + resizable: false, + alwaysOnTop: true, + show: false, + backgroundColor: '#667eea' + }); + + // Show window when ready to prevent flash + this.window.once('ready-to-show', () => { + this.window.show(); + }); + + // Log any errors + this.window.webContents.on('crashed', (e) => { + console.error('Loading screen crashed:', e); + }); + + this.window.webContents.on('did-fail-load', (event, code, description) => { + console.error('Failed to load loading screen:', code, description); + }); + + const htmlPath = isDev + ? path.join(__dirname, 'loading.html') + : path.join(app.getAppPath(), 'electron', 'loading.html'); + + console.log('Loading screen from:', htmlPath); + this.window.loadFile(htmlPath); + } + + setStatus(message, type = 'info', progress = null) { + if (!this.window) return; + + const data = { + message: message, + type: type, + progress: progress, + timestamp: new Date().toISOString() + }; + + console.log(`[Loading] Setting status:`, data); + this.window.webContents.send('status', data); + } + + // Notify that main window is ready + notifyMainWindowReady() { + if (this.window && !this.window.isDestroyed()) { + this.window.webContents.send('main-window-ready'); + } + } + + // Hide the loading screen + hide() { + if (this.window && !this.window.isDestroyed()) { + this.window.webContents.send('hide-loading'); + } + } + + // Set always on top property + setAlwaysOnTop(alwaysOnTop) { + if (this.window && !this.window.isDestroyed()) { + this.window.setAlwaysOnTop(alwaysOnTop); + } + } + + // Close the loading screen + close() { + if (this.window) { + this.window.close(); + this.window = null; + } + } + + // Check if window exists and is not destroyed + isValid() { + return this.window && !this.window.isDestroyed(); + } +} + +module.exports = LoadingScreen; \ No newline at end of file diff --git a/electron/main.cjs b/electron/main.cjs new file mode 100644 index 00000000..5e10c4f8 --- /dev/null +++ b/electron/main.cjs @@ -0,0 +1,7843 @@ +const { app, BrowserWindow, ipcMain, dialog, systemPreferences, Menu, shell, protocol, globalShortcut, Tray, nativeImage, desktopCapturer } = require('electron'); +const path = require('path'); +const fs = require('fs'); +const fsSync = require('fs'); +const log = require('electron-log'); +const https = require('https'); +const http = require('http'); +const { URL } = require('url'); +// const { pipeline } = require('stream/promises'); +// const crypto = require('crypto'); +// const { spawn } = require('child_process'); +const DockerSetup = require('./dockerSetup.cjs'); +const { setupAutoUpdater, checkForUpdates, getUpdateInfo } = require('./updateService.cjs'); +// const SplashScreen = require('./splash.cjs'); +// const LoadingScreen = require('./loadingScreen.cjs'); +const FeatureSelectionScreen = require('./featureSelection.cjs'); +const { createAppMenu } = require('./menu.cjs'); + +const MCPService = require('./mcpService.cjs'); +const WatchdogService = require('./watchdogService.cjs'); +const ComfyUIModelService = require('./comfyUIModelService.cjs'); +const PlatformManager = require('./platformManager.cjs'); +const { platformUpdateService } = require('./updateService.cjs'); +const { debugPaths, logDebugInfo } = require('./debug-paths.cjs'); +const IPCLogger = require('./ipcLogger.cjs'); +const WidgetService = require('./widgetService.cjs'); +const { SchedulerElectronService } = require('./schedulerElectronService.cjs'); +const { setupRemoteServerIPC } = require('./remoteServerIPC.cjs'); + +// NEW: Enhanced service management system (Backward compatible) +const CentralServiceManager = require('./centralServiceManager.cjs'); +const ServiceConfigurationManager = require('./serviceConfiguration.cjs'); +const { getPlatformCompatibility, getCompatibleServices } = require('./serviceDefinitions.cjs'); +const ClaraCoreRemoteService = require('./claraCoreRemoteService.cjs'); + +// Network Service Manager to prevent UI refreshes during crashes +const NetworkServiceManager = require('./networkServiceManager.cjs'); + +// Netlify OAuth Handler +const NetlifyOAuthHandler = require('./netlifyOAuthHandler.cjs'); + +// Immutable Startup Settings Manager + +// Global helper functions for container configuration +// These functions can be called from anywhere and will create container configs if needed + +// Helper function to get or create ComfyUI configuration +const getComfyUIConfig = () => { + if (!dockerSetup) { + throw new Error('Docker setup not initialized'); + } + + // Check if ComfyUI is supported on this platform + if (process.platform !== 'win32') { + throw new Error(`ComfyUI is not supported on ${process.platform}. It requires Windows with NVIDIA GPU support.`); + } + + // Get ComfyUI configuration + let comfyuiConfig = dockerSetup.containers.comfyui; + + // If ComfyUI config is not available (was filtered out during setup), create it + if (!comfyuiConfig) { + log.info('ComfyUI configuration not found in enabled containers, creating configuration...'); + comfyuiConfig = { + name: 'clara_comfyui', + image: dockerSetup.getArchSpecificImage('clara17verse/clara-comfyui', 'with-custom-nodes'), + port: 8188, + internalPort: 8188, + healthCheck: dockerSetup.isComfyUIRunning.bind(dockerSetup), + volumes: dockerSetup.getComfyUIVolumes(), + environment: [ + 'NVIDIA_VISIBLE_DEVICES=all', + 'CUDA_VISIBLE_DEVICES=0', + 'PYTORCH_CUDA_ALLOC_CONF=max_split_size_mb:2048,expandable_segments:True', + 'CUDA_LAUNCH_BLOCKING=0', + 'TORCH_CUDNN_V8_API_ENABLED=1', + 'CUDA_MODULE_LOADING=LAZY', + 'XFORMERS_MORE_DETAILS=0', + 'COMFYUI_FORCE_FP16=1', + 'COMFYUI_DISABLE_XFORMERS_WARNING=1', + 'COMFYUI_HIGHVRAM=1', + 'COMFYUI_DISABLE_MODEL_OFFLOAD=1', + 'COMFYUI_VRAM_USAGE=gpu-only' + ], + runtime: 'nvidia', + restartPolicy: 'unless-stopped' + }; + + // Add the ComfyUI config back to the containers object + dockerSetup.containers.comfyui = comfyuiConfig; + } + + return comfyuiConfig; +}; + +// Helper function to get or create N8N configuration +const getN8NConfig = () => { + if (!dockerSetup) { + throw new Error('Docker setup not initialized'); + } + + // Get N8N configuration + let n8nConfig = dockerSetup.containers.n8n; + + // If N8N config is not available (was filtered out during setup), create it + if (!n8nConfig) { + log.info('N8N configuration not found in enabled containers, creating configuration...'); + n8nConfig = { + name: 'clara_n8n', + image: dockerSetup.getArchSpecificImage('n8nio/n8n', 'latest'), + port: 5678, + internalPort: 5678, + healthCheck: dockerSetup.checkN8NHealth.bind(dockerSetup), + volumes: [ + `${require('path').join(require('os').homedir(), '.clara', 'n8n')}:/home/node/.n8n` + ] + }; + + // Add the N8N config back to the containers object + dockerSetup.containers.n8n = n8nConfig; + } + + return n8nConfig; +}; + +// Helper function to get or create Python backend configuration +const getPythonConfig = () => { + if (!dockerSetup) { + throw new Error('Docker setup not initialized'); + } + + // Get Python configuration + let pythonConfig = dockerSetup.containers.python; + + // If Python config is not available (was filtered out during setup), create it + if (!pythonConfig) { + log.info('Python backend configuration not found in enabled containers, creating configuration...'); + pythonConfig = { + name: 'clara_python', + image: dockerSetup.getArchSpecificImage('clara17verse/clara-backend', 'latest'), + // On Linux (host network mode), use port 5000. On Windows/Mac (bridge mode), use port 5001 + port: process.platform === 'linux' ? 5000 : 5001, + internalPort: 5000, + healthCheck: dockerSetup.isPythonRunning.bind(dockerSetup), + volumes: [ + // Mount the python_backend_data folder as the clara user's home directory + `${dockerSetup.pythonBackendDataPath}:/home/clara`, + // Keep backward compatibility for existing data paths + 'clara_python_models:/app/models' + ], + volumeNames: ['clara_python_models'], + // Request GPU runtime for AI acceleration (e.g., Whisper, image generation) + runtime: 'nvidia', + // GPU-specific environment variables for Python AI libraries + gpuEnvironment: [ + 'CUDA_VISIBLE_DEVICES=0', + 'TF_CPP_MIN_LOG_LEVEL=2', + 'PYTORCH_CUDA_ALLOC_CONF=max_split_size_mb:512' + ] + }; + + // Add the Python config back to the containers object + dockerSetup.containers.python = pythonConfig; + } + + return pythonConfig; +}; + +/** + * Helper function to show dialogs properly during startup when loading screen is active + * Temporarily disables alwaysOnTop to allow dialogs to appear above loading screen + */ +async function showStartupDialog(loadingScreen, dialogType, title, message, buttons = ['OK']) { + // Temporarily disable alwaysOnTop for loading screen + if (loadingScreen) { + loadingScreen.setAlwaysOnTop(false); + } + + try { + // Show dialog with proper window options + const result = await dialog.showMessageBox(loadingScreen ? loadingScreen.window : null, { + type: dialogType, + title: title, + message: message, + buttons: buttons, + alwaysOnTop: true, + modal: true + }); + return result; + } finally { + // Re-enable alwaysOnTop for loading screen + if (loadingScreen) { + loadingScreen.setAlwaysOnTop(true); + } + } +} + +// Configure the main process logger +log.transports.file.level = 'info'; +log.info('Application starting...'); + +// Single instance lock - prevent multiple instances of the app +const gotTheLock = app.requestSingleInstanceLock(); + +if (!gotTheLock) { + log.info('Another instance of ClaraVerse is already running. Exiting this instance.'); + app.quit(); +} else { + // Handle second instance attempts + app.on('second-instance', (event, commandLine, workingDirectory) => { + log.info('Second instance attempted to start. Focusing main window.'); + + // Someone tried to run a second instance, focus the existing window instead + if (mainWindow) { + if (mainWindow.isMinimized()) { + mainWindow.restore(); + } + mainWindow.focus(); + mainWindow.show(); + + // CRITICAL FIX: Force webContents focus when second instance launches + if (mainWindow.webContents && !mainWindow.webContents.isDestroyed()) { + setTimeout(() => { + if (mainWindow && !mainWindow.isDestroyed() && mainWindow.webContents) { + mainWindow.webContents.focus(); + } + }, 100); + } + } else { + // If no window exists, create one + createMainWindow().catch(error => { + log.error('Error creating main window from second instance:', error); + }); + } + }); +} + +// Initialize IPC Logger +let ipcLogger; + +// Wrap ipcMain.handle to log all IPC calls +const originalHandle = ipcMain.handle; +ipcMain.handle = function(channel, handler) { + return originalHandle.call(this, channel, async (event, ...args) => { + if (ipcLogger) { + ipcLogger.logIPC(channel, args, 'incoming'); + } + + try { + const result = await handler(event, ...args); + if (ipcLogger) { + ipcLogger.logIPC(channel, result, 'outgoing'); + } + return result; + } catch (error) { + if (ipcLogger) { + ipcLogger.logError(channel, error, 'outgoing'); + } + throw error; + } + }); +}; + +// Wrap ipcMain.on to log all IPC calls +const originalOn = ipcMain.on; +ipcMain.on = function(channel, handler) { + return originalOn.call(this, channel, (event, ...args) => { + if (ipcLogger) { + ipcLogger.logIPC(channel, args, 'incoming'); + } + + try { + const result = handler(event, ...args); + return result; + } catch (error) { + if (ipcLogger) { + ipcLogger.logError(channel, error, 'outgoing'); + } + throw error; + } + }); +}; + +// Initialize IPC Logger after app is ready +app.whenReady().then(() => { + ipcLogger = new IPCLogger(); + ipcLogger.logSystem('Application started'); + + // Initialize Network Service Manager to prevent UI refreshes during crashes + networkServiceManager = new NetworkServiceManager(); + log.info('🛡️ Network Service Manager initialized'); + + // Note: Startup Settings Manager is initialized later in the main initialization flow (line 4530) + // This duplicate initialization has been removed to prevent errors +}); + +// macOS Security Configuration - Prevent unnecessary firewall prompts +if (process.platform === 'darwin') { + // Request network permissions early if needed + try { + const hasNetworkAccess = systemPreferences.getMediaAccessStatus('microphone') === 'granted'; + if (!hasNetworkAccess) { + log.info('Preparing network access permissions for local AI services...'); + } + } catch (error) { + log.warn('Could not check network permissions:', error); + } +} + +// Global variables +let mainWindow; +let splash; +let loadingScreen; +let dockerSetup; +let mcpService; +let watchdogService; +let updateService; +let comfyUIModelService; +let widgetService; +let schedulerService; +let initializationInProgress = false; +let initializationComplete = false; + +// NEW: Enhanced service management (Coexists with existing services) +let serviceConfigManager; +let centralServiceManager; + +// Network Service Manager to prevent UI refreshes +let networkServiceManager; + +// Track active downloads for stop functionality +const activeDownloads = new Map(); + +// Add tray-related variables at the top level +let tray = null; +let isQuitting = false; + +// Local static server for packaged builds +let staticServer = null; +let staticServerPort = null; +const STATIC_SERVER_HOST = '127.0.0.1'; +const DEFAULT_STATIC_SERVER_PORT = 37117; + +// Helper function to format bytes +function formatBytes(bytes, decimals = 2) { + if (bytes === 0) return '0 Bytes'; + + const k = 1024; + const dm = decimals < 0 ? 0 : decimals; + const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB']; + + const i = Math.floor(Math.log(bytes) / Math.log(k)); + + return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + ' ' + sizes[i]; +} + +/** + * Stop all local services before switching to remote + * This ensures no conflicts and clean handoff to remote deployment + */ +async function stopAllLocalServices(serviceName = null) { + const stoppedServices = []; + const errors = []; + + try { + // If serviceName specified, stop only that service. Otherwise stop all. + const servicesToStop = serviceName ? [serviceName] : ['claracore', 'comfyui', 'llamacpp', 'searxng', 'python_backend']; + + for (const service of servicesToStop) { + try { + // Stop native service via centralServiceManager + if (centralServiceManager && centralServiceManager.services.has(service)) { + log.info(`Stopping local ${service} native service...`); + await centralServiceManager.stopService(service); + stoppedServices.push(`${service} (native)`); + } + + // Stop Docker services (claraCoreDocker may not be defined in this scope) + // Skip Docker stop as it's handled by centralServiceManager + // if (service === 'claracore' && claraCoreDockerService) { + // log.info('Stopping local ClaraCore Docker service...'); + // await claraCoreDockerService.stopService(); + // stoppedServices.push('claracore (docker)'); + // } + + if (service === 'comfyui' && comfyUIService) { + log.info('Stopping local ComfyUI Docker service...'); + await comfyUIService.stopService(); + stoppedServices.push('comfyui (docker)'); + } + + if (service === 'llamacpp' && llamaCppService) { + log.info('Stopping local LlamaCpp Docker service...'); + await llamaCppService.stopService(); + stoppedServices.push('llamacpp (docker)'); + } + + } catch (serviceError) { + log.warn(`Error stopping ${service} (continuing):`, serviceError.message); + errors.push({ service, error: serviceError.message }); + } + } + + if (stoppedServices.length > 0) { + log.info(`✅ Stopped local services: ${stoppedServices.join(', ')}`); + } + + } catch (error) { + log.error('Error in stopAllLocalServices:', error); + errors.push({ service: 'general', error: error.message }); + } + + return { + stopped: stoppedServices, + errors: errors + }; +} + +/** + * Ensure a loopback static server is running when the packaged app loads renderer assets. + * Using a fixed loopback port keeps the renderer origin stable so persisted storage survives restarts. + */ +async function ensureStaticServer() { + if (staticServer && staticServer.listening && staticServerPort) { + return staticServerPort; + } + + const distPath = path.join(__dirname, '../dist'); + if (!fs.existsSync(distPath)) { + throw new Error(`Renderer dist directory not found at ${distPath}`); + } + + const getMimeType = (filePath) => { + const ext = path.extname(filePath).toLowerCase(); + switch (ext) { + case '.html': + return 'text/html; charset=utf-8'; + case '.js': + return 'application/javascript; charset=utf-8'; + case '.css': + return 'text/css; charset=utf-8'; + case '.json': + return 'application/json; charset=utf-8'; + case '.png': + return 'image/png'; + case '.jpg': + case '.jpeg': + return 'image/jpeg'; + case '.svg': + return 'image/svg+xml'; + case '.webp': + return 'image/webp'; + case '.woff': + return 'font/woff'; + case '.woff2': + return 'font/woff2'; + case '.ttf': + return 'font/ttf'; + case '.ico': + return 'image/x-icon'; + default: + return 'application/octet-stream'; + } + }; + + const sendFile = (res, filePath, statusCode = 200) => { + res.statusCode = statusCode; + res.setHeader('Cache-Control', 'no-store'); + res.setHeader('Cross-Origin-Embedder-Policy', 'credentialless'); + res.setHeader('Cross-Origin-Opener-Policy', 'same-origin'); + res.setHeader('Cross-Origin-Resource-Policy', 'cross-origin'); + res.setHeader('Content-Type', getMimeType(filePath)); + + const stream = fs.createReadStream(filePath); + stream.on('error', (error) => { + log.error('Static server stream error:', error); + if (!res.headersSent) { + res.writeHead(500, { 'Content-Type': 'text/plain; charset=utf-8' }); + } + res.end('Internal Server Error'); + }); + stream.pipe(res); + }; + + const server = http.createServer((req, res) => { + try { + const requestUrl = new URL(req.url, `http://${STATIC_SERVER_HOST}`); + let pathname = decodeURIComponent(requestUrl.pathname); + + if (pathname.endsWith('/')) { + pathname += 'index.html'; + } + + const resolvedPath = path.normalize(path.join(distPath, pathname)); + + if (!resolvedPath.startsWith(distPath)) { + res.writeHead(403, { 'Content-Type': 'text/plain; charset=utf-8' }); + res.end('Forbidden'); + return; + } + + if (fs.existsSync(resolvedPath) && fs.statSync(resolvedPath).isFile()) { + sendFile(res, resolvedPath); + } else { + const fallback = path.join(distPath, 'index.html'); + if (fs.existsSync(fallback)) { + sendFile(res, fallback); + } else { + res.writeHead(404, { 'Content-Type': 'text/plain; charset=utf-8' }); + res.end('Not Found'); + } + } + } catch (error) { + log.error('Static server request error:', error); + res.writeHead(500, { 'Content-Type': 'text/plain; charset=utf-8' }); + res.end('Internal Server Error'); + } + }); + + const listenOnPort = (port) => new Promise((resolve, reject) => { + const handleError = (error) => { + server.off('listening', handleListening); + reject(error); + }; + + const handleListening = () => { + server.off('error', handleError); + resolve(server.address().port); + }; + + server.once('error', handleError); + server.once('listening', handleListening); + server.listen(port, STATIC_SERVER_HOST); + }); + + const desiredPort = Number(process.env.CLARA_STATIC_PORT) || DEFAULT_STATIC_SERVER_PORT; + let boundPort = desiredPort; + + try { + boundPort = await listenOnPort(desiredPort); + } catch (error) { + if (error && error.code === 'EADDRINUSE') { + // CRITICAL: Using a different port will break data persistence (localStorage/IndexedDB) + // because the origin (http://127.0.0.1:) will be different + log.error(`Static server port ${desiredPort} already in use. This will cause data loss!`); + log.error(`Another instance of the app may be running, or another service is using port ${desiredPort}.`); + log.error(`All user data (settings, chats, etc.) is tied to the origin http://127.0.0.1:${desiredPort}`); + + // Show error dialog to user + const { dialog } = require('electron'); + const choice = await dialog.showMessageBox({ + type: 'error', + title: 'Port Conflict - Data Loss Risk', + message: `Cannot start on port ${desiredPort}`, + detail: `Port ${desiredPort} is already in use. This could mean:\n\n` + + `1. Another instance of ClaraVerse is running\n` + + `2. Another application is using this port\n\n` + + `Using a different port will cause you to lose access to your saved data ` + + `(settings, chat history, etc.) because browser storage is tied to the port number.\n\n` + + `What would you like to do?`, + buttons: ['Quit and Fix Port Conflict', 'Continue Anyway (Data Will Be Lost)'], + defaultId: 0, + cancelId: 0 + }); + + if (choice.response === 0) { + // User chose to quit + app.quit(); + throw new Error('Port conflict - user chose to quit'); + } else { + // User chose to continue with random port (data loss) + log.warn(`User chose to continue despite port conflict. Using random port - DATA WILL BE LOST!`); + boundPort = await listenOnPort(0); + } + } else { + throw error; + } + } + + staticServer = server; + staticServerPort = boundPort; + log.info(`Loopback static server ready on http://${STATIC_SERVER_HOST}:${staticServerPort}`); + + return staticServerPort; +} + +// Register Docker container management IPC handlers +function registerDockerContainerHandlers() { + // Get all containers + ipcMain.handle('get-containers', async () => { + try { + if (!dockerSetup || !dockerSetup.docker) { + log.error('Docker setup not initialized'); + return []; + } + + const docker = dockerSetup.docker; + const containers = await docker.listContainers({ all: true }); + + return containers.map((container) => { + const ports = container.Ports.map((p) => + p.PublicPort ? `${p.PublicPort}:${p.PrivatePort}` : `${p.PrivatePort}` + ); + + return { + id: container.Id, + name: container.Names[0].replace(/^\//, ''), + image: container.Image, + status: container.Status, + state: container.State === 'running' ? 'running' : + container.State === 'exited' ? 'stopped' : container.State, + ports: ports, + created: new Date(container.Created * 1000).toLocaleString() + }; + }); + } catch (error) { + log.error('Error listing containers:', error); + return []; + } + }); + + // Container actions (start, stop, restart, remove) + ipcMain.handle('container-action', async (_event, { containerId, action }) => { + try { + if (!dockerSetup || !dockerSetup.docker) { + log.error('Docker setup not initialized'); + throw new Error('Docker setup not initialized'); + } + + const docker = dockerSetup.docker; + const container = docker.getContainer(containerId); + + switch (action) { + case 'start': + await container.start(); + break; + case 'stop': + await container.stop(); + break; + case 'restart': + await container.restart(); + break; + case 'remove': + await container.remove({ force: true }); + break; + default: + throw new Error(`Unknown action: ${action}`); + } + + return { success: true }; + } catch (error) { + log.error(`Error performing action ${action} on container:`, error); + return { success: false, error: error.message }; + } + }); + + // Create new container + ipcMain.handle('create-container', async (_event, containerConfig) => { + try { + if (!dockerSetup || !dockerSetup.docker) { + log.error('Docker setup not initialized'); + throw new Error('Docker setup not initialized'); + } + + const docker = dockerSetup.docker; + + // Format ports for Docker API + const portBindings = {}; + const exposedPorts = {}; + + containerConfig.ports.forEach((port) => { + const containerPort = `${port.container}/tcp`; + exposedPorts[containerPort] = {}; + portBindings[containerPort] = [{ HostPort: port.host.toString() }]; + }); + + // Format volumes for Docker API + const binds = containerConfig.volumes.map((volume) => + `${volume.host}:${volume.container}` + ); + + // Format environment variables + const env = Object.entries(containerConfig.env || {}).map(([key, value]) => `${key}=${value}`); + + // Create container + const container = await docker.createContainer({ + Image: containerConfig.image, + name: containerConfig.name, + ExposedPorts: exposedPorts, + Env: env, + HostConfig: { + PortBindings: portBindings, + Binds: binds, + NetworkMode: 'clara_network' + } + }); + + // Start the container + await container.start(); + + return { success: true, id: container.id }; + } catch (error) { + log.error('Error creating container:', error); + return { success: false, error: error.message }; + } + }); + + // Get container stats + ipcMain.handle('get-container-stats', async (_event, containerId) => { + try { + if (!dockerSetup || !dockerSetup.docker) { + log.error('Docker setup not initialized'); + throw new Error('Docker setup not initialized'); + } + + const docker = dockerSetup.docker; + const container = docker.getContainer(containerId); + + const stats = await container.stats({ stream: false }); + + // Calculate CPU usage percentage + const cpuDelta = stats.cpu_stats.cpu_usage.total_usage - stats.precpu_stats.cpu_usage.total_usage; + const systemCpuDelta = stats.cpu_stats.system_cpu_usage - stats.precpu_stats.system_cpu_usage; + const cpuCount = stats.cpu_stats.online_cpus || 1; + const cpuPercent = (cpuDelta / systemCpuDelta) * cpuCount * 100; + + // Calculate memory usage + const memoryUsage = stats.memory_stats.usage || 0; + const memoryLimit = stats.memory_stats.limit || 1; + const memoryPercent = (memoryUsage / memoryLimit) * 100; + + // Format network I/O + let networkRx = 0; + let networkTx = 0; + + if (stats.networks) { + Object.keys(stats.networks).forEach(iface => { + networkRx += stats.networks[iface].rx_bytes || 0; + networkTx += stats.networks[iface].tx_bytes || 0; + }); + } + + return { + cpu: `${cpuPercent.toFixed(2)}%`, + memory: `${formatBytes(memoryUsage)} / ${formatBytes(memoryLimit)} (${memoryPercent.toFixed(2)}%)`, + network: `↓ ${formatBytes(networkRx)} / ↑ ${formatBytes(networkTx)}` + }; + } catch (error) { + log.error('Error getting container stats:', error); + return { cpu: 'N/A', memory: 'N/A', network: 'N/A' }; + } + }); + + // Get container logs + ipcMain.handle('get-container-logs', async (_event, containerId) => { + try { + if (!dockerSetup || !dockerSetup.docker) { + log.error('Docker setup not initialized'); + throw new Error('Docker setup not initialized'); + } + + const docker = dockerSetup.docker; + const container = docker.getContainer(containerId); + + const logs = await container.logs({ + stdout: true, + stderr: true, + tail: 100, + follow: false + }); + + return logs.toString(); + } catch (error) { + log.error('Error getting container logs:', error); + return ''; + } + }); +} + +// Register MCP service IPC handlers +function registerMCPHandlers() { + // Helper function to ensure MCP service is initialized + function ensureMCPService() { + if (!mcpService) { + log.info('MCP service not initialized, creating new instance...'); + mcpService = new MCPService(); + } + return mcpService; + } + + // Helper function to ensure Service Config Manager is initialized + function ensureServiceConfigManager() { + if (!serviceConfigManager) { + log.info('Service config manager not initialized, creating new instance...'); + try { + serviceConfigManager = new ServiceConfigurationManager(); + if (!centralServiceManager) { + centralServiceManager = new CentralServiceManager(serviceConfigManager); + + const { SERVICE_DEFINITIONS } = require('./serviceDefinitions.cjs'); + Object.keys(SERVICE_DEFINITIONS).forEach(serviceName => { + const serviceDefinition = SERVICE_DEFINITIONS[serviceName]; + centralServiceManager.registerService(serviceName, serviceDefinition); + }); + } + } catch (error) { + log.warn('Failed to initialize service config manager:', error); + return null; + } + } + return serviceConfigManager; + } + + // Get all MCP servers + ipcMain.handle('mcp-get-servers', async () => { + try { + const service = ensureMCPService(); + return service.getAllServers(); + } catch (error) { + log.error('Error getting MCP servers:', error); + return []; + } + }); + + // Add MCP server + ipcMain.handle('mcp-add-server', async (event, serverConfig) => { + try { + const service = ensureMCPService(); + const result = await service.addServer(serverConfig); + + // Automatically start the newly added server + if (result === true && serverConfig.name) { + try { + log.info(`Auto-starting newly added MCP server: ${serverConfig.name}`); + await service.startServer(serverConfig.name); + log.info(`Successfully auto-started MCP server: ${serverConfig.name}`); + } catch (startError) { + log.warn(`Failed to auto-start newly added MCP server ${serverConfig.name}:`, startError); + // Don't throw here - server was added successfully, just auto-start failed + } + } + + return result; + } catch (error) { + log.error('Error adding MCP server:', error); + throw error; + } + }); + + // Remove MCP server + ipcMain.handle('mcp-remove-server', async (event, name) => { + try { + const service = ensureMCPService(); + return await service.removeServer(name); + } catch (error) { + log.error('Error removing MCP server:', error); + throw error; + } + }); + + // Update MCP server + ipcMain.handle('mcp-update-server', async (event, name, updates) => { + try { + const service = ensureMCPService(); + return await service.updateServer(name, updates); + } catch (error) { + log.error('Error updating MCP server:', error); + throw error; + } + }); + + // Start MCP server + ipcMain.handle('mcp-start-server', async (event, name) => { + try { + const service = ensureMCPService(); + const serverInfo = await service.startServer(name); + + // Return only serializable data, excluding the process object + return { + name: serverInfo.name, + config: serverInfo.config, + startedAt: serverInfo.startedAt, + status: serverInfo.status, + pid: serverInfo.process?.pid + }; + } catch (error) { + log.error('Error starting MCP server:', error); + throw error; + } + }); + + // Stop MCP server + ipcMain.handle('mcp-stop-server', async (event, name) => { + try { + const service = ensureMCPService(); + return await service.stopServer(name); + } catch (error) { + log.error('Error stopping MCP server:', error); + throw error; + } + }); + + // Restart MCP server + ipcMain.handle('mcp-restart-server', async (event, name) => { + try { + const service = ensureMCPService(); + const serverInfo = await service.restartServer(name); + + // Return only serializable data, excluding the process object + return { + name: serverInfo.name, + config: serverInfo.config, + startedAt: serverInfo.startedAt, + status: serverInfo.status, + pid: serverInfo.process?.pid + }; + } catch (error) { + log.error('Error restarting MCP server:', error); + throw error; + } + }); + + // Get MCP server status + ipcMain.handle('mcp-get-server-status', async (event, name) => { + try { + const service = ensureMCPService(); + return service.getServerStatus(name); + } catch (error) { + log.error('Error getting MCP server status:', error); + return null; + } + }); + + // Test MCP server + ipcMain.handle('mcp-test-server', async (event, name) => { + try { + const service = ensureMCPService(); + return await service.testServer(name); + } catch (error) { + log.error('Error testing MCP server:', error); + return { success: false, error: error.message }; + } + }); + + // Get MCP server templates + ipcMain.handle('mcp-get-templates', async () => { + try { + const service = ensureMCPService(); + return service.getServerTemplates(); + } catch (error) { + log.error('Error getting MCP templates:', error); + return []; + } + }); + + // Start all enabled MCP servers + ipcMain.handle('mcp-start-all-enabled', async () => { + try { + const service = ensureMCPService(); + return await service.startAllEnabledServers(); + } catch (error) { + log.error('Error starting all enabled MCP servers:', error); + throw error; + } + }); + + // Stop all MCP servers + ipcMain.handle('mcp-stop-all', async () => { + try { + const service = ensureMCPService(); + return await service.stopAllServers(); + } catch (error) { + log.error('Error stopping all MCP servers:', error); + throw error; + } + }); + + // Import from Claude Desktop config + ipcMain.handle('mcp-import-claude-config', async (event, configPath) => { + try { + const service = ensureMCPService(); + const result = await service.importFromClaudeConfig(configPath); + + // Automatically start all imported servers + if (result && result.imported > 0) { + log.info(`Auto-starting ${result.imported} imported MCP servers`); + + // Get the list of all servers to find the newly imported ones + const allServers = await service.getServers(); + const recentlyImported = Object.keys(allServers).filter(name => { + const server = allServers[name]; + return server.description && server.description.includes('Imported from Claude Desktop'); + }); + + for (const serverName of recentlyImported) { + try { + await service.startServer(serverName); + log.info(`Successfully auto-started imported MCP server: ${serverName}`); + } catch (startError) { + log.warn(`Failed to auto-start imported MCP server ${serverName}:`, startError); + // Continue with other servers even if one fails + } + } + } + + return result; + } catch (error) { + log.error('Error importing Claude config:', error); + throw error; + } + }); + + // Start previously running servers + ipcMain.handle('mcp-start-previously-running', async () => { + try { + const service = ensureMCPService(); + return await service.startPreviouslyRunningServers(); + } catch (error) { + log.error('Error starting previously running MCP servers:', error); + throw error; + } + }); + + // Save current running state + ipcMain.handle('mcp-save-running-state', async () => { + try { + const service = ensureMCPService(); + service.saveRunningState(); + return true; + } catch (error) { + log.error('Error saving MCP server running state:', error); + throw error; + } + }); + + // Execute MCP tool call + ipcMain.handle('mcp-execute-tool', async (event, toolCall) => { + try { + const service = ensureMCPService(); + return await service.executeToolCall(toolCall); + } catch (error) { + log.error('Error executing MCP tool call:', error); + throw error; + } + }); + + // Diagnose Node.js installation + ipcMain.handle('mcp-diagnose-node', async () => { + try { + const service = ensureMCPService(); + return await service.diagnoseNodeInstallation(); + } catch (error) { + log.error('Error diagnosing Node.js installation:', error); + return { + nodeAvailable: false, + npmAvailable: false, + npxAvailable: false, + suggestions: ['Error occurred while diagnosing Node.js installation: ' + error.message] + }; + } + }); +} + +// NEW: Register service configuration IPC handlers (Backward compatible) +function registerServiceConfigurationHandlers() { + console.log('[main] Registering service configuration IPC handlers...'); + + // Helper function to ensure Service Config Manager is initialized + function ensureServiceConfigManager() { + if (!serviceConfigManager) { + log.info('Service config manager not initialized, creating new instance...'); + try { + serviceConfigManager = new ServiceConfigurationManager(); + if (!centralServiceManager) { + centralServiceManager = new CentralServiceManager(serviceConfigManager); + + const { SERVICE_DEFINITIONS } = require('./serviceDefinitions.cjs'); + Object.keys(SERVICE_DEFINITIONS).forEach(serviceName => { + const serviceDefinition = SERVICE_DEFINITIONS[serviceName]; + centralServiceManager.registerService(serviceName, serviceDefinition); + }); + } + } catch (error) { + log.warn('Failed to initialize service config manager:', error); + return null; + } + } + return serviceConfigManager; + } + + // Get platform compatibility information + ipcMain.handle('service-config:get-platform-compatibility', async () => { + try { + return getPlatformCompatibility(); + } catch (error) { + log.error('Error getting platform compatibility:', error); + return {}; + } + }); + + // Get all service configurations + ipcMain.handle('service-config:get-all-configs', async () => { + try { + const configManager = ensureServiceConfigManager(); + if (!configManager || typeof configManager.getConfigSummary !== 'function') { + return {}; + } + return configManager.getConfigSummary(); + } catch (error) { + log.error('Error getting service configurations:', error); + return {}; + } + }); + + // Set service configuration (mode and URL) + ipcMain.handle('service-config:set-config', async (event, serviceName, mode, url = null) => { + try { + log.info(`📝 [Config] Received set-config request: serviceName=${serviceName}, mode=${mode}, url=${url}`); + + const configManager = ensureServiceConfigManager(); + if (!configManager || typeof configManager.setServiceConfig !== 'function') { + throw new Error('Service configuration manager not initialized or setServiceConfig method not available'); + } + + configManager.setServiceConfig(serviceName, mode, url); + log.info(`✅ [Config] Service ${serviceName} configured: mode=${mode}${url ? `, url=${url}` : ''}`); + + // Verify it was saved + const savedMode = configManager.getServiceMode(serviceName); + const savedUrl = configManager.getServiceUrl(serviceName); + log.info(`🔍 [Config] Verification - saved mode=${savedMode}, saved url=${savedUrl}`); + + // Update CentralServiceManager if the service exists + if (centralServiceManager) { + const service = centralServiceManager.services.get(serviceName); + if (service) { + service.deploymentMode = mode; + log.info(`✅ Updated CentralServiceManager: ${serviceName} deploymentMode=${mode}`); + + // If switching to remote/manual mode with a URL, check if it's running + if ((mode === 'remote' || mode === 'manual') && url) { + try { + // Test the health of the remote service + const healthCheck = await configManager.testManualService(serviceName, url, '/health'); + if (healthCheck.success) { + centralServiceManager.setState(serviceName, centralServiceManager.states.RUNNING); + centralServiceManager.setServiceUrl(serviceName, url); + service.serviceUrl = url; + log.info(`✅ Remote service ${serviceName} is running at ${url}`); + } else { + centralServiceManager.setState(serviceName, centralServiceManager.states.STOPPED); + log.warn(`⚠️ Remote service ${serviceName} at ${url} is not healthy`); + } + } catch (error) { + log.warn(`⚠️ Could not verify remote service ${serviceName}:`, error.message); + // Don't fail the config change, just warn + centralServiceManager.setState(serviceName, centralServiceManager.states.STOPPED); + } + } else if (mode === 'local' || mode === 'docker') { + // If switching back to local/docker, mark as stopped (user needs to start it) + centralServiceManager.setState(serviceName, centralServiceManager.states.STOPPED); + service.serviceUrl = null; + } + } + } + + return { success: true }; + } catch (error) { + log.error(`❌ [Config] Error setting service configuration for ${serviceName}:`, error); + return { success: false, error: error.message }; + } + }); + + // Alias for onboarding compatibility + ipcMain.handle('service-config:set-manual-url', async (event, serviceName, url) => { + try { + const configManager = ensureServiceConfigManager(); + if (!configManager || typeof configManager.setServiceConfig !== 'function') { + throw new Error('Service configuration manager not initialized or setServiceConfig method not available'); + } + + configManager.setServiceConfig(serviceName, 'manual', url); + log.info(`Service ${serviceName} configured with manual URL: ${url}`); + + return { success: true }; + } catch (error) { + log.error(`Error setting manual URL for ${serviceName}:`, error); + return { success: false, error: error.message }; + } + }); + + // Test manual service connectivity + ipcMain.handle('service-config:test-manual-service', async (event, serviceName, url, healthEndpoint = '/') => { + try { + const configManager = ensureServiceConfigManager(); + if (!configManager || typeof configManager.testManualService !== 'function') { + throw new Error('Service configuration manager not initialized or testManualService method not available'); + } + + const result = await configManager.testManualService(serviceName, url, healthEndpoint); + return result; + } catch (error) { + log.error(`Error testing manual service ${serviceName}:`, error); + return { + success: false, + error: error.message, + timestamp: Date.now() + }; + } + }); + + // Get supported deployment modes for a service + ipcMain.handle('service-config:get-supported-modes', async (event, serviceName) => { + try { + const configManager = ensureServiceConfigManager(); + if (!configManager || typeof configManager.getSupportedModes !== 'function') { + return ['docker']; // Default fallback + } + + return configManager.getSupportedModes(serviceName); + } catch (error) { + log.error(`Error getting supported modes for ${serviceName}:`, error); + return ['docker']; + } + }); + + // Reset service configuration to defaults + ipcMain.handle('service-config:reset-config', async (event, serviceName) => { + try { + const configManager = ensureServiceConfigManager(); + if (!configManager || typeof configManager.removeServiceConfig !== 'function') { + throw new Error('Service configuration manager not initialized or removeServiceConfig method not available'); + } + + configManager.removeServiceConfig(serviceName); + log.info(`Service ${serviceName} configuration reset to defaults`); + + return { success: true }; + } catch (error) { + log.error(`Error resetting service configuration for ${serviceName}:`, error); + return { success: false, error: error.message }; + } + }); + + // Get enhanced service status (includes deployment mode info) + let lastLoggedServiceStatus = ''; + ipcMain.handle('service-config:get-enhanced-status', async () => { + try { + if (!centralServiceManager) { + log.warn('⚠️ Central service manager not available, returning empty status'); + return {}; + } + + const status = centralServiceManager.getServicesStatus(); + + // Check health of remote/manual services and update their state + const http = require('http'); + const https = require('https'); + + // Define service-specific health endpoints + const healthEndpoints = { + 'comfyui': '/', // ComfyUI uses root endpoint + 'n8n': '/healthz', // N8N uses /healthz + 'python-backend': '/health', // Python backend uses /health + 'claracore': '/health', // ClaraCore uses /health + 'mcp': '/health' // MCP uses /health + }; + + for (const [serviceName, serviceStatus] of Object.entries(status)) { + if ((serviceStatus.deploymentMode === 'remote' || serviceStatus.deploymentMode === 'manual') && serviceStatus.serviceUrl) { + try { + // Get service-specific health endpoint + const healthPath = healthEndpoints[serviceName] || '/health'; + const url = new URL(serviceStatus.serviceUrl); + const protocol = url.protocol === 'https:' ? https : http; + + // Build health endpoint URL + const baseUrl = serviceStatus.serviceUrl.replace(/\/$/, ''); // Remove trailing slash + const healthEndpoint = `${baseUrl}${healthPath}`; + + const isHealthy = await new Promise((resolve) => { + const req = protocol.get(healthEndpoint, { timeout: 3000 }, (res) => { + // Accept 200-299 status codes as healthy + resolve(res.statusCode >= 200 && res.statusCode < 300); + }); + req.on('error', (err) => { + // Only log non-connection errors + if (err.code !== 'ECONNREFUSED' && err.code !== 'ETIMEDOUT') { + log.debug(`Health check error for ${serviceName}:`, err.message); + } + resolve(false); + }); + req.on('timeout', () => { + req.destroy(); + resolve(false); + }); + }); + + // Update state based on health check + if (isHealthy) { + centralServiceManager.setState(serviceName, centralServiceManager.states.RUNNING); + status[serviceName].state = 'running'; + status[serviceName].lastHealthCheck = Date.now(); + } else { + centralServiceManager.setState(serviceName, centralServiceManager.states.STOPPED); + status[serviceName].state = 'stopped'; + } + } catch (error) { + // If health check fails, mark as stopped + if (error.code !== 'ECONNREFUSED' && error.code !== 'ETIMEDOUT') { + log.debug(`Health check exception for ${serviceName}:`, error.message); + } + centralServiceManager.setState(serviceName, centralServiceManager.states.STOPPED); + status[serviceName].state = 'stopped'; + } + } + } + + // Only log if meaningful status has changed (exclude dynamic fields like uptime, lastHealthCheck) + const stableStatus = {}; + for (const [serviceName, serviceStatus] of Object.entries(status)) { + stableStatus[serviceName] = { + state: serviceStatus.state, + deploymentMode: serviceStatus.deploymentMode, + restartAttempts: serviceStatus.restartAttempts, + serviceUrl: serviceStatus.serviceUrl, + isManual: serviceStatus.isManual, + canRestart: serviceStatus.canRestart, + supportedModes: serviceStatus.supportedModes, + lastError: serviceStatus.lastError + }; + } + + const stableStatusString = JSON.stringify(stableStatus); + if (stableStatusString !== lastLoggedServiceStatus) { + log.info('📊 Enhanced service status changed:', stableStatus); + lastLoggedServiceStatus = stableStatusString; + } + + return status; + } catch (error) { + log.error('Error getting enhanced service status:', error); + return {}; + } + }); + + console.log('[main] Service configuration IPC handlers registered successfully'); +} + +// Register widget service IPC handlers +function registerWidgetServiceHandlers() { + // Initialize widget service + ipcMain.handle('widget-service:init', async () => { + try { + if (!widgetService) { + widgetService = new WidgetService(); + log.info('Widget service initialized'); + } + return { success: true }; + } catch (error) { + log.error('Error initializing widget service:', error); + return { success: false, error: error.message }; + } + }); + + // Register a widget as active + ipcMain.handle('widget-service:register-widget', async (event, widgetType) => { + try { + if (!widgetService) { + widgetService = new WidgetService(); + } + + widgetService.registerWidget(widgetType); + const status = await widgetService.getStatus(); + return { success: true, status }; + } catch (error) { + log.error('Error registering widget:', error); + return { success: false, error: error.message }; + } + }); + + // Unregister a widget + ipcMain.handle('widget-service:unregister-widget', async (event, widgetType) => { + try { + if (!widgetService) { + return { success: true, status: { running: false, activeWidgets: [] } }; + } + + widgetService.unregisterWidget(widgetType); + const status = await widgetService.getStatus(); + return { success: true, status }; + } catch (error) { + log.error('Error unregistering widget:', error); + return { success: false, error: error.message }; + } + }); + + // Get widget service status + ipcMain.handle('widget-service:get-status', async () => { + try { + if (!widgetService) { + return { + success: true, + status: { + running: false, + port: 8765, + activeWidgets: [], + shouldRun: false + } + }; + } + + const status = await widgetService.getStatus(); + return { success: true, status }; + } catch (error) { + log.error('Error getting widget service status:', error); + return { success: false, error: error.message }; + } + }); + + // Start widget service manually + ipcMain.handle('widget-service:start', async () => { + try { + if (!widgetService) { + widgetService = new WidgetService(); + } + + const result = await widgetService.startService(); + return result; + } catch (error) { + log.error('Error starting widget service:', error); + return { success: false, error: error.message }; + } + }); + + // Stop widget service manually + ipcMain.handle('widget-service:stop', async () => { + try { + if (!widgetService) { + return { success: true, message: 'Service not running' }; + } + + const result = await widgetService.stopService(); + return result; + } catch (error) { + log.error('Error stopping widget service:', error); + return { success: false, error: error.message }; + } + }); + + // Restart widget service + ipcMain.handle('widget-service:restart', async () => { + try { + if (!widgetService) { + widgetService = new WidgetService(); + } + + const result = await widgetService.restartService(); + return result; + } catch (error) { + log.error('Error restarting widget service:', error); + return { success: false, error: error.message }; + } + }); + + // Manage service based on active widgets + ipcMain.handle('widget-service:manage', async () => { + try { + if (!widgetService) { + widgetService = new WidgetService(); + } + + const result = await widgetService.manageService(); + return { success: true, status: result }; + } catch (error) { + log.error('Error managing widget service:', error); + return { success: false, error: error.message }; + } + }); + + // Get service health status + ipcMain.handle('widget-service:health', async () => { + try { + if (!widgetService) { + return { success: true, healthy: false }; + } + + const healthy = await widgetService.isServiceRunning(); + return { success: true, healthy }; + } catch (error) { + log.error('Error checking widget service health:', error); + return { success: false, error: error.message, healthy: false }; + } + }); + + // Enable auto-start for current platform + ipcMain.handle('widget-service:enable-autostart', async () => { + try { + if (!widgetService) { + widgetService = new WidgetService(); + } + + widgetService.enableAutoStart(); + const status = await widgetService.getStatus(); + return { success: true, status }; + } catch (error) { + log.error('Error enabling widget service auto-start:', error); + return { success: false, error: error.message }; + } + }); + + // Disable auto-start for current platform + ipcMain.handle('widget-service:disable-autostart', async () => { + try { + if (!widgetService) { + widgetService = new WidgetService(); + } + + widgetService.disableAutoStart(); + const status = await widgetService.getStatus(); + return { success: true, status }; + } catch (error) { + log.error('Error disabling widget service auto-start:', error); + return { success: false, error: error.message }; + } + }); + + log.info('Widget service IPC handlers registered'); +} + +// Register N8N specific IPC handlers +function registerN8NHandlers() { + // Check Docker status + ipcMain.handle('n8n:check-docker-status', async () => { + try { + if (!dockerSetup) { + return { dockerRunning: false, error: 'Docker setup not initialized' }; + } + + const dockerRunning = await dockerSetup.isDockerRunning(); + return { dockerRunning }; + } catch (error) { + log.error('Error checking Docker status:', error); + return { dockerRunning: false, error: error.message }; + } + }); + + // Check N8N service status + ipcMain.handle('n8n:check-service-status', async () => { + try { + if (!dockerSetup) { + return { running: false, error: 'Docker setup not initialized' }; + } + + // Check service configuration mode + let n8nRunning = false; + let serviceUrl = 'http://localhost:5678'; + + if (serviceConfigManager && typeof serviceConfigManager.getServiceMode === 'function') { + try { + const n8nMode = serviceConfigManager.getServiceMode('n8n'); + if ((n8nMode === 'manual' || n8nMode === 'remote') && typeof serviceConfigManager.getServiceUrl === 'function') { + const n8nUrl = serviceConfigManager.getServiceUrl('n8n'); + if (n8nUrl) { + serviceUrl = n8nUrl; + try { + const { createManualHealthCheck } = require('./serviceDefinitions.cjs'); + const healthCheck = createManualHealthCheck(n8nUrl, '/healthz'); + n8nRunning = await healthCheck(); + } catch (error) { + log.debug(`N8N ${n8nMode} health check failed: ${error.message}`); + n8nRunning = false; + } + } + } else { + const healthResult = await dockerSetup.checkN8NHealth(); + n8nRunning = healthResult.success; + if (dockerSetup.ports && dockerSetup.ports.n8n) { + serviceUrl = `http://localhost:${dockerSetup.ports.n8n}`; + } + } + } catch (configError) { + log.warn('Error getting N8N service config, using default mode:', configError.message); + const healthResult = await dockerSetup.checkN8NHealth(); + n8nRunning = healthResult.success; + if (dockerSetup.ports && dockerSetup.ports.n8n) { + serviceUrl = `http://localhost:${dockerSetup.ports.n8n}`; + } + } + } else { + const healthResult = await dockerSetup.checkN8NHealth(); + n8nRunning = healthResult.success; + if (dockerSetup.ports && dockerSetup.ports.n8n) { + serviceUrl = `http://localhost:${dockerSetup.ports.n8n}`; + } + } + + return { running: n8nRunning, serviceUrl }; + } catch (error) { + log.error('Error checking N8N service status:', error); + return { running: false, error: error.message }; + } + }); + + // Start N8N container + ipcMain.handle('n8n:start-container', async () => { + try { + if (!dockerSetup) { + return { success: false, error: 'Docker setup not initialized' }; + } + + const dockerRunning = await dockerSetup.isDockerRunning(); + if (!dockerRunning) { + return { success: false, error: 'Docker is not running' }; + } + + // Start the N8N container + log.info('Starting N8N container...'); + + // Get N8N configuration (creates it if needed) + const n8nConfig = getN8NConfig(); + + // Add progress callback for docker pull operations + const n8nConfigWithProgress = { + ...n8nConfig, + statusCallback: (message, type, details) => { + if (mainWindow && !mainWindow.isDestroyed()) { + // Send docker pull progress events + mainWindow.webContents.send('n8n:startup-progress', { + message: message, + progress: details?.percentage || 0, + type: type || 'info', + stage: 'pulling' + }); + } + } + }; + + await dockerSetup.startContainer(n8nConfigWithProgress); + + // Wait for the service to be healthy with timeout + const maxAttempts = 30; // 30 seconds timeout + let attempts = 0; + let healthResult = { success: false }; + + while (attempts < maxAttempts && !healthResult.success) { + await new Promise(resolve => setTimeout(resolve, 1000)); // Wait 1 second + healthResult = await dockerSetup.checkN8NHealth(); + attempts++; + + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('n8n:startup-progress', { + message: `Starting N8N... (${attempts}/${maxAttempts})`, + progress: Math.round((attempts / maxAttempts) * 100), + stage: 'starting' + }); + } + } + + if (healthResult.success) { + const serviceUrl = dockerSetup.ports && dockerSetup.ports.n8n + ? `http://localhost:${dockerSetup.ports.n8n}` + : 'http://localhost:5678'; + + log.info('N8N container started successfully'); + return { success: true, serviceUrl }; + } else { + log.warn('N8N container started but health check failed'); + return { success: false, error: 'N8N started but is not responding to health checks' }; + } + } catch (error) { + log.error('Error starting N8N container:', error); + return { success: false, error: error.message }; + } + }); + + log.info('N8N IPC handlers registered'); +} + +// Register ComfyUI specific IPC handlers +function registerComfyUIHandlers() { + // Check Docker status + ipcMain.handle('comfyui:check-docker-status', async () => { + try { + if (!dockerSetup) { + return { dockerRunning: false, error: 'Docker setup not initialized' }; + } + + const dockerRunning = await dockerSetup.isDockerRunning(); + return { dockerRunning }; + } catch (error) { + log.error('Error checking Docker status:', error); + return { dockerRunning: false, error: error.message }; + } + }); + + // Check ComfyUI service status + ipcMain.handle('comfyui:check-service-status', async () => { + try { + if (!dockerSetup) { + return { running: false, error: 'Docker setup not initialized' }; + } + + // Check service configuration mode + let comfyuiRunning = false; + let serviceUrl = 'http://localhost:8188'; + + if (serviceConfigManager && typeof serviceConfigManager.getServiceMode === 'function') { + try { + const comfyuiMode = serviceConfigManager.getServiceMode('comfyui'); + if ((comfyuiMode === 'manual' || comfyuiMode === 'remote') && typeof serviceConfigManager.getServiceUrl === 'function') { + const comfyuiUrl = serviceConfigManager.getServiceUrl('comfyui'); + if (comfyuiUrl) { + serviceUrl = comfyuiUrl; + try { + const { createManualHealthCheck } = require('./serviceDefinitions.cjs'); + const healthCheck = createManualHealthCheck(comfyuiUrl, '/'); + comfyuiRunning = await healthCheck(); + } catch (error) { + log.debug(`ComfyUI ${comfyuiMode} health check failed: ${error.message}`); + comfyuiRunning = false; + } + } + } else { + const healthResult = await dockerSetup.isComfyUIRunning(); + comfyuiRunning = healthResult; + if (dockerSetup.ports && dockerSetup.ports.comfyui) { + serviceUrl = `http://localhost:${dockerSetup.ports.comfyui}`; + } + } + } catch (configError) { + log.warn('Error getting ComfyUI service config, using default mode:', configError.message); + const healthResult = await dockerSetup.isComfyUIRunning(); + comfyuiRunning = healthResult; + if (dockerSetup.ports && dockerSetup.ports.comfyui) { + serviceUrl = `http://localhost:${dockerSetup.ports.comfyui}`; + } + } + } else { + const healthResult = await dockerSetup.isComfyUIRunning(); + comfyuiRunning = healthResult; + if (dockerSetup.ports && dockerSetup.ports.comfyui) { + serviceUrl = `http://localhost:${dockerSetup.ports.comfyui}`; + } + } + + return { running: comfyuiRunning, serviceUrl }; + } catch (error) { + log.error('Error checking ComfyUI service status:', error); + return { running: false, error: error.message }; + } + }); + + // Start ComfyUI container + ipcMain.handle('comfyui:start-container', async () => { + try { + if (!dockerSetup) { + return { success: false, error: 'Docker setup not initialized' }; + } + + const dockerRunning = await dockerSetup.isDockerRunning(); + if (!dockerRunning) { + return { success: false, error: 'Docker is not running' }; + } + + // Get ComfyUI configuration (creates it if needed) + const comfyuiConfig = getComfyUIConfig(); + + // Add progress callback for docker pull operations + const originalHealthCheck = comfyuiConfig.healthCheck; + const comfyuiConfigWithProgress = { + ...comfyuiConfig, + statusCallback: (message, type, details) => { + if (mainWindow && !mainWindow.isDestroyed()) { + // Send docker pull progress events + mainWindow.webContents.send('comfyui:startup-progress', { + message: message, + progress: details?.percentage || 0, + type: type || 'info', + stage: 'pulling' + }); + } + } + }; + + // Start the ComfyUI container + log.info('Starting ComfyUI container...'); + await dockerSetup.startContainer(comfyuiConfigWithProgress); + + // Wait for the service to be healthy with timeout (ComfyUI takes longer) + const maxAttempts = 60; // 60 seconds timeout + let attempts = 0; + let healthResult = { success: false }; + + while (attempts < maxAttempts && !healthResult.success) { + await new Promise(resolve => setTimeout(resolve, 1000)); // Wait 1 second + healthResult = { success: await dockerSetup.isComfyUIRunning() }; + attempts++; + + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('comfyui:startup-progress', { + message: `Starting ComfyUI... (${attempts}/${maxAttempts})`, + progress: Math.round((attempts / maxAttempts) * 100), + stage: 'starting' + }); + } + } + + if (healthResult.success) { + const serviceUrl = dockerSetup.ports && dockerSetup.ports.comfyui + ? `http://localhost:${dockerSetup.ports.comfyui}` + : 'http://localhost:8188'; + + log.info('ComfyUI container started successfully'); + return { success: true, serviceUrl }; + } else { + log.warn('ComfyUI container started but health check failed'); + return { success: false, error: 'ComfyUI started but is not responding to health checks' }; + } + } catch (error) { + log.error('Error starting ComfyUI container:', error); + return { success: false, error: error.message }; + } + }); + + log.info('ComfyUI IPC handlers registered'); +} + +// Register Python Backend specific IPC handlers +function registerPythonBackendHandlers() { + // Helper function to ensure Service Config Manager is initialized + function ensureServiceConfigManager() { + if (!serviceConfigManager) { + log.info('Service config manager not initialized, creating new instance...'); + try { + serviceConfigManager = new ServiceConfigurationManager(); + if (!centralServiceManager) { + centralServiceManager = new CentralServiceManager(serviceConfigManager); + + const { SERVICE_DEFINITIONS } = require('./serviceDefinitions.cjs'); + Object.keys(SERVICE_DEFINITIONS).forEach(serviceName => { + const serviceDefinition = SERVICE_DEFINITIONS[serviceName]; + centralServiceManager.registerService(serviceName, serviceDefinition); + }); + } + } catch (error) { + log.warn('Failed to initialize service config manager:', error); + return null; + } + } + return serviceConfigManager; + } + + // Check Docker status + ipcMain.handle('check-docker-status', async () => { + try { + if (!dockerSetup) { + return { isRunning: false, error: 'Docker setup not initialized' }; + } + + const isRunning = await dockerSetup.isDockerRunning(); + return { isRunning }; + } catch (error) { + log.error('Error checking Docker status:', error); + return { isRunning: false, error: error.message }; + } + }); + + // Check Python backend service status (using unified service manager) + ipcMain.handle('check-python-status', async () => { + try { + // First check CentralServiceManager for current status + if (centralServiceManager) { + const serviceStatus = centralServiceManager.getServicesStatus()['python-backend']; + if (serviceStatus) { + const mode = serviceStatus.deploymentMode || 'docker'; + const serviceUrl = serviceStatus.serviceUrl; + const stateIsRunning = serviceStatus.state === 'running'; + + log.info(`📊 Python Backend Status from CentralServiceManager: mode=${mode}, state=${serviceStatus.state}, url=${serviceUrl}`); + + // CRITICAL FIX: Don't trust the state alone - actually verify health! + // The state might say 'running' but the container could be stopped/crashed + let actualHealthy = false; + + if (stateIsRunning) { + // Verify actual health based on mode + if (mode === 'docker') { + // Check Docker container health + actualHealthy = dockerSetup ? await dockerSetup.isPythonRunning() : false; + log.info(`🔍 Docker health check result: ${actualHealthy}`); + } else if (mode === 'remote' || mode === 'manual') { + // Check remote/manual endpoint health + try { + const { createManualHealthCheck } = require('./serviceDefinitions.cjs'); + const healthCheck = createManualHealthCheck(serviceUrl, '/health'); + actualHealthy = await healthCheck(); + log.info(`🔍 ${mode} health check result: ${actualHealthy}`); + } catch (err) { + log.warn(`Health check failed for ${mode} mode:`, err); + actualHealthy = false; + } + } + } + + // Return different messages based on deployment mode and ACTUAL health + if (mode === 'docker') { + if (!actualHealthy) { + // Check if Docker is available to provide better error message + const dockerRunning = dockerSetup ? await dockerSetup.isDockerRunning() : false; + if (!dockerRunning) { + return { + isHealthy: false, + serviceUrl: null, + mode, + error: 'Docker is not running. Please start Docker Desktop.' + }; + } + return { + isHealthy: false, + serviceUrl: null, + mode, + error: 'Python Backend Docker container is not running' + }; + } + return { isHealthy: true, serviceUrl: serviceUrl || 'http://localhost:5001', mode }; + } else if (mode === 'remote') { + if (!actualHealthy) { + return { + isHealthy: false, + serviceUrl, + mode, + error: `Remote server at ${serviceUrl || 'unknown'} is unreachable. Would you like to start the Docker container instead?` + }; + } + return { isHealthy: true, serviceUrl, mode }; + } else if (mode === 'manual') { + if (!actualHealthy) { + return { + isHealthy: false, + serviceUrl, + mode, + error: `Manual server at ${serviceUrl || 'unknown'} is unreachable. Would you like to start the Docker container instead?` + }; + } + return { isHealthy: true, serviceUrl, mode }; + } + } + } + + // Fallback to legacy check if CentralServiceManager is not available + if (!dockerSetup) { + return { isHealthy: false, serviceUrl: null, mode: 'docker', error: 'Service manager not initialized' }; + } + + // Legacy Docker check + const dockerRunning = await dockerSetup.isDockerRunning(); + if (dockerRunning) { + const healthResult = await dockerSetup.isPythonRunning(); + const serviceUrl = dockerSetup.ports && dockerSetup.ports.python + ? `http://localhost:${dockerSetup.ports.python}` + : 'http://localhost:5001'; + return { isHealthy: healthResult, serviceUrl, mode: 'docker' }; + } else { + return { isHealthy: false, serviceUrl: null, mode: 'docker', error: 'Docker is not running' }; + } + } catch (error) { + log.error('Error checking Python backend status:', error); + return { isHealthy: false, serviceUrl: null, mode: 'docker', error: error.message }; + } + }); + + // Check Python Backend service status (unified pattern like N8N/ComfyUI) + ipcMain.handle('python-backend:check-service-status', async () => { + try { + if (!dockerSetup) { + return { running: false, error: 'Docker setup not initialized' }; + } + + // Check service configuration mode + let pythonRunning = false; + let serviceUrl = 'http://localhost:5001'; + + if (serviceConfigManager && typeof serviceConfigManager.getServiceMode === 'function') { + try { + const pythonMode = serviceConfigManager.getServiceMode('python-backend'); + if (pythonMode === 'manual' && typeof serviceConfigManager.getServiceUrl === 'function') { + const pythonUrl = serviceConfigManager.getServiceUrl('python-backend'); + if (pythonUrl) { + serviceUrl = pythonUrl; + try { + const { createManualHealthCheck } = require('./serviceDefinitions.cjs'); + const healthCheck = createManualHealthCheck(pythonUrl, '/health'); + pythonRunning = await healthCheck(); + } catch (error) { + log.error('Error checking manual Python Backend health:', error); + pythonRunning = false; + } + } + } else if (pythonMode === 'remote' && typeof serviceConfigManager.getServiceUrl === 'function') { + const pythonUrl = serviceConfigManager.getServiceUrl('python-backend'); + if (pythonUrl) { + serviceUrl = pythonUrl; + try { + const { createManualHealthCheck } = require('./serviceDefinitions.cjs'); + const healthCheck = createManualHealthCheck(pythonUrl, '/health'); + pythonRunning = await healthCheck(); + } catch (error) { + log.error('Error checking remote Python Backend health:', error); + pythonRunning = false; + } + } + } else { + // Docker mode + pythonRunning = await dockerSetup.isPythonRunning(); + if (dockerSetup.ports && dockerSetup.ports.python) { + serviceUrl = `http://localhost:${dockerSetup.ports.python}`; + } + } + } catch (error) { + log.warn('Error checking service config, falling back to Docker check:', error); + pythonRunning = await dockerSetup.isPythonRunning(); + } + } else { + // Fallback to Docker check if service config manager not available + pythonRunning = await dockerSetup.isPythonRunning(); + } + + return { running: pythonRunning, serviceUrl }; + } catch (error) { + log.error('Error checking Python Backend status:', error); + return { running: false, serviceUrl: 'http://localhost:5001', error: error.message }; + } + }); + + // Get Python Backend URL (for frontend services to use) + ipcMain.handle('python-backend:get-url', async () => { + try { + // Default URL + let serviceUrl = 'http://localhost:5001'; + + if (serviceConfigManager && typeof serviceConfigManager.getServiceMode === 'function') { + try { + const pythonMode = serviceConfigManager.getServiceMode('python-backend'); + if ((pythonMode === 'manual' || pythonMode === 'remote') && typeof serviceConfigManager.getServiceUrl === 'function') { + const pythonUrl = serviceConfigManager.getServiceUrl('python-backend'); + if (pythonUrl) { + serviceUrl = pythonUrl; + } + } else if (pythonMode === 'docker') { + // Docker mode - check if container is running and get actual port + if (dockerSetup && dockerSetup.ports && dockerSetup.ports.python) { + serviceUrl = `http://localhost:${dockerSetup.ports.python}`; + } + } + } catch (error) { + log.warn('Error getting Python Backend URL from config, using default:', error); + } + } + + return { success: true, url: serviceUrl }; + } catch (error) { + log.error('Error getting Python Backend URL:', error); + return { success: false, url: 'http://localhost:5001', error: error.message }; + } + }); + + // Start Python backend container + // Returns: { success: boolean, status?: ServiceStatus, error?: string } + // Where ServiceStatus = { isHealthy: boolean, serviceUrl: string, mode: 'docker' | 'remote' | 'manual' } + // The status object is returned directly to avoid race conditions with check-python-status + ipcMain.handle('start-python-container', async () => { + try { + if (!dockerSetup) { + return { success: false, error: 'Docker setup not initialized' }; + } + + const dockerRunning = await dockerSetup.isDockerRunning(); + if (!dockerRunning) { + return { success: false, error: 'Docker is not running. Please start Docker first.' }; + } + + // Get Python container configuration (creates it if needed) + const pythonConfig = getPythonConfig(); + + // Add progress callback for docker pull operations + const pythonConfigWithProgress = { + ...pythonConfig, + statusCallback: (message, type, details) => { + if (mainWindow && !mainWindow.isDestroyed()) { + // Send docker pull progress events + mainWindow.webContents.send('python:startup-progress', { + message: message, + progress: details?.percentage || 0, + type: type || 'info', + stage: 'pulling' + }); + } + } + }; + + log.info('Starting Python backend container...'); + await dockerSetup.startContainer(pythonConfigWithProgress); + + // Wait for the container to be healthy with timeout + const maxAttempts = 30; // 60 seconds max + let attempts = 0; + + while (attempts < maxAttempts) { + await new Promise(resolve => setTimeout(resolve, 2000)); + + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('python:startup-progress', { + message: `Health check ${attempts + 1}/${maxAttempts} for Python backend...`, + progress: Math.round(((attempts + 1) / maxAttempts) * 100), + stage: 'starting' + }); + } + + const isHealthy = await dockerSetup.isPythonRunning(); + + if (isHealthy) { + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('python:startup-progress', { + message: 'Python backend is healthy and ready!', + progress: 100, + stage: 'ready' + }); + } + log.info('Python backend container started and is healthy'); + + // Update CentralServiceManager state + const serviceUrl = dockerSetup.ports && dockerSetup.ports.python + ? `http://localhost:${dockerSetup.ports.python}` + : 'http://localhost:5001'; + + if (centralServiceManager) { + centralServiceManager.setServiceUrl('python-backend', serviceUrl); + centralServiceManager.setState('python-backend', centralServiceManager.states.RUNNING); + log.info('✅ Updated CentralServiceManager: python-backend is running'); + } + + // Return the status directly to avoid race condition with check-python-status + return { + success: true, + status: { + isHealthy: true, + serviceUrl: serviceUrl, + mode: 'docker' + } + }; + } + + attempts++; + } + + return { success: false, error: 'Python backend started but is not responding to health checks' }; + } catch (error) { + log.error('Error starting Python backend container:', error); + return { success: false, error: error.message }; + } + }); + + log.info('Python Backend IPC handlers registered'); +} + +// Register isolated startup settings IPC handlers +function registerStartupSettingsHandlers() { + // Initialize startup settings manager + const manager = ensureStartupSettingsManager(); + log.info('🔒 Isolated startup settings manager initialized'); + + // Get startup settings (read-only for other services) + ipcMain.handle('startup-settings:get', async () => { + try { + const settings = manager.getSettingsWithSync(); + log.info('📄 Startup settings requested:', Object.keys(settings)); + return { success: true, settings }; + } catch (error) { + log.error('Error getting startup settings:', error); + return { success: false, error: error.message, settings: startupSettingsManager.defaultSettings }; + } + }); + + // Update startup settings (requires explicit consent through this handler only) + ipcMain.handle('startup-settings:update', async (event, newSettings, userConsent = false) => { + try { + if (!userConsent) { + log.warn('⚠️ Startup settings update attempted without user consent'); + return { success: false, error: 'User consent required for startup settings changes' }; + } + + log.info('🔒 Startup settings update with user consent:', Object.keys(newSettings)); + + const updatedSettings = manager.updateSettings(newSettings); + + // Apply auto-start behaviour when relevant settings change + const shouldUpdateLoginItem = + newSettings.autoStart !== undefined || newSettings.startMinimized !== undefined; + + if (shouldUpdateLoginItem && (process.platform === 'darwin' || process.platform === 'win32')) { + const openAtLogin = !updatedSettings.isDevelopment && !!updatedSettings.autoStart; + const openAsHidden = !!updatedSettings.startMinimized; + + if (!openAtLogin) { + log.info('Disabling login item for ClaraVerse'); + app.setLoginItemSettings({ + openAtLogin: false, + openAsHidden: false + }); + } else { + log.info('Updating login item preferences for ClaraVerse'); + app.setLoginItemSettings({ + openAtLogin, + openAsHidden, + path: process.execPath, + args: [] + }); + } + } + + return { success: true, settings: updatedSettings }; + } catch (error) { + log.error('Error updating startup settings:', error); + return { success: false, error: error.message }; + } + }); + + // Validate settings integrity + ipcMain.handle('startup-settings:validate', async (event, frontendChecksum) => { + try { + const settings = manager.getSettingsWithSync(); + const isValid = settings.checksum === frontendChecksum; + + if (!isValid) { + log.warn('⚠️ Startup settings checksum mismatch detected!'); + log.warn('Backend checksum:', settings.checksum); + log.warn('Frontend checksum:', frontendChecksum); + } + + return { + success: true, + isValid, + settings: isValid ? null : settings // Only send settings if mismatch + }; + } catch (error) { + log.error('Error validating startup settings:', error); + return { success: false, error: error.message, isValid: false }; + } + }); + + // Reset to defaults (requires explicit confirmation) + ipcMain.handle('startup-settings:reset', async (event, confirmed = false) => { + try { + if (!confirmed) { + return { success: false, error: 'Reset confirmation required' }; + } + + log.info('🔄 Resetting startup settings to defaults'); + const defaultSettings = { ...manager.defaultSettings }; + manager.writeSettings(defaultSettings); + + return { success: true, settings: defaultSettings }; + } catch (error) { + log.error('Error resetting startup settings:', error); + return { success: false, error: error.message }; + } + }); + + // Get file status (for debugging) + ipcMain.handle('startup-settings:get-file-status', async () => { + try { + const stats = { + mainFileExists: fs.existsSync(manager.settingsFile), + backupFileExists: fs.existsSync(manager.backupFile), + lockFileExists: fs.existsSync(manager.lockFile), + filePath: manager.settingsFile + }; + + if (stats.mainFileExists) { + const fileStats = fs.statSync(manager.settingsFile); + stats.lastModified = fileStats.mtime; + stats.fileSize = fileStats.size; + } + + return { success: true, stats }; + } catch (error) { + log.error('Error getting startup settings file status:', error); + return { success: false, error: error.message }; + } + }); + + log.info('🔒 Isolated startup settings IPC handlers registered'); +} + +// Dedicated startup settings manager - isolated from other services +class StartupSettingsManager { + constructor() { + this.settingsFile = path.join(app.getPath('userData'), 'clara-startup-settings.json'); + this.lockFile = this.settingsFile + '.lock'; + this.backupFile = this.settingsFile + '.backup'; + this.defaultSettings = { + startFullscreen: false, + startMinimized: false, + autoStart: false, + checkUpdates: true, + restoreLastSession: true, + autoStartMCP: true, + isDevelopment: process.env.NODE_ENV === 'development' || !app.isPackaged, + version: 1, + lastModified: Date.now() + }; + } + + // Check if file is locked by another operation + isLocked() { + return fs.existsSync(this.lockFile); + } + + // Create lock file to prevent concurrent access + createLock() { + fs.writeFileSync(this.lockFile, JSON.stringify({ pid: process.pid, timestamp: Date.now() })); + } + + // Remove lock file + removeLock() { + try { + if (fs.existsSync(this.lockFile)) { + fs.unlinkSync(this.lockFile); + } + } catch (error) { + log.warn('Failed to remove startup settings lock:', error); + } + } + + // Read startup settings with backup recovery + readSettings() { + try { + let settings = null; + + // Try main file first + if (fs.existsSync(this.settingsFile)) { + try { + const data = fs.readFileSync(this.settingsFile, 'utf8'); + settings = JSON.parse(data); + log.info('📄 Startup settings loaded from main file'); + } catch (parseError) { + log.warn('Main startup settings file corrupted, trying backup...', parseError); + + // Try backup file + if (fs.existsSync(this.backupFile)) { + try { + const backupData = fs.readFileSync(this.backupFile, 'utf8'); + settings = JSON.parse(backupData); + log.info('📄 Startup settings recovered from backup file'); + + // Restore main file from backup + fs.writeFileSync(this.settingsFile, backupData); + log.info('📄 Main startup settings file restored from backup'); + } catch (backupError) { + log.error('Backup startup settings file also corrupted:', backupError); + } + } + } + } + + // If no valid settings found, use defaults + if (!settings || typeof settings !== 'object') { + log.info('📄 Using default startup settings'); + settings = { ...this.defaultSettings }; + this.writeSettings(settings); // Create initial file + } + + // Merge with defaults for any missing properties + return { ...this.defaultSettings, ...settings }; + } catch (error) { + log.error('Error reading startup settings:', error); + return { ...this.defaultSettings }; + } + } + + // Write startup settings with atomic operation and backup + writeSettings(settings) { + if (this.isLocked()) { + throw new Error('Startup settings are locked by another operation'); + } + + try { + this.createLock(); + + // Create backup of current file + if (fs.existsSync(this.settingsFile)) { + fs.copyFileSync(this.settingsFile, this.backupFile); + } + + // Add metadata + const settingsWithMeta = { + ...settings, + version: this.defaultSettings.version, + lastModified: Date.now(), + isDevelopment: process.env.NODE_ENV === 'development' || !app.isPackaged + }; + + // Write to temporary file first (atomic operation) + const tempFile = this.settingsFile + '.tmp'; + fs.writeFileSync(tempFile, JSON.stringify(settingsWithMeta, null, 2)); + + // Verify write was successful + const verifyData = fs.readFileSync(tempFile, 'utf8'); + const verifySettings = JSON.parse(verifyData); + + if (!verifySettings || typeof verifySettings !== 'object') { + throw new Error('Settings verification failed - corrupted data'); + } + + // Atomic rename (replaces main file) + fs.renameSync(tempFile, this.settingsFile); + + log.info('📄 Startup settings saved successfully:', { + startFullscreen: settingsWithMeta.startFullscreen, + startMinimized: settingsWithMeta.startMinimized, + autoStart: settingsWithMeta.autoStart, + checkUpdates: settingsWithMeta.checkUpdates, + restoreLastSession: settingsWithMeta.restoreLastSession, + autoStartMCP: settingsWithMeta.autoStartMCP, + isDevelopment: settingsWithMeta.isDevelopment + }); + + return { success: true }; + } catch (error) { + log.error('Error writing startup settings:', error); + + // Clean up temp file if it exists + const tempFile = this.settingsFile + '.tmp'; + if (fs.existsSync(tempFile)) { + try { + fs.unlinkSync(tempFile); + } catch (cleanupError) { + log.warn('Failed to cleanup temp file:', cleanupError); + } + } + + throw error; + } finally { + this.removeLock(); + } + } + + // Update specific startup settings + updateSettings(newSettings) { + const currentSettings = this.readSettings(); + const updatedSettings = { ...currentSettings, ...newSettings }; + this.writeSettings(updatedSettings); + return updatedSettings; + } + + // Get settings with frontend sync validation + getSettingsWithSync() { + const settings = this.readSettings(); + + // Add checksum for frontend validation + const settingsString = JSON.stringify({ + startFullscreen: settings.startFullscreen, + startMinimized: settings.startMinimized, + autoStart: settings.autoStart, + checkUpdates: settings.checkUpdates, + restoreLastSession: settings.restoreLastSession, + autoStartMCP: settings.autoStartMCP + }); + + const crypto = require('crypto'); + const checksum = crypto.createHash('md5').update(settingsString).digest('hex'); + + return { + ...settings, + checksum + }; + } +} + +// Global startup settings manager instance (initialized in app ready event) +let startupSettingsManager; + +function ensureStartupSettingsManager() { + if (!startupSettingsManager) { + startupSettingsManager = new StartupSettingsManager(); + } + return startupSettingsManager; +} + +function loadLegacyStartupPreferences() { + try { + const legacyPath = path.join(app.getPath('userData'), 'clara-settings.json'); + if (!fs.existsSync(legacyPath)) { + return null; + } + + const legacyContent = JSON.parse(fs.readFileSync(legacyPath, 'utf8')); + const legacyStartup = legacyContent.startup || {}; + + return { + startFullscreen: + typeof legacyStartup.startFullscreen === 'boolean' + ? legacyStartup.startFullscreen + : typeof legacyContent.fullscreen_startup === 'boolean' + ? legacyContent.fullscreen_startup + : undefined, + startMinimized: + typeof legacyStartup.startMinimized === 'boolean' + ? legacyStartup.startMinimized + : undefined, + autoStartMCP: + typeof legacyStartup.autoStartMCP === 'boolean' + ? legacyStartup.autoStartMCP + : undefined + }; + } catch (error) { + log.warn('Failed to read legacy startup settings:', error); + return null; + } +} + +function getStartupPreferences() { + let settings; + try { + const manager = ensureStartupSettingsManager(); + settings = manager.readSettings(); + } catch (error) { + log.error('Error loading startup settings, using defaults:', error); + settings = { + startFullscreen: false, + startMinimized: false, + autoStart: false, + checkUpdates: true, + restoreLastSession: true, + autoStartMCP: true, + isDevelopment: process.env.NODE_ENV === 'development' || !app.isPackaged + }; + } + + const legacy = loadLegacyStartupPreferences(); + if (legacy) { + if (typeof settings.startFullscreen !== 'boolean' && typeof legacy.startFullscreen === 'boolean') { + settings.startFullscreen = legacy.startFullscreen; + } + if (typeof settings.startMinimized !== 'boolean' && typeof legacy.startMinimized === 'boolean') { + settings.startMinimized = legacy.startMinimized; + } + if (typeof settings.autoStartMCP !== 'boolean' && typeof legacy.autoStartMCP === 'boolean') { + settings.autoStartMCP = legacy.autoStartMCP; + } + } + + return { + settings, + startFullscreen: !!settings.startFullscreen, + startMinimized: !!settings.startMinimized, + autoStartMCP: settings.autoStartMCP !== false, + isDevelopment: !!settings.isDevelopment + }; +} + +// Register handlers for various app functions +function registerHandlers() { + console.log('[main] Registering IPC handlers...'); + + // Setup activity tracking for adaptive health checks (MUST BE FIRST) + const { setupActivityTracking } = require('./activityTracking.cjs'); + setupActivityTracking(); + + registerDockerContainerHandlers(); + registerMCPHandlers(); + registerServiceConfigurationHandlers(); // NEW: Add service configuration handlers + registerWidgetServiceHandlers(); // NEW: Add widget service handlers + registerN8NHandlers(); // NEW: Add N8N specific handlers + registerComfyUIHandlers(); // NEW: Add ComfyUI specific handlers + registerPythonBackendHandlers(); // NEW: Add Python Backend specific handlers + registerStartupSettingsHandlers(); // NEW: Add isolated startup settings handlers + + // Add new chat handler + ipcMain.handle('new-chat', async () => { + log.info('New chat requested via IPC'); + return { success: true }; + }); + + // Add dialog handler for folder picker + ipcMain.handle('show-open-dialog', async (_event, options) => { + console.log('[main] show-open-dialog handler called with options:', options); + try { + return await dialog.showOpenDialog(options); + } catch (error) { + log.error('Error showing open dialog:', error); + return { canceled: true, filePaths: [] }; + } + }); + console.log('[main] show-open-dialog handler registered successfully'); + + // App info handlers + ipcMain.handle('get-app-path', () => app.getPath('userData')); + ipcMain.handle('getWorkflowsPath', () => { + return path.join(app.getAppPath(), 'workflows', 'n8n_workflows_full.json'); + }); + + // Developer log handlers + ipcMain.handle('developer-logs:read', async (event, lines = 1000) => { + try { + if (!ipcLogger) { + return 'IPC Logger not initialized'; + } + return await ipcLogger.readLogs(lines); + } catch (error) { + log.error('Error reading developer logs:', error); + return `Error reading logs: ${error.message}`; + } + }); + + ipcMain.handle('developer-logs:get-files', async () => { + try { + if (!ipcLogger) { + return []; + } + return await ipcLogger.getLogFiles(); + } catch (error) { + log.error('Error getting log files:', error); + return []; + } + }); + + ipcMain.handle('developer-logs:clear', async () => { + try { + if (!ipcLogger) { + return { success: false, error: 'IPC Logger not initialized' }; + } + return await ipcLogger.clearLogs(); + } catch (error) { + log.error('Error clearing logs:', error); + return { success: false, error: error.message }; + } + }); + + // Fast startup handlers for dashboard + ipcMain.handle('get-initialization-state', async () => { + return { + needsFeatureSelection: global.needsFeatureSelection || false, + selectedFeatures: global.selectedFeatures || null, + systemConfig: global.systemConfig || null, + dockerAvailable: dockerSetup ? await dockerSetup.isDockerRunning() : false, + servicesStatus: { + mcp: mcpService ? true : false, + docker: dockerSetup ? true : false, + watchdog: watchdogService ? watchdogService.isRunning : false + } + }; + }); + + ipcMain.handle('save-feature-selection', async (event, features) => { + try { + const featureSelection = new FeatureSelectionScreen(); + featureSelection.saveConfig(features); + global.selectedFeatures = features; + global.needsFeatureSelection = false; + return { success: true }; + } catch (error) { + log.error('Error saving feature selection:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('initialize-service', async (event, serviceName) => { + try { + const sendUpdate = (status, message) => { + event.sender.send('service-init-progress', { service: serviceName, status, message }); + }; + + switch(serviceName) { + case 'docker': + if (!dockerSetup) dockerSetup = new DockerSetup(); + const dockerAvailable = await dockerSetup.isDockerRunning(); + if (!dockerAvailable) { + throw new Error('Docker is not running'); + } + await dockerSetup.setup(global.selectedFeatures, (status) => { + sendUpdate('progress', status); + }); + break; + + case 'mcp': + if (!mcpService) mcpService = new MCPService(); + await mcpService.startAllEnabledServers(); + break; + + case 'watchdog': + if (!watchdogService && dockerSetup?.docker) { + watchdogService = new WatchdogService(dockerSetup, mcpService); + watchdogService.start(); + } + break; + } + + return { success: true }; + } catch (error) { + log.error(`Error initializing service ${serviceName}:`, error); + return { success: false, error: error.message }; + } + }); + + // Update handlers + ipcMain.handle('check-for-updates', () => { + return checkForUpdates(); + }); + + ipcMain.handle('get-update-info', () => { + return getUpdateInfo(); + }); + + // Enhanced update handlers for in-app downloading + ipcMain.handle('start-in-app-download', async (event, updateInfo) => { + try { + // Import the enhanced update service + const { enhancedPlatformUpdateService } = require('./updateService.cjs'); + + if (!enhancedPlatformUpdateService) { + throw new Error('Enhanced update service not available'); + } + + const result = await enhancedPlatformUpdateService.startInAppDownload(updateInfo); + + // If download completed successfully, show completion dialog + if (result.success && result.filePath) { + // Send completion event to renderer + mainWindow.webContents.send('update-download-completed', { + filePath: result.filePath, + fileName: result.fileName + }); + + const response = await dialog.showMessageBox(mainWindow, { + type: 'info', + title: '✅ Download Complete!', + message: `Clara ${updateInfo.latestVersion} has been downloaded`, + detail: `The installer has been saved to:\n${result.filePath}\n\nWould you like to open it now?`, + buttons: ['Open Installer', 'Open Downloads Folder', 'Later'], + defaultId: 0 + }); + + if (response.response === 0) { + // Open the installer + shell.openPath(result.filePath); + } else if (response.response === 1) { + // Open downloads folder + shell.showItemInFolder(result.filePath); + } + } else if (!result.success) { + // Send error event to renderer + mainWindow.webContents.send('update-download-error', { + error: result.error + }); + } + + return result; + } catch (error) { + log.error('Error starting in-app download:', error); + + // Send error event to renderer + if (mainWindow && mainWindow.webContents) { + mainWindow.webContents.send('update-download-error', { + error: error.message + }); + } + + return { + success: false, + error: error.message + }; + } + }); + + // Permissions handler + ipcMain.handle('request-microphone-permission', async () => { + if (process.platform === 'darwin') { + const status = await systemPreferences.getMediaAccessStatus('microphone'); + if (status === 'not-determined') { + return await systemPreferences.askForMediaAccess('microphone'); + } + return status === 'granted'; + } + return true; + }); + + // Service info handlers + ipcMain.handle('get-service-ports', () => { + if (dockerSetup && dockerSetup.ports) { + return dockerSetup.ports; + } + return null; + }); + + ipcMain.handle('get-python-port', () => { + if (dockerSetup && dockerSetup.ports && dockerSetup.ports.python) { + return dockerSetup.ports.python; + } + return null; + }); + + ipcMain.handle('check-python-backend', async () => { + try { + if (!dockerSetup || !dockerSetup.ports || !dockerSetup.ports.python) { + return { status: 'error', message: 'Python backend not configured' }; + } + + const isRunning = await dockerSetup.isPythonRunning(); + if (!isRunning) { + return { status: 'error', message: 'Python backend container not running' }; + } + + return { + status: 'running', + port: dockerSetup.ports.python + }; + } catch (error) { + log.error('Error checking Python backend:', error); + return { status: 'error', message: error.message }; + } + }); + + ipcMain.handle('check-docker-services', async () => { + try { + if (!dockerSetup) { + return { + dockerAvailable: false, + n8nAvailable: false, + pythonAvailable: false, + comfyuiAvailable: false, + message: 'Docker setup not initialized' + }; + } + + const dockerRunning = await dockerSetup.isDockerRunning(); + if (!dockerRunning) { + return { + dockerAvailable: false, + n8nAvailable: false, + pythonAvailable: false, + comfyuiAvailable: false, + message: 'Docker is not running' + }; + } + + // Check service modes before testing Docker containers + let n8nRunning = false; + let comfyuiRunning = false; + + if (serviceConfigManager && typeof serviceConfigManager.getServiceMode === 'function') { + try { + // N8N health check + const n8nMode = serviceConfigManager.getServiceMode('n8n'); + if (n8nMode === 'manual' && typeof serviceConfigManager.getServiceUrl === 'function') { + const n8nUrl = serviceConfigManager.getServiceUrl('n8n'); + if (n8nUrl) { + try { + const { createManualHealthCheck } = require('./serviceDefinitions.cjs'); + const healthCheck = createManualHealthCheck(n8nUrl, '/'); + n8nRunning = await healthCheck(); + log.debug(`🔗 N8N manual service health: ${n8nRunning}`); + } catch (error) { + log.debug(`N8N manual health check failed: ${error.message}`); + n8nRunning = false; + } + } + } else { + n8nRunning = await dockerSetup.checkN8NHealth().then(result => result.success).catch(() => false); + } + + // ComfyUI health check + const comfyuiMode = serviceConfigManager.getServiceMode('comfyui'); + if (comfyuiMode === 'manual' && typeof serviceConfigManager.getServiceUrl === 'function') { + const comfyuiUrl = serviceConfigManager.getServiceUrl('comfyui'); + if (comfyuiUrl) { + try { + const { createManualHealthCheck } = require('./serviceDefinitions.cjs'); + const healthCheck = createManualHealthCheck(comfyuiUrl, '/'); + comfyuiRunning = await healthCheck(); + log.debug(`🔗 ComfyUI manual service health: ${comfyuiRunning}`); + } catch (error) { + log.debug(`ComfyUI manual health check failed: ${error.message}`); + comfyuiRunning = false; + } + } + } else { + comfyuiRunning = await dockerSetup.isComfyUIRunning().catch(() => false); + } + } catch (configError) { + log.warn('Error getting service configs, using Docker fallback:', configError.message); + // Fallback to Docker checks + n8nRunning = await dockerSetup.checkN8NHealth().then(result => result.success).catch(() => false); + comfyuiRunning = await dockerSetup.isComfyUIRunning().catch(() => false); + } + } else { + // Fallback to Docker checks + n8nRunning = await dockerSetup.checkN8NHealth().then(result => result.success).catch(() => false); + comfyuiRunning = await dockerSetup.isComfyUIRunning().catch(() => false); + } + + const pythonRunning = await dockerSetup.isPythonRunning().catch(() => false); + + return { + dockerAvailable: true, + n8nAvailable: n8nRunning, + pythonAvailable: pythonRunning, + comfyuiAvailable: comfyuiRunning, + ports: dockerSetup.ports + }; + } catch (error) { + log.error('Error checking Docker services:', error); + return { + dockerAvailable: false, + n8nAvailable: false, + pythonAvailable: false, + comfyuiAvailable: false, + message: error.message + }; + } + }); + + // Get Python backend information + ipcMain.handle('get-python-backend-info', async () => { + try { + if (!dockerSetup) { + throw new Error('Docker setup not initialized'); + } + + return dockerSetup.getPythonBackendInfo(); + } catch (error) { + log.error('Error getting Python backend info:', error); + return { error: error.message }; + } + }); + + // Check for container updates + ipcMain.handle('docker-check-updates', async () => { + try { + if (!dockerSetup) { + throw new Error('Docker not initialized'); + } + + return await dockerSetup.checkForUpdates((status) => { + log.info('Update check:', status); + }); + } catch (error) { + log.error('Error checking for updates:', error); + throw error; + } + }); + + // ComfyUI specific handlers + ipcMain.handle('comfyui-status', async () => { + try { + // NEW: Check if ComfyUI is configured for manual mode + if (serviceConfigManager && typeof serviceConfigManager.getServiceMode === 'function') { + const comfyuiMode = serviceConfigManager.getServiceMode('comfyui'); + if (comfyuiMode === 'manual' && typeof serviceConfigManager.getServiceUrl === 'function') { + const comfyuiUrl = serviceConfigManager.getServiceUrl('comfyui'); + log.info(`🔗 ComfyUI in manual mode, checking ${comfyuiUrl} instead of Docker`); + + if (!comfyuiUrl) { + return { running: false, error: 'Manual mode but no URL configured', mode: 'manual' }; + } + + // Test manual service connectivity + try { + const { createManualHealthCheck } = require('./serviceDefinitions.cjs'); + const healthCheck = createManualHealthCheck(comfyuiUrl, '/'); + const isHealthy = await healthCheck(); + + return { + running: isHealthy, + url: comfyuiUrl, + mode: 'manual', + containerName: 'manual-service' + }; + } catch (error) { + return { running: false, error: `Manual service health check failed: ${error.message}`, mode: 'manual' }; + } + } + } + + // Fallback to Docker mode + if (!dockerSetup) { + return { running: false, error: 'Docker not initialized' }; + } + + const isRunning = await dockerSetup.isComfyUIRunning(); + return { + running: isRunning, + port: dockerSetup.ports.comfyui || 8188, + containerName: 'clara_comfyui', + mode: 'docker' + }; + } catch (error) { + log.error('Error checking ComfyUI status:', error); + return { running: false, error: error.message }; + } + }); + + ipcMain.handle('comfyui-start', async () => { + try { + if (!dockerSetup) { + throw new Error('Docker not initialized'); + } + + // Get ComfyUI configuration (creates it if needed) + const comfyuiConfig = getComfyUIConfig(); + + await dockerSetup.startContainer(comfyuiConfig); + return { success: true }; + } catch (error) { + log.error('Error starting ComfyUI:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('comfyui-stop', async () => { + try { + if (!dockerSetup) { + throw new Error('Docker not initialized'); + } + + const container = await dockerSetup.docker.getContainer('clara_comfyui'); + await container.stop(); + return { success: true }; + } catch (error) { + log.error('Error stopping ComfyUI:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('comfyui-restart', async () => { + try { + if (!dockerSetup) { + throw new Error('Docker not initialized'); + } + + const container = await dockerSetup.docker.getContainer('clara_comfyui'); + await container.restart(); + return { success: true }; + } catch (error) { + log.error('Error restarting ComfyUI:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('comfyui-logs', async () => { + try { + if (!dockerSetup) { + throw new Error('Docker not initialized'); + } + + const container = await dockerSetup.docker.getContainer('clara_comfyui'); + const logs = await container.logs({ + stdout: true, + stderr: true, + tail: 100 + }); + return { success: true, logs: logs.toString() }; + } catch (error) { + log.error('Error getting ComfyUI logs:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('comfyui-optimize', async () => { + try { + if (!dockerSetup) { + throw new Error('Docker not initialized'); + } + + log.info('Manual ComfyUI optimization requested'); + await dockerSetup.optimizeComfyUIContainer(); + return { success: true, message: 'ComfyUI optimization completed' }; + } catch (error) { + log.error('Error optimizing ComfyUI:', error); + return { success: false, error: error.message }; + } + }); + + // ClaraCore Service IPC Handlers + ipcMain.handle('claracore-start', async () => { + try { + log.info('Starting ClaraCore service...'); + + // Check if service manager has claracore service + if (centralServiceManager && centralServiceManager.services.has('claracore')) { + await centralServiceManager.startService('claracore'); + return { success: true, message: 'ClaraCore service started successfully' }; + } + + return { success: false, error: 'ClaraCore service not registered' }; + } catch (error) { + log.error('Error starting ClaraCore:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('claracore-stop', async () => { + try { + log.info('Stopping ClaraCore service...'); + + if (centralServiceManager && centralServiceManager.services.has('claracore')) { + await centralServiceManager.stopService('claracore'); + return { success: true, message: 'ClaraCore service stopped successfully' }; + } + + return { success: false, error: 'ClaraCore service not registered' }; + } catch (error) { + log.error('Error stopping ClaraCore:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('claracore-restart', async () => { + try { + log.info('Restarting ClaraCore service...'); + + if (centralServiceManager && centralServiceManager.services.has('claracore')) { + await centralServiceManager.restartService('claracore'); + return { success: true, message: 'ClaraCore service restarted successfully' }; + } + + return { success: false, error: 'ClaraCore service not registered' }; + } catch (error) { + log.error('Error restarting ClaraCore:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('claracore-status', async () => { + try { + if (centralServiceManager && centralServiceManager.services.has('claracore')) { + const service = centralServiceManager.services.get('claracore'); + const status = { + isRunning: service.state === 'running', + state: service.state, + pid: service.instance?.process?.pid || null, + uptime: service.instance?.startTime ? Date.now() - service.instance.startTime : 0, + restartAttempts: service.restartAttempts || 0, + url: 'http://localhost:8091' + }; + return { success: true, status }; + } + + return { success: false, error: 'ClaraCore service not registered' }; + } catch (error) { + log.error('Error getting ClaraCore status:', error); + return { success: false, error: error.message }; + } + }); + + // ClaraCore Docker mode handlers + let claraCoreDockerService = null; + const getClaraCoreDockerService = () => { + if (!claraCoreDockerService) { + const ClaraCoreDockerService = require('./claraCoreDockerService.cjs'); + claraCoreDockerService = new ClaraCoreDockerService(); + } + return claraCoreDockerService; + }; + + ipcMain.handle('claracore-docker-start', async (event, options = {}) => { + try { + log.info('Starting ClaraCore in Docker mode...'); + const service = getClaraCoreDockerService(); + const result = await service.start(options); + + // Update central service manager state + if (centralServiceManager) { + const claraCoreService = centralServiceManager.services.get('claracore'); + if (claraCoreService) { + centralServiceManager.setState('claracore', centralServiceManager.states.RUNNING); + claraCoreService.deploymentMode = 'docker'; + claraCoreService.serviceUrl = 'http://localhost:8091'; + claraCoreService.instance = service; + log.info('✅ Updated CentralServiceManager: ClaraCore is now RUNNING in Docker mode'); + } + } + + // Save deployment mode preference to configuration + if (serviceConfigManager) { + try { + serviceConfigManager.setServiceConfig('claracore', 'docker', null); + log.info('✅ Saved ClaraCore deployment mode preference: docker'); + } catch (error) { + log.warn('Failed to save deployment mode preference:', error); + } + } + + return { success: true, ...result }; + } catch (error) { + log.error('Error starting ClaraCore Docker:', error); + + // Update central service manager state on error + if (centralServiceManager) { + const claraCoreService = centralServiceManager.services.get('claracore'); + if (claraCoreService) { + centralServiceManager.setState('claracore', centralServiceManager.states.ERROR); + claraCoreService.lastError = error; + } + } + + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('claracore-docker-stop', async () => { + try { + log.info('Stopping ClaraCore Docker container...'); + const service = getClaraCoreDockerService(); + const result = await service.stop(); + + // Update central service manager state + if (centralServiceManager) { + const claraCoreService = centralServiceManager.services.get('claracore'); + if (claraCoreService) { + centralServiceManager.setState('claracore', centralServiceManager.states.STOPPED); + claraCoreService.deploymentMode = null; + claraCoreService.serviceUrl = null; + claraCoreService.instance = null; + log.info('✅ Updated CentralServiceManager: ClaraCore is now STOPPED'); + } + } + + return { success: true, ...result }; + } catch (error) { + log.error('Error stopping ClaraCore Docker:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('claracore-docker-restart', async () => { + try { + log.info('Restarting ClaraCore Docker container...'); + const service = getClaraCoreDockerService(); + const result = await service.restart(); + + // Update central service manager state + if (centralServiceManager) { + const claraCoreService = centralServiceManager.services.get('claracore'); + if (claraCoreService) { + centralServiceManager.setState('claracore', centralServiceManager.states.RUNNING); + claraCoreService.deploymentMode = 'docker'; + claraCoreService.serviceUrl = 'http://localhost:8091'; + claraCoreService.instance = service; + log.info('✅ Updated CentralServiceManager: ClaraCore restarted in Docker mode'); + } + } + + return { success: true, ...result }; + } catch (error) { + log.error('Error restarting ClaraCore Docker:', error); + + // Update central service manager state on error + if (centralServiceManager) { + const claraCoreService = centralServiceManager.services.get('claracore'); + if (claraCoreService) { + centralServiceManager.setState('claracore', centralServiceManager.states.ERROR); + claraCoreService.lastError = error; + } + } + + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('claracore-docker-status', async () => { + try { + const service = getClaraCoreDockerService(); + const status = await service.getStatus(); + return { success: true, status }; + } catch (error) { + log.error('Error getting ClaraCore Docker status:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('claracore-docker-detect-gpu', async () => { + try { + log.info('Detecting GPU for ClaraCore Docker...'); + const service = getClaraCoreDockerService(); + const gpuInfo = await service.detectGPU(); + return { success: true, gpuInfo }; + } catch (error) { + log.error('Error detecting GPU:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('claracore-docker-remove', async () => { + try { + log.info('Removing ClaraCore Docker container...'); + const service = getClaraCoreDockerService(); + const result = await service.remove(); + return { success: true, ...result }; + } catch (error) { + log.error('Error removing ClaraCore Docker container:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('claracore-docker-logs', async (event, options = {}) => { + try { + const service = getClaraCoreDockerService(); + const logs = await service.getLogs(options); + return { success: true, logs }; + } catch (error) { + log.error('Error getting ClaraCore Docker logs:', error); + return { success: false, error: error.message }; + } + }); + + // System information handlers + ipcMain.handle('get-system-info', async () => { + try { + const os = require('os'); + return { + platform: os.platform(), + arch: os.arch(), + cpus: os.cpus().length, + totalMemory: os.totalmem(), + freeMemory: os.freemem(), + hostname: os.hostname(), + release: os.release(), + type: os.type() + }; + } catch (error) { + log.error('Error getting system info:', error); + return { error: error.message }; + } + }); + + ipcMain.handle('get-performance-mode', async () => { + try { + if (global.systemConfig) { + return { + performanceMode: global.systemConfig.performanceMode, + enabledFeatures: global.systemConfig.enabledFeatures, + resourceLimitations: global.systemConfig.resourceLimitations + }; + } + + return { performanceMode: 'unknown', enabledFeatures: {}, resourceLimitations: {} }; + } catch (error) { + log.error('❌ Error getting performance mode:', error); + return { error: error.message }; + } + }); + + // ComfyUI consent management + ipcMain.handle('save-comfyui-consent', async (event, hasConsented) => { + try { + const fs = require('fs'); + const path = require('path'); + + const userDataPath = app.getPath('userData'); + const consentFile = path.join(userDataPath, 'comfyui-consent.json'); + + const consentData = { + hasConsented, + timestamp: new Date().toISOString(), + version: '1.0' + }; + + fs.writeFileSync(consentFile, JSON.stringify(consentData, null, 2)); + log.info(`ComfyUI consent saved: ${hasConsented}`); + + // Update watchdog service if it's running + if (watchdogService) { + watchdogService.setComfyUIMonitoring(hasConsented); + } + + return { success: true }; + } catch (error) { + log.error('Error saving ComfyUI consent:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('get-comfyui-consent', async () => { + try { + const fs = require('fs'); + const path = require('path'); + + const userDataPath = app.getPath('userData'); + const consentFile = path.join(userDataPath, 'comfyui-consent.json'); + + if (fs.existsSync(consentFile)) { + const consentData = JSON.parse(fs.readFileSync(consentFile, 'utf8')); + return consentData; + } + + return null; + } catch (error) { + log.error('Error reading ComfyUI consent:', error); + return null; + } + }); + + // Direct GPU information handler + ipcMain.handle('get-gpu-info', async () => { + try { + const { spawn, spawnSync } = require('child_process'); + const os = require('os'); + + let hasNvidiaGPU = false; + let gpuName = ''; + let isAMD = false; + let gpuMemoryMB = 0; + + // Try nvidia-smi first (most reliable for NVIDIA GPUs) + try { + const nvidiaSmi = spawnSync('nvidia-smi', [ + '--query-gpu=name,memory.total', + '--format=csv,noheader,nounits' + ], { encoding: 'utf8', timeout: 5000 }); + + if (nvidiaSmi.status === 0 && nvidiaSmi.stdout) { + const lines = nvidiaSmi.stdout.trim().split('\n'); + if (lines.length > 0 && lines[0].trim()) { + const parts = lines[0].split(','); + if (parts.length >= 2) { + gpuName = parts[0].trim(); + gpuMemoryMB = parseInt(parts[1].trim()) || 0; + hasNvidiaGPU = true; + + log.info(`NVIDIA GPU detected via nvidia-smi: ${gpuName} (${gpuMemoryMB}MB)`); + } + } + } + } catch (error) { + log.debug('nvidia-smi not available or failed:', error.message); + } + + // If nvidia-smi failed, try WMIC on Windows + if (!hasNvidiaGPU && os.platform() === 'win32') { + try { + const wmic = spawnSync('wmic', [ + 'path', 'win32_VideoController', + 'get', 'name,AdapterRAM', + '/format:csv' + ], { encoding: 'utf8', timeout: 10000 }); + + if (wmic.status === 0 && wmic.stdout) { + const lines = wmic.stdout.split('\n').filter(line => line.trim() && !line.startsWith('Node')); + + for (const line of lines) { + const parts = line.split(','); + if (parts.length >= 3) { + const ramStr = parts[1]?.trim(); + const nameStr = parts[2]?.trim(); + + if (nameStr && ramStr && !isNaN(parseInt(ramStr))) { + const ramBytes = parseInt(ramStr); + const ramMB = Math.round(ramBytes / (1024 * 1024)); + + // Check if this is a better GPU than what we found + if (ramMB > gpuMemoryMB) { + gpuName = nameStr; + gpuMemoryMB = ramMB; + + const lowerName = nameStr.toLowerCase(); + hasNvidiaGPU = lowerName.includes('nvidia') || + lowerName.includes('geforce') || + lowerName.includes('rtx') || + lowerName.includes('gtx'); + isAMD = lowerName.includes('amd') || lowerName.includes('radeon'); + + log.info(`GPU detected via WMIC: ${gpuName} (${gpuMemoryMB}MB)`); + } + } + } + } + } + } catch (error) { + log.debug('WMIC GPU detection failed:', error.message); + } + } + + // Try PowerShell as another fallback on Windows + if (!hasNvidiaGPU && !gpuName && os.platform() === 'win32') { + try { + const powershell = spawnSync('powershell', [ + '-Command', + 'Get-WmiObject -Class Win32_VideoController | Select-Object Name, AdapterRAM | ConvertTo-Json' + ], { encoding: 'utf8', timeout: 10000 }); + + if (powershell.status === 0 && powershell.stdout) { + const gpuData = JSON.parse(powershell.stdout); + const gpus = Array.isArray(gpuData) ? gpuData : [gpuData]; + + for (const gpu of gpus) { + if (gpu.Name && gpu.AdapterRAM) { + const ramMB = Math.round(gpu.AdapterRAM / (1024 * 1024)); + + if (ramMB > gpuMemoryMB) { + gpuName = gpu.Name; + gpuMemoryMB = ramMB; + + const lowerName = gpu.Name.toLowerCase(); + hasNvidiaGPU = lowerName.includes('nvidia') || + lowerName.includes('geforce') || + lowerName.includes('rtx') || + lowerName.includes('gtx'); + isAMD = lowerName.includes('amd') || lowerName.includes('radeon'); + + log.info(`GPU detected via PowerShell: ${gpuName} (${gpuMemoryMB}MB)`); + } + } + } + } + } catch (error) { + log.debug('PowerShell GPU detection failed:', error.message); + } + } + + return { + success: true, + gpuInfo: { + hasNvidiaGPU, + gpuName, + isAMD, + gpuMemoryMB, + gpuMemoryGB: Math.round(gpuMemoryMB / 1024 * 10) / 10, + platform: os.platform() + } + }; + } catch (error) { + log.error('Error getting GPU info:', error); + return { + success: false, + error: error.message + }; + } + }); + + // Get watchdog service status including ComfyUI + ipcMain.handle('get-services-status', async () => { + try { + if (!watchdogService) { + return { error: 'Watchdog service not initialized' }; + } + + return { + services: watchdogService.getServicesStatus(), + overallHealth: watchdogService.getOverallHealth() + }; + } catch (error) { + log.error('Error getting services status:', error); + return { error: error.message }; + } + }); + + // Watchdog service handlers + ipcMain.handle('watchdog-get-services-status', async () => { + try { + if (!watchdogService) { + return { success: false, error: 'Watchdog service not initialized' }; + } + + return { + success: true, + services: watchdogService.getServicesStatus(), + overallHealth: watchdogService.getOverallHealth() + }; + } catch (error) { + log.error('Error getting watchdog services status:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('watchdog-get-overall-health', async () => { + try { + if (!watchdogService) { + return { success: false, error: 'Watchdog service not initialized' }; + } + + return { + success: true, + health: watchdogService.getOverallHealth() + }; + } catch (error) { + log.error('Error getting watchdog overall health:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('watchdog-perform-manual-health-check', async () => { + try { + if (!watchdogService) { + return { success: false, error: 'Watchdog service not initialized' }; + } + + const services = await watchdogService.performManualHealthCheck(); + return { + success: true, + services: services + }; + } catch (error) { + log.error('Error performing manual health check:', error); + return { success: false, error: error.message }; + } + }); + + // Update containers + ipcMain.handle('docker-update-containers', async (event, containerNames) => { + try { + if (!dockerSetup) { + throw new Error('Docker not initialized'); + } + + return await dockerSetup.updateContainers(containerNames, (status, type = 'info') => { + log.info('Container update:', status); + // Send progress updates to the renderer + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('docker-update-progress', { status, type }); + } + }); + } catch (error) { + log.error('Error updating containers:', error); + throw error; + } + }); + + // Get system architecture info + ipcMain.handle('docker-get-system-info', async () => { + try { + if (!dockerSetup) { + throw new Error('Docker not initialized'); + } + + return { + architecture: dockerSetup.systemArch, + platform: process.platform, + arch: process.arch + }; + } catch (error) { + log.error('Error getting system info:', error); + throw error; + } + }); + + // Enhanced Docker detection + ipcMain.handle('docker-detect-installations', async () => { + try { + if (!dockerSetup) { + throw new Error('Docker not initialized'); + } + + const installations = await dockerSetup.detectDockerInstallations(); + return installations.map(install => ({ + type: install.type, + method: install.method, + priority: install.priority, + path: install.path, + host: install.host, + port: install.port, + contextName: install.contextName, + machineName: install.machineName, + isPodman: install.isPodman || false, + isNamedPipe: install.isNamedPipe || false + })); + } catch (error) { + log.error('Error detecting Docker installations:', error); + throw error; + } + }); + + // Get Docker detection report + ipcMain.handle('docker-get-detection-report', async () => { + try { + if (!dockerSetup) { + throw new Error('Docker not initialized'); + } + + return await dockerSetup.getDockerDetectionReport(); + } catch (error) { + log.error('Error getting Docker detection report:', error); + throw error; + } + }); + + // Test all Docker installations + ipcMain.handle('docker-test-all-installations', async () => { + try { + if (!dockerSetup) { + throw new Error('Docker not initialized'); + } + + return await dockerSetup.testAllDockerInstallations(); + } catch (error) { + log.error('Error testing Docker installations:', error); + throw error; + } + }); + + // Event handlers + ipcMain.on('backend-status', (event, status) => { + if (mainWindow) { + mainWindow.webContents.send('backend-status', status); + } + }); + + ipcMain.on('python-status', (event, status) => { + if (mainWindow) { + mainWindow.webContents.send('python-status', status); + } + }); + + + // Handle loading screen completion + ipcMain.on('loading-complete', () => { + log.info('Loading screen fade-out complete'); + if (loadingScreen) { + loadingScreen.close(); + loadingScreen = null; + } + }); + + // Handle React app ready signal + ipcMain.on('react-app-ready', async () => { + log.info('React app fully initialized and ready'); + if (loadingScreen && loadingScreen.isValid()) { + loadingScreen.notifyMainWindowReady(); + } + + // Auto-restore MCP servers when React app is ready (if not already restored) + if (mcpService && !global.mcpServersRestored) { + try { + log.info('React app ready - checking MCP auto-start setting...'); + + // Check startup settings for MCP auto-start using isolated startup settings + let shouldAutoStartMCP = true; // Default to true for backward compatibility + + try { + const { autoStartMCP } = getStartupPreferences(); + shouldAutoStartMCP = autoStartMCP; + log.info('🔒 Using isolated startup settings for MCP auto-start:', shouldAutoStartMCP); + } catch (settingsError) { + log.warn('Error reading isolated startup settings for MCP auto-start:', settingsError); + // Default to true on error to maintain existing behavior + } + + if (shouldAutoStartMCP) { + log.info('React app ready - attempting to restore previously running MCP servers...'); + const restoreResults = await mcpService.startPreviouslyRunningServers(); + const successCount = restoreResults.filter(r => r.success).length; + const totalCount = restoreResults.length; + + if (totalCount > 0) { + log.info(`MCP restoration on app ready: ${successCount}/${totalCount} servers restored`); + } else { + log.info('MCP restoration on app ready: No servers to restore'); + } + } else { + log.info('MCP auto-start disabled in settings - skipping server restoration'); + } + global.mcpServersRestored = true; + } catch (error) { + log.error('Error auto-restoring MCP servers on app ready:', error); + } + + // Start MCP HTTP Proxy service for browser support + try { + log.info('🚀 Starting MCP HTTP proxy for browser support...'); + const MCPProxyService = require('./mcpProxyService.cjs'); + const mcpProxyService = new MCPProxyService(mcpService); + await mcpProxyService.start(8092); + log.info('✅ MCP HTTP proxy service started successfully on port 8092'); + } catch (proxyError) { + log.error('❌ Error starting MCP HTTP proxy:', proxyError); + } + } + }); + + // Handle app close request + ipcMain.on('app-close', async (event) => { + log.info('App close requested from renderer'); + + try { + // Set quitting flag first + isQuitting = true; + + // Close the main window gracefully if it exists + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.removeAllListeners('close'); // Remove close handler to allow closing + mainWindow.close(); + } + + // Give a small delay for the IPC response to be sent back + // before actually quitting the app + setTimeout(() => { + app.quit(); + }, 100); + } catch (error) { + log.error('Error during app close:', error); + // Force quit if there's an error + app.quit(); + } + }); + + // Add IPC handler for tray control + ipcMain.on('hide-to-tray', () => { + if (mainWindow) { + mainWindow.hide(); + } + }); + + ipcMain.on('show-from-tray', () => { + if (mainWindow) { + if (mainWindow.isMinimized()) { + mainWindow.restore(); + } + mainWindow.show(); + mainWindow.focus(); + + // CRITICAL FIX: Force webContents focus when showing from tray + if (mainWindow.webContents && !mainWindow.webContents.isDestroyed()) { + setTimeout(() => { + if (mainWindow && !mainWindow.isDestroyed() && mainWindow.webContents) { + mainWindow.webContents.focus(); + } + }, 100); + } + } else { + createMainWindow(); + } + }); + + // Window management handlers + ipcMain.handle('get-fullscreen-startup-preference', async () => { + const { startFullscreen } = getStartupPreferences(); + return startFullscreen; + }); + + ipcMain.handle('set-fullscreen-startup-preference', async (event, enabled) => { + try { + const updated = ensureStartupSettingsManager().updateSettings({ startFullscreen: !!enabled }); + log.info(`Fullscreen startup preference set to: ${!!updated.startFullscreen}`); + return true; + } catch (error) { + log.error('Error saving fullscreen startup preference:', error); + return false; + } + }); + + ipcMain.handle('toggle-fullscreen', async () => { + if (mainWindow && !mainWindow.isDestroyed()) { + const isFullscreen = mainWindow.isFullScreen(); + mainWindow.setFullScreen(!isFullscreen); + log.info(`Window fullscreen toggled to: ${!isFullscreen}`); + return !isFullscreen; + } + return false; + }); + + ipcMain.handle('get-fullscreen-status', async () => { + if (mainWindow && !mainWindow.isDestroyed()) { + return mainWindow.isFullScreen(); + } + return false; + }); + + // Docker Desktop startup handler + ipcMain.handle('start-docker-desktop', async () => { + try { + log.info('Received request to start Docker Desktop from onboarding'); + + // Get Docker path + const dockerPath = await checkDockerDesktopInstalled(); + if (!dockerPath) { + return { + success: false, + error: 'Docker Desktop not found on system' + }; + } + + // Check if Docker is already running + const isRunning = dockerSetup ? await dockerSetup.isDockerRunning() : false; + if (isRunning) { + return { + success: true, + message: 'Docker Desktop is already running' + }; + } + + // Start Docker Desktop + const startSuccess = await startDockerDesktop(dockerPath); + + if (startSuccess) { + return { + success: true, + message: 'Docker Desktop startup initiated' + }; + } else { + return { + success: false, + error: 'Failed to start Docker Desktop' + }; + } + } catch (error) { + log.error('Error starting Docker Desktop:', error); + return { + success: false, + error: error.message || 'Unknown error starting Docker Desktop' + }; + } + }); + + // Generic Docker service control handlers + ipcMain.handle('start-docker-service', async (event, serviceName) => { + try { + if (!dockerSetup) { + throw new Error('Docker setup not initialized'); + } + + switch (serviceName) { + case 'n8n': + // Get N8N configuration (creates it if needed) + const n8nConfig = getN8NConfig(); + await dockerSetup.startContainer(n8nConfig); + break; + case 'python': + // Get Python configuration (creates it if needed) + const pythonConfig = getPythonConfig(); + await dockerSetup.startContainer(pythonConfig); + break; + case 'comfyui': + // Get ComfyUI configuration (creates it if needed) + const comfyuiConfig = getComfyUIConfig(); + await dockerSetup.startContainer(comfyuiConfig); + break; + default: + throw new Error(`Unknown service: ${serviceName}`); + } + + return { success: true }; + } catch (error) { + log.error(`Error starting ${serviceName} service:`, error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('stop-docker-service', async (event, serviceName) => { + try { + if (!dockerSetup) { + throw new Error('Docker setup not initialized'); + } + + const containerName = `clara_${serviceName}`; + const container = await dockerSetup.docker.getContainer(containerName); + await container.stop(); + + return { success: true }; + } catch (error) { + log.error(`Error stopping ${serviceName} service:`, error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('restart-docker-service', async (event, serviceName) => { + try { + if (!dockerSetup) { + throw new Error('Docker setup not initialized'); + } + + const containerName = `clara_${serviceName}`; + const container = await dockerSetup.docker.getContainer(containerName); + await container.restart(); + + return { success: true }; + } catch (error) { + log.error(`Error restarting ${serviceName} service:`, error); + return { success: false, error: error.message }; + } + }); + + // Remote Docker connection IPC handlers + ipcMain.handle('docker-get-connection-info', async () => { + try { + if (!dockerSetup) { + return { mode: 'local', config: null, activeTunnels: [], isRemote: false }; + } + return dockerSetup.getConnectionInfo(); + } catch (error) { + log.error('Error getting Docker connection info:', error); + return { error: error.message }; + } + }); + + ipcMain.handle('docker-test-remote-connection', async (event, config) => { + try { + if (!dockerSetup) { + throw new Error('Docker setup not initialized'); + } + return await dockerSetup.testRemoteConnection(config); + } catch (error) { + log.error('Error testing remote Docker connection:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('docker-switch-connection', async (event, config) => { + try { + if (!dockerSetup) { + throw new Error('Docker setup not initialized'); + } + const result = await dockerSetup.switchConnection(config); + + // If switching to remote and successful, set up SSH tunnels for services + if (result.success && config.mode === 'remote') { + // Create tunnels for each service that will be used + const services = [ + { name: 'python', localPort: 5001, remotePort: 5001 }, + { name: 'n8n', localPort: 5678, remotePort: 5678 }, + { name: 'comfyui', localPort: 8188, remotePort: 8188 } + ]; + + for (const service of services) { + try { + await dockerSetup.createSSHTunnel(service.localPort, service.remotePort, service.name); + } catch (tunnelError) { + log.warn(`Failed to create SSH tunnel for ${service.name}:`, tunnelError); + } + } + } + + return result; + } catch (error) { + log.error('Error switching Docker connection:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('docker-create-ssh-tunnel', async (event, { localPort, remotePort, serviceName }) => { + try { + if (!dockerSetup) { + throw new Error('Docker setup not initialized'); + } + await dockerSetup.createSSHTunnel(localPort, remotePort, serviceName); + return { success: true }; + } catch (error) { + log.error(`Error creating SSH tunnel for ${serviceName}:`, error); + return { success: false, error: error.message }; + } + }); + + // ClaraCore Remote Deployment IPC Handlers + ipcMain.handle('claracore-remote-test-setup', async (event, config) => { + try { + log.info('Testing ClaraCore remote setup...'); + const service = new ClaraCoreRemoteService(); + const result = await service.testSetup(config); + return result; + } catch (error) { + log.error('ClaraCore remote test error:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('claracore-remote-deploy', async (event, config) => { + try { + log.info('Deploying ClaraCore to remote server...'); + + // Show confirmation dialog for sudo operations + const { dialog } = require('electron'); + const deploymentMethod = config.hardwareType === 'cuda' ? 'Docker' : 'Native Installation'; + const methodDetails = config.hardwareType === 'cuda' + ? 'The deployment will install Docker and NVIDIA Container Toolkit on the remote server.' + : 'The deployment will install ClaraCore natively using the official installation script.'; + + const { response } = await dialog.showMessageBox({ + type: 'warning', + title: 'Sudo Password Confirmation', + message: 'ClaraCore deployment requires sudo privileges', + detail: `Deployment Method: ${deploymentMethod}\n${methodDetails}\n\nServer: ${config.host}\n\nYour SSH password will be used for sudo commands. Continue?`, + buttons: ['Continue', 'Cancel'], + defaultId: 0, + cancelId: 1, + icon: null + }); + + if (response === 1) { + log.info('User cancelled deployment'); + return { success: false, error: 'Deployment cancelled by user' }; + } + + // Stop local ClaraCore services before deploying to remote + log.info('🛑 Stopping local ClaraCore services before remote deployment...'); + const stopResult = await stopAllLocalServices('claracore'); + + if (stopResult.stopped.length > 0) { + log.info(`✅ Stopped: ${stopResult.stopped.join(', ')}`); + } + if (stopResult.errors.length > 0) { + log.warn(`⚠️ Some services had errors during stop (continuing): ${JSON.stringify(stopResult.errors)}`); + } + + const service = new ClaraCoreRemoteService(); + const result = await service.deploy(config); + + if (result.success) { + log.info('✅ Successfully deployed ClaraCore to remote server'); + result.localServicesStopped = stopResult.stopped; + + // Automatically switch ClaraCore service to remote mode + try { + log.info('🔄 Auto-configuring ClaraCore service to use remote deployment...'); + + // Set service to remote mode with the deployed URL + if (serviceConfigManager) { + await serviceConfigManager.setServiceConfig('claracore', 'remote', result.url); + log.info(`✅ ClaraCore service configured: mode=remote, url=${result.url}`); + } + + // Update CentralServiceManager state + if (centralServiceManager) { + // Get the service to update its deployment mode from config + const service = centralServiceManager.services.get('claracore'); + if (service) { + service.deploymentMode = 'remote'; + } + + centralServiceManager.setServiceUrl('claracore', result.url); + centralServiceManager.setState('claracore', centralServiceManager.states.RUNNING); + log.info('✅ CentralServiceManager updated: ClaraCore is now RUNNING in remote mode'); + } + + // Notify frontend to update Clara's Core provider + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('claracore:remote-deployed', { + url: result.url, + hardwareType: config.hardwareType + }); + log.info('✅ Notified frontend to update Clara\'s Core provider'); + } + + result.autoConfigured = true; + } catch (configError) { + log.error('Failed to auto-configure service (deployment still successful):', configError); + result.autoConfigured = false; + result.configError = configError.message; + } + } + + return result; + } catch (error) { + log.error('ClaraCore remote deployment error:', error); + return { success: false, error: error.message }; + } + }); + + // Monitor remote ClaraCore services + ipcMain.handle('claracore-remote:monitor', async (event, config) => { + try { + log.info('Monitoring remote ClaraCore services...'); + const service = new ClaraCoreRemoteService(); + const result = await service.monitorRemoteServices(config); + return result; + } catch (error) { + log.error('Remote monitoring error:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('docker-close-ssh-tunnel', async (event, serviceName) => { + try { + if (!dockerSetup) { + throw new Error('Docker setup not initialized'); + } + await dockerSetup.closeSSHTunnel(serviceName); + return { success: true }; + } catch (error) { + log.error(`Error closing SSH tunnel for ${serviceName}:`, error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('docker-get-active-tunnels', async () => { + try { + if (!dockerSetup) { + return []; + } + return dockerSetup.getActiveTunnels(); + } catch (error) { + log.error('Error getting active SSH tunnels:', error); + return []; + } + }); + + // Screen sharing IPC handlers for Electron + ipcMain.handle('get-desktop-sources', async () => { + try { + const sources = await desktopCapturer.getSources({ + types: ['screen', 'window'], + thumbnailSize: { width: 300, height: 200 } + }); + + return sources.map(source => ({ + id: source.id, + name: source.name, + thumbnail: source.thumbnail.toDataURL() + })); + } catch (error) { + log.error('Error getting desktop sources:', error); + return { error: error.message }; + } + }); + + ipcMain.handle('get-screen-access-status', async () => { + try { + if (process.platform === 'darwin') { + const status = systemPreferences.getMediaAccessStatus('screen'); + return { status }; + } + // On Windows/Linux, screen access is generally available + return { status: 'granted' }; + } catch (error) { + log.error('Error checking screen access:', error); + return { status: 'unknown', error: error.message }; + } + }); + + ipcMain.handle('request-screen-access', async () => { + try { + if (process.platform === 'darwin') { + const granted = await systemPreferences.askForMediaAccess('screen'); + return { granted }; + } + // On Windows/Linux, screen access is generally available + return { granted: true }; + } catch (error) { + log.error('Error requesting screen access:', error); + return { granted: false, error: error.message }; + } + }); +} + +/** + * Check if Docker Desktop is installed on Windows/macOS/Linux + */ +async function checkDockerDesktopInstalled() { + try { + const { exec } = require('child_process'); + const { promisify } = require('util'); + const execAsync = promisify(exec); + + if (process.platform === 'win32') { + // Windows Docker Desktop detection + const possiblePaths = [ + 'C:\\Program Files\\Docker\\Docker\\Docker Desktop.exe', + 'C:\\Program Files (x86)\\Docker\\Docker\\Docker Desktop.exe' + ]; + + for (const dockerPath of possiblePaths) { + if (fs.existsSync(dockerPath)) { + return dockerPath; + } + } + + // Also check via registry or Windows features + try { + await execAsync('docker --version'); + return true; // Docker CLI is available + } catch (error) { + // Docker CLI not available + } + + return false; + + } else if (process.platform === 'darwin') { + // macOS Docker Desktop detection + const dockerAppPath = '/Applications/Docker.app'; + + // Check if Docker.app exists + if (fs.existsSync(dockerAppPath)) { + return dockerAppPath; + } + + // Also check if Docker CLI is available (could be installed via Homebrew or other methods) + try { + await execAsync('docker --version'); + return true; // Docker CLI is available + } catch (error) { + // Docker CLI not available + } + + return false; + + } else if (process.platform === 'linux') { + // Linux Docker Desktop detection + const possiblePaths = [ + '/opt/docker-desktop/bin/docker-desktop', + '/usr/bin/docker-desktop', + '/usr/local/bin/docker-desktop' + ]; + + // Check for Docker Desktop executable + for (const dockerPath of possiblePaths) { + if (fs.existsSync(dockerPath)) { + return dockerPath; + } + } + + // Check if Docker Desktop is installed via package manager + try { + await execAsync('which docker-desktop'); + return 'docker-desktop'; // Docker Desktop is in PATH + } catch (error) { + // Docker Desktop not found in PATH + } + + // Check if Docker CLI is available (could be Docker Engine or Docker Desktop) + try { + await execAsync('docker --version'); + + // Try to determine if it's Docker Desktop by checking for desktop-specific features + try { + const { stdout } = await execAsync('docker context ls --format json'); + const contexts = stdout.trim().split('\n').map(line => JSON.parse(line)); + const hasDesktopContext = contexts.some(ctx => + ctx.Name === 'desktop-linux' || + ctx.DockerEndpoint && ctx.DockerEndpoint.includes('desktop') + ); + + if (hasDesktopContext) { + return 'docker-desktop'; // Docker Desktop detected via context + } + } catch (contextError) { + // Context check failed, continue with regular Docker check + } + + return true; // Docker CLI is available (could be Docker Engine) + } catch (error) { + // Docker CLI not available + } + + return false; + + } else { + // Other platforms - just check for Docker CLI + try { + await execAsync('docker --version'); + return true; + } catch (error) { + return false; + } + } + + } catch (error) { + log.error('Error checking Docker Desktop installation:', error); + return false; + } +} + +/** + * Attempt to start Docker Desktop on Windows/macOS/Linux + */ +async function startDockerDesktop(dockerPath) { + try { + const { spawn, exec } = require('child_process'); + const { promisify } = require('util'); + const execAsync = promisify(exec); + + if (process.platform === 'win32') { + // Windows Docker Desktop startup + if (typeof dockerPath === 'string' && dockerPath.endsWith('.exe')) { + // Start Docker Desktop executable + const dockerProcess = spawn(dockerPath, [], { + detached: true, + stdio: 'ignore' + }); + dockerProcess.on('error', (error) => { + log.warn('Docker Desktop spawn error:', error.message); + }); + dockerProcess.unref(); + + log.info('Docker Desktop startup initiated'); + return true; + } else { + // Try to start via Windows service or PowerShell + try { + await execAsync('Start-Process "Docker Desktop" -WindowStyle Hidden', { shell: 'powershell' }); + log.info('Docker Desktop startup initiated via PowerShell'); + return true; + } catch (error) { + log.warn('Failed to start Docker Desktop via PowerShell:', error.message); + return false; + } + } + + } else if (process.platform === 'darwin') { + // macOS Docker Desktop startup + if (typeof dockerPath === 'string' && dockerPath.endsWith('.app')) { + // Start Docker.app using 'open' command + try { + await execAsync(`open "${dockerPath}"`); + log.info('Docker Desktop startup initiated via open command'); + return true; + } catch (error) { + log.warn('Failed to start Docker Desktop via open command:', error.message); + return false; + } + } else { + // Try alternative methods to start Docker + try { + // Check if Docker is already running first + try { + await execAsync('docker info', { timeout: 5000 }); + log.info('Docker is already running'); + return true; + } catch (checkError) { + // Docker not running, try to start it + } + + // Try using 'open' with application name + await execAsync('open -a Docker'); + log.info('Docker Desktop startup initiated via open -a Docker'); + return true; + } catch (error) { + try { + // Try using launchctl (if Docker is set up as a service) + await execAsync('launchctl load ~/Library/LaunchAgents/com.docker.docker.plist 2>/dev/null || true'); + await execAsync('launchctl start com.docker.docker'); + log.info('Docker Desktop startup initiated via launchctl'); + return true; + } catch (launchError) { + log.warn('Failed to start Docker Desktop via launchctl:', launchError.message); + + // Final attempt: try to start Docker via Spotlight/Launch Services + try { + await execAsync('osascript -e \'tell application "Docker" to activate\''); + log.info('Docker Desktop startup initiated via AppleScript'); + return true; + } catch (scriptError) { + log.warn('All Docker startup methods failed'); + return false; + } + } + } + } + + } else if (process.platform === 'linux') { + // Linux Docker Desktop startup + if (typeof dockerPath === 'string' && dockerPath.includes('docker-desktop')) { + // Start Docker Desktop executable directly + try { + if (dockerPath === 'docker-desktop') { + // Docker Desktop is in PATH + const dockerProcess = spawn('docker-desktop', [], { + detached: true, + stdio: 'ignore' + }); + dockerProcess.on('error', (error) => { + log.warn('Docker Desktop spawn error:', error.message); + }); + dockerProcess.unref(); + } else { + // Docker Desktop is at specific path + const dockerProcess = spawn(dockerPath, [], { + detached: true, + stdio: 'ignore' + }); + dockerProcess.on('error', (error) => { + log.warn('Docker Desktop spawn error:', error.message); + }); + dockerProcess.unref(); + } + + log.info('Docker Desktop startup initiated via executable'); + return true; + } catch (error) { + log.warn('Failed to start Docker Desktop via executable:', error.message); + } + } + + // Try alternative methods to start Docker Desktop on Linux + try { + // Check if Docker is already running first + try { + await execAsync('docker info', { timeout: 5000 }); + log.info('Docker is already running'); + return true; + } catch (checkError) { + // Docker not running, try to start it + } + + // Try to start Docker Desktop via desktop entry + try { + await execAsync('gtk-launch docker-desktop || true'); + log.info('Docker Desktop startup initiated via gtk-launch'); + return true; + } catch (gtkError) { + // gtk-launch failed, try other methods + } + + // Try to start via XDG desktop entry + try { + await execAsync('xdg-open /usr/share/applications/docker-desktop.desktop || true'); + log.info('Docker Desktop startup initiated via xdg-open'); + return true; + } catch (xdgError) { + // XDG method failed + } + + // Try to start Docker service as fallback (Docker Engine) + try { + await execAsync('sudo systemctl start docker'); + log.info('Docker service startup initiated via systemctl'); + return true; + } catch (systemctlError) { + log.warn('Failed to start Docker service via systemctl:', systemctlError.message); + } + + return false; + } catch (error) { + log.warn('All Linux Docker startup methods failed:', error.message); + return false; + } + + } else { + // Other platforms - try to start docker service + try { + await execAsync('sudo systemctl start docker'); + log.info('Docker service startup initiated via systemctl'); + return true; + } catch (error) { + log.warn('Failed to start Docker service via systemctl:', error.message); + return false; + } + } + + } catch (error) { + log.error('Error starting Docker Desktop:', error); + return false; + } +} + +/** + * Ask user if they want to start Docker Desktop when it's not running + */ +async function askToStartDockerDesktop(loadingScreen) { + try { + const dockerPath = await checkDockerDesktopInstalled(); + + if (!dockerPath) { + log.info('Docker Desktop not detected on system'); + + // Show dialog asking user to install Docker Desktop + const platformName = process.platform === 'darwin' ? 'macOS' : process.platform === 'win32' ? 'Windows' : 'Linux'; + const downloadUrl = process.platform === 'darwin' + ? 'https://docs.docker.com/desktop/install/mac-install/' + : process.platform === 'win32' + ? 'https://docs.docker.com/desktop/install/windows-install/' + : 'https://docs.docker.com/desktop/install/linux-install/'; + + const result = await showStartupDialog( + loadingScreen, + 'info', + 'Docker Desktop Not Installed', + `Docker Desktop is not installed on your system. Docker Desktop enables advanced features like ComfyUI, n8n workflows, and other AI services.\n\nWould you like to:\n\n• Download Docker Desktop for ${platformName}\n• Continue without Docker (lightweight mode)\n• Cancel startup`, + ['Download Docker Desktop', 'Continue without Docker', 'Cancel'] + ); + + if (result.response === 0) { // Download Docker Desktop + const { shell } = require('electron'); + shell.openExternal(downloadUrl); + + await showStartupDialog( + loadingScreen, + 'info', + 'Docker Installation', + 'Docker Desktop download page has been opened in your browser.\n\nAfter installing Docker Desktop, please restart Clara to enable all features.', + ['OK'] + ); + + return false; + } else if (result.response === 1) { // Continue without Docker + return false; + } else { // Cancel + app.quit(); + return false; + } + } + + log.info('Docker Desktop is installed but not running'); + + // Show dialog asking user if they want to start Docker Desktop + const platformName = process.platform === 'darwin' ? 'macOS' : process.platform === 'win32' ? 'Windows' : 'Linux'; + const result = await showStartupDialog( + loadingScreen, + 'question', + 'Docker Desktop Not Running', + `Docker Desktop is installed but not currently running on ${platformName}. Would you like to start it now?\n\nStarting Docker Desktop will enable advanced features like ComfyUI, n8n workflows, and other AI services.`, + ['Start Docker Desktop', 'Continue without Docker', 'Cancel'] + ); + + if (result.response === 0) { // Start Docker Desktop + loadingScreen?.setStatus('Starting Docker Desktop...', 'info'); + + const startSuccess = await startDockerDesktop(dockerPath); + + if (startSuccess) { + loadingScreen?.setStatus('Docker Desktop is starting... Please wait...', 'info'); + + // Wait for Docker Desktop to start (up to 60 seconds) + let dockerStarted = false; + const maxWaitTime = 60000; // 60 seconds + const checkInterval = 2000; // 2 seconds + const maxAttempts = maxWaitTime / checkInterval; + + for (let attempt = 0; attempt < maxAttempts; attempt++) { + await new Promise(resolve => setTimeout(resolve, checkInterval)); + + try { + const tempDockerSetup = new DockerSetup(); + const isRunning = await tempDockerSetup.isDockerRunning(); + + if (isRunning) { + dockerStarted = true; + loadingScreen?.setStatus('Docker Desktop started successfully!', 'success'); + log.info('Docker Desktop started successfully'); + break; + } else { + loadingScreen?.setStatus(`Waiting for Docker Desktop to start... (${Math.round((attempt + 1) * checkInterval / 1000)}s)`, 'info'); + } + } catch (error) { + // Continue waiting + loadingScreen?.setStatus(`Waiting for Docker Desktop to start... (${Math.round((attempt + 1) * checkInterval / 1000)}s)`, 'info'); + } + } + + if (!dockerStarted) { + loadingScreen?.setStatus('Docker Desktop is taking longer than expected to start. Continuing without Docker...', 'warning'); + await showStartupDialog( + loadingScreen, + 'warning', + 'Docker Startup Timeout', + 'Docker Desktop is taking longer than expected to start. The application will continue in lightweight mode.\n\nYou can try restarting the application once Docker Desktop is fully running.', + ['OK'] + ); + return false; + } + + return dockerStarted; + } else { + loadingScreen?.setStatus('Failed to start Docker Desktop. Continuing without Docker...', 'warning'); + await showStartupDialog( + loadingScreen, + 'warning', + 'Docker Startup Failed', + 'Failed to start Docker Desktop automatically. The application will continue in lightweight mode.\n\nYou can manually start Docker Desktop and restart the application to enable full features.', + ['OK'] + ); + return false; + } + + } else if (result.response === 1) { // Continue without Docker + loadingScreen?.setStatus('Continuing without Docker...', 'info'); + log.info('User chose to continue without Docker'); + return false; + + } else { // Cancel + loadingScreen?.setStatus('Startup cancelled by user', 'warning'); + log.info('User cancelled startup'); + app.quit(); + return false; + } + + } catch (error) { + log.error('Error in askToStartDockerDesktop:', error); + return false; + } +} + +/** + * Main initialization function that determines startup flow based on Docker availability + * and initializes all necessary services + */ +async function initialize() { + try { + console.log('🚀 Starting application initialization (fast mode)'); + + // Check if this is first time launch + const featureSelection = new FeatureSelectionScreen(); + let selectedFeatures = null; + + if (featureSelection.isFirstTimeLaunch()) { + console.log('🎯 First time launch detected - will show onboarding in main app'); + // DO NOT auto-start any services on first launch - wait for user consent + selectedFeatures = { + comfyUI: false, + n8n: false, + ragAndTts: false, + claraCore: true // Only Clara Core is always enabled + }; + // Mark that we need to show onboarding in the main app + global.needsFeatureSelection = true; + } else { + // Load existing feature configuration (user has completed onboarding) + selectedFeatures = FeatureSelectionScreen.getCurrentConfig(); + console.log('📋 Loaded existing feature configuration:', selectedFeatures); + global.needsFeatureSelection = false; + } + + // Store selected features globally for use throughout initialization + global.selectedFeatures = selectedFeatures; + + // Skip loading screen - go directly to main window creation + console.log('⚡ Fast startup mode - skipping splash screen'); + + // Register handlers early (needed for IPC communication) + if (!global.handlersRegistered) { + registerHandlers(); + global.handlersRegistered = true; + } + + // Create main window immediately for fast startup + console.log('📱 Creating main window immediately...'); + await createMainWindow(); + + // Send initial app state to renderer + mainWindow.webContents.once('did-finish-load', () => { + mainWindow.webContents.send('app-initialization-state', { + needsFeatureSelection: global.needsFeatureSelection, + selectedFeatures: selectedFeatures, + status: 'initializing' + }); + }); + + // Initialize everything else in the background + initializeInBackground(selectedFeatures); + + } catch (error) { + log.error(`Initialization error: ${error.message}`, error); + // Create main window even if initialization fails + if (!mainWindow) { + await createMainWindow(); + } + // Send error state to renderer + mainWindow.webContents.send('app-initialization-state', { + status: 'error', + error: error.message + }); + } +} + +/** + * Background initialization function that runs after main window is shown + */ +async function initializeInBackground(selectedFeatures) { + // Set initialization flag + initializationInProgress = true; + + try { + // Send status update + const sendStatusUpdate = (status, details = {}) => { + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('service-status-update', { status, ...details }); + } + }; + + // Validate system resources + sendStatusUpdate('validating', { message: 'Validating system resources...' }); + let systemConfig; + try { + const platformManager = new PlatformManager(path.join(__dirname, 'llamacpp-binaries')); + systemConfig = await platformManager.validateSystemResources(); + + // Handle critical OS compatibility issues + if (systemConfig.osCompatibility && !systemConfig.osCompatibility.isSupported) { + log.error('🚨 Critical OS compatibility issue detected'); + systemConfig.performanceMode = 'core-only'; + systemConfig.enabledFeatures = { + claraCore: false, + dockerServices: false, + comfyUI: false, + advancedFeatures: false + }; + sendStatusUpdate('warning', { + message: 'OS compatibility issue - Limited functionality', + osCompatibility: systemConfig.osCompatibility + }); + } + } catch (error) { + log.error('System resource validation failed:', error); + systemConfig = null; + } + + global.systemConfig = systemConfig; + + // Initialize service configuration managers + sendStatusUpdate('initializing', { message: 'Initializing service configuration...' }); + try { + serviceConfigManager = new ServiceConfigurationManager(); + centralServiceManager = new CentralServiceManager(serviceConfigManager); + + const { SERVICE_DEFINITIONS } = require('./serviceDefinitions.cjs'); + Object.keys(SERVICE_DEFINITIONS).forEach(serviceName => { + const serviceDefinition = SERVICE_DEFINITIONS[serviceName]; + centralServiceManager.registerService(serviceName, serviceDefinition); + }); + + // Auto-start critical core services (ClaraCore) regardless of user settings + // ClaraCore is essential and should always run + try { + sendStatusUpdate('starting-claracore', { message: 'Starting Clara Core AI Engine...' }); + + // Check which mode Clara Core was last used in + const claraCoreMode = serviceConfigManager.getServiceMode('claracore') || 'local'; + log.info(`🔍 Clara Core deployment mode: ${claraCoreMode}`); + + if (claraCoreMode === 'docker') { + // Start in Docker mode + log.info('Starting Clara Core in Docker mode...'); + const ClaraCoreDockerService = require('./claraCoreDockerService.cjs'); + const dockerService = new ClaraCoreDockerService(); + await dockerService.start(); + log.info('✅ Clara Core AI Engine started in Docker mode'); + } else if (claraCoreMode === 'remote') { + // Remote mode - don't start, just log + log.info('Clara Core is configured in Remote mode, skipping local startup'); + } else { + // Start in Local binary mode (default) + log.info('Starting Clara Core in Local binary mode...'); + await centralServiceManager.startService('claracore'); + log.info('✅ Clara Core AI Engine started in Local mode'); + } + } catch (claraCoreError) { + log.error('❌ Failed to start Clara Core AI Engine:', claraCoreError); + // Continue with app startup even if ClaraCore fails + } + } catch (error) { + log.warn('Service configuration managers initialization failed:', error); + } + + // Check Docker availability + sendStatusUpdate('checking-docker', { message: 'Checking Docker availability...' }); + dockerSetup = new DockerSetup(); + let isDockerAvailable = false; + + if (!systemConfig || systemConfig.enabledFeatures.dockerServices !== false) { + isDockerAvailable = await dockerSetup.isDockerRunning(); + } + + // Always ensure core binaries are available (regardless of consent status) + // This is essential for Clara Core to function properly + sendStatusUpdate('ensuring-binaries', { message: 'Ensuring core binaries are available...' }); + + // Only initialize services if user has completed onboarding and given consent + const hasUserConsent = !global.needsFeatureSelection; + + if (hasUserConsent) { + // Check if user has enabled auto-start for services + let shouldAutoStartServices = false; + try { + // Since we can't easily access the frontend db from main process, + // and the default is false (which is what we want for security), + // we'll default to false unless explicitly set + shouldAutoStartServices = false; + + // TODO: In the future, we could save startup preferences to a separate file + // that both frontend and backend can access, or use IPC communication + } catch (error) { + log.warn('Could not check auto-start preference, defaulting to false:', error); + shouldAutoStartServices = false; + } + + if (shouldAutoStartServices) { + // User has explicitly enabled auto-start - initialize services + console.log('✅ User consent obtained and auto-start enabled - initializing selected services'); + if (isDockerAvailable) { + sendStatusUpdate('docker-available', { message: 'Docker detected - Setting up services...' }); + await initializeServicesWithDocker(selectedFeatures, sendStatusUpdate); + } else { + sendStatusUpdate('docker-not-available', { message: 'Docker not available - Running in lightweight mode...' }); + await initializeServicesWithoutDocker(selectedFeatures, sendStatusUpdate); + } + } else { + // User has consent but auto-start is disabled - wait for manual service start + console.log('✅ User consent obtained but auto-start disabled - services available on demand'); + sendStatusUpdate('consent-no-autostart', { + message: 'Services available - start them manually when needed', + dockerAvailable: isDockerAvailable + }); + } + } else { + // First time launch - wait for user to complete onboarding + console.log('⏳ First time launch - waiting for user consent before starting services'); + sendStatusUpdate('waiting-for-consent', { + message: 'Waiting for user to complete onboarding before starting services...', + dockerAvailable: isDockerAvailable + }); + } + + // Initialize ClaraVerse Scheduler Service + sendStatusUpdate('initializing-scheduler', { message: 'Initializing task scheduler...' }); + try { + if (!schedulerService) { + schedulerService = new SchedulerElectronService(mainWindow); + log.info('✅ ClaraVerse Scheduler initialized successfully'); + } + } catch (error) { + log.error('❌ Failed to initialize scheduler service:', error); + // Continue without scheduler if it fails + } + + sendStatusUpdate('ready', { message: 'All services initialized' }); + + // Mark initialization as complete + initializationComplete = true; + initializationInProgress = false; + + } catch (error) { + log.error('Background initialization error:', error); + initializationInProgress = false; // Reset flag even on error + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('service-status-update', { + status: 'error', + error: error.message + }); + } + } +} + +/** + * Initialize services with Docker support + */ +async function initializeServicesWithDocker(selectedFeatures, sendStatusUpdate) { + try { + // Initialize core services + updateService = platformUpdateService; + + if (selectedFeatures.ragAndTts) { + mcpService = new MCPService(); + } + + // Initialize Docker services + sendStatusUpdate('docker-initializing', { message: 'Setting up Docker containers...' }); + await dockerSetup.setup(selectedFeatures, (status) => { + sendStatusUpdate('docker-setup', { message: status }); + }); + + // Start background services + sendStatusUpdate('starting-services', { message: 'Starting background services...' }); + await initializeServicesInBackground(); + + // Initialize watchdog + watchdogService = new WatchdogService(dockerSetup, mcpService); + watchdogService.start(); + + } catch (error) { + log.error('Error initializing services with Docker:', error); + throw error; + } +} + +/** + * Initialize services without Docker + */ +async function initializeServicesWithoutDocker(selectedFeatures, sendStatusUpdate) { + try { + // Initialize only essential services + updateService = platformUpdateService; + + if (selectedFeatures.ragAndTts) { + mcpService = new MCPService(); + } + + // Start background services + sendStatusUpdate('starting-services', { message: 'Starting services...' }); + await initializeServicesInBackground(); + + } catch (error) { + log.error('Error initializing services without Docker:', error); + throw error; + } +} + +async function initializeWithDocker() { + try { + // Register handlers for various app functions (only if not already registered) + if (!global.handlersRegistered) { + registerHandlers(); + global.handlersRegistered = true; + } + + // Get user's feature selections - be conservative during onboarding + const selectedFeatures = global.selectedFeatures || { + comfyUI: false, // Conservative default - only start if explicitly selected + n8n: false, // Conservative default - only start if explicitly selected + ragAndTts: false, // Conservative default - prevent unwanted Python backend downloads + claraCore: false // Always enable core functionality + }; + const systemConfig = global.systemConfig; + + // Check system configuration before initializing services + if (systemConfig && !systemConfig.enabledFeatures.dockerServices) { + log.warn('🔧 Docker services disabled due to system resource limitations, falling back to lightweight mode'); + return await initializeWithoutDocker(); + } + + // Initialize MCP service only if RAG & TTS is selected + if (selectedFeatures && selectedFeatures.ragAndTts) { + log.info('🧠 Initializing MCP service (RAG & TTS enabled)'); + mcpService = new MCPService(); + } else { + log.info('🧠 MCP service disabled (RAG & TTS not selected)'); + } + + + // Only initialize ComfyUI if selected by user AND system supports it + if (selectedFeatures && selectedFeatures.comfyUI && + (!systemConfig || systemConfig.enabledFeatures.comfyUI)) { + log.info('🎨 Initializing ComfyUI service (selected by user)'); + comfyUIModelService = new ComfyUIModelService(); + } else { + // Always initialize the model service for model downloads, even if ComfyUI isn't enabled + log.info('🎨 Initializing ComfyUI Model Service for model downloads'); + comfyUIModelService = new ComfyUIModelService(); + + if (!selectedFeatures?.comfyUI) { + log.info('🎨 ComfyUI UI disabled (not selected by user) - but model downloads available'); + } else { + log.info('🎨 ComfyUI UI disabled due to system resource limitations - but model downloads available'); + } + } + + + + // Setup Docker services with progress updates to splash screen + loadingScreen.setStatus('Setting up Docker environment...', 'info'); + + const dockerSuccess = await dockerSetup.setup(selectedFeatures, async (status, type = 'info', progress = null) => { + loadingScreen.setStatus(status, type, progress); + + // Also log to console + if (progress && progress.percentage) { + console.log(`[Docker Setup] ${status} (${progress.percentage}%)`); + } else { + console.log(`[Docker Setup] ${status}`); + } + }); + + if (dockerSuccess) { + loadingScreen.setStatus('Docker services ready - Starting application...', 'success'); + } else { + loadingScreen.setStatus('Docker setup incomplete - Starting in limited mode...', 'warning'); + } + + // Create the main window + loadingScreen.setStatus('Loading main application...', 'info'); + await createMainWindow(); + + // Initialize remaining services in background + initializeServicesInBackground(); + + } catch (error) { + log.error(`Docker initialization error: ${error.message}`, error); + loadingScreen?.setStatus(`Docker setup failed: ${error.message}`, 'error'); + + // Fallback to lightweight mode + setTimeout(async () => { + await initializeWithoutDocker(); + }, 2000); + } +} + +async function initializeWithoutDocker() { + try { + // Register handlers for various app functions (only if not already registered) + if (!global.handlersRegistered) { + registerHandlers(); + global.handlersRegistered = true; + } + + // Initialize essential services only + mcpService = new MCPService(); + updateService = platformUpdateService; + + + // Create the main window immediately for fast startup + loadingScreen.setStatus('Starting main application...', 'success'); + await createMainWindow(); + + // Initialize lightweight services in background + initializeLightweightServicesInBackground(); + + } catch (error) { + log.error(`Lightweight initialization error: ${error.message}`, error); + loadingScreen?.setStatus(`Error: ${error.message}`, 'error'); + + // For critical startup errors, create main window anyway and show error + await createMainWindow(); + await showStartupDialog(loadingScreen, 'error', 'Startup Error', `Critical error during startup: ${error.message}\n\nSome features may not work properly.`); + } +} + + +async function continueNormalInitialization() { + // This function is deprecated - replaced by the new two-type startup flow + console.warn('continueNormalInitialization is deprecated - using new startup flow'); + await initialize(); +} + +/** + * Initialize lightweight services in background when Docker is not available + * This provides fast startup with limited functionality + */ +async function initializeLightweightServicesInBackground() { + try { + log.info('Starting lightweight service initialization...'); + + // Send initialization status to renderer if main window is ready + const sendStatus = (service, status, type = 'info') => { + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('background-service-status', { service, status, type }); + } + log.info(`[Lightweight] ${service}: ${status}`); + }; + + + + // Initialize MCP service + sendStatus('MCP', 'Initializing MCP service...', 'info'); + try { + sendStatus('MCP', 'MCP service initialized', 'success'); + + // Check startup settings for MCP auto-start + sendStatus('MCP', 'Checking startup settings...', 'info'); + let shouldAutoStartMCP = true; // Default to true for backward compatibility + + try { + const { autoStartMCP } = getStartupPreferences(); + shouldAutoStartMCP = autoStartMCP; + } catch (settingsError) { + log.warn('Error reading startup settings for MCP auto-start:', settingsError); + // Default to true on error to maintain existing behavior + } + + if (shouldAutoStartMCP) { + // Auto-start previously running servers + sendStatus('MCP', 'Restoring MCP servers...', 'info'); + try { + const restoreResults = await mcpService.startPreviouslyRunningServers(); + const successCount = restoreResults.filter(r => r.success).length; + const totalCount = restoreResults.length; + + if (totalCount > 0) { + sendStatus('MCP', `Restored ${successCount}/${totalCount} MCP servers`, successCount === totalCount ? 'success' : 'warning'); + } else { + sendStatus('MCP', 'No MCP servers to restore', 'info'); + } + global.mcpServersRestored = true; // Mark as restored to prevent duplicate restoration + } catch (restoreError) { + log.error('Error restoring MCP servers:', restoreError); + sendStatus('MCP', 'Failed to restore some MCP servers', 'warning'); + } + } else { + sendStatus('MCP', 'MCP auto-start disabled in settings', 'info'); + log.info('MCP auto-start is disabled in startup settings'); + global.mcpServersRestored = true; // Mark as "restored" to prevent later attempts + } + } catch (mcpError) { + log.error('Error initializing MCP service:', mcpError); + sendStatus('MCP', 'MCP service initialization failed', 'warning'); + } + + // Initialize MCP HTTP Proxy service for browser support + sendStatus('MCP Proxy', 'Starting MCP HTTP proxy...', 'info'); + try { + if (mcpService) { + const MCPProxyService = require('./mcpProxyService.cjs'); + const mcpProxyService = new MCPProxyService(mcpService); + await mcpProxyService.start(8092); + sendStatus('MCP Proxy', 'MCP HTTP proxy started on port 8092', 'success'); + log.info('✅ MCP HTTP proxy service started successfully'); + } else { + log.warn('⚠️ MCP service not available, skipping proxy start'); + sendStatus('MCP Proxy', 'Skipped (MCP service not available)', 'warning'); + } + } catch (proxyError) { + log.error('Error starting MCP HTTP proxy:', proxyError); + sendStatus('MCP Proxy', 'MCP proxy failed to start', 'warning'); + } + + // Initialize Watchdog service (lightweight mode) + sendStatus('Watchdog', 'Initializing Watchdog service...', 'info'); + try { + watchdogService = new WatchdogService(null, mcpService, ipcLogger); // No Docker in lightweight mode + + // Set up event listeners for watchdog events + watchdogService.on('serviceRestored', (serviceKey, service) => { + log.info(`Watchdog: ${service.name} has been restored`); + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('watchdog-service-restored', { serviceKey, service: service.name }); + } + }); + + watchdogService.on('serviceFailed', (serviceKey, service) => { + log.error(`Watchdog: ${service.name} has failed after maximum retry attempts`); + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('watchdog-service-failed', { serviceKey, service: service.name }); + } + }); + + watchdogService.on('serviceRestarted', (serviceKey, service) => { + log.info(`Watchdog: ${service.name} has been restarted successfully`); + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('watchdog-service-restarted', { serviceKey, service: service.name }); + } + }); + + // Start the watchdog monitoring + watchdogService.start(); + + sendStatus('Watchdog', 'Watchdog service started successfully', 'success'); + } catch (watchdogError) { + log.error('Error initializing Watchdog service:', watchdogError); + sendStatus('Watchdog', 'Watchdog service initialization failed', 'warning'); + } + + // Notify that lightweight initialization is complete + sendStatus('System', 'Lightweight initialization complete', 'success'); + log.info('Lightweight service initialization completed'); + + } catch (error) { + log.error('Error during lightweight service initialization:', error); + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('background-service-error', { + service: 'System', + error: `Lightweight initialization error: ${error.message}` + }); + } + } +} + +/** + * Initialize all services in background after main window is ready (Docker mode) + * This provides fast startup while services initialize progressively + */ +async function initializeServicesInBackground() { + try { + log.info('Starting remaining services initialization (Docker mode)...'); + + // Send initialization status to renderer if main window is ready + const sendStatus = (service, status, type = 'info') => { + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('background-service-status', { service, status, type }); + } + log.info(`[Docker Mode] ${service}: ${status}`); + }; + + + // Initialize MCP service in background + sendStatus('MCP', 'Initializing MCP service...', 'info'); + try { + // Initialize MCP service if not already initialized + if (!mcpService) { + mcpService = new MCPService(); + } + sendStatus('MCP', 'MCP service initialized', 'success'); + + // Auto-start previously running servers based on startup settings + try { + const { autoStartMCP } = getStartupPreferences(); + + if (autoStartMCP) { + sendStatus('MCP', 'Restoring MCP servers...', 'info'); + const restoreResults = await mcpService.startPreviouslyRunningServers(); + const successCount = restoreResults.filter(r => r.success).length; + const totalCount = restoreResults.length; + + if (totalCount > 0) { + sendStatus('MCP', `Restored ${successCount}/${totalCount} MCP servers`, successCount === totalCount ? 'success' : 'warning'); + } else { + sendStatus('MCP', 'No MCP servers to restore', 'info'); + } + global.mcpServersRestored = true; // Mark as restored to prevent duplicate restoration + } else { + sendStatus('MCP', 'MCP auto-start disabled in settings', 'info'); + log.info('MCP server auto-start is disabled in startup settings'); + global.mcpServersRestored = true; // Mark as if restored to prevent later restoration attempts + } + } catch (restoreError) { + log.error('Error restoring MCP servers:', restoreError); + sendStatus('MCP', 'Failed to restore some MCP servers', 'warning'); + } + } catch (mcpError) { + log.error('Error initializing MCP service:', mcpError); + sendStatus('MCP', 'MCP service initialization failed', 'warning'); + } + + // Initialize MCP HTTP Proxy service for browser support + sendStatus('MCP Proxy', 'Starting MCP HTTP proxy...', 'info'); + try { + if (mcpService) { + const MCPProxyService = require('./mcpProxyService.cjs'); + const mcpProxyService = new MCPProxyService(mcpService); + await mcpProxyService.start(8092); + sendStatus('MCP Proxy', 'MCP HTTP proxy started on port 8092', 'success'); + log.info('✅ MCP HTTP proxy service started successfully'); + } else { + log.warn('⚠️ MCP service not available, skipping proxy start'); + sendStatus('MCP Proxy', 'Skipped (MCP service not available)', 'warning'); + } + } catch (proxyError) { + log.error('Error starting MCP HTTP proxy:', proxyError); + sendStatus('MCP Proxy', 'MCP proxy failed to start', 'warning'); + } + + // Initialize Watchdog service in background (with Docker support) + sendStatus('Watchdog', 'Initializing Watchdog service...', 'info'); + try { + watchdogService = new WatchdogService(dockerSetup, mcpService, ipcLogger); + + // Set up event listeners for watchdog events + watchdogService.on('serviceRestored', (serviceKey, service) => { + log.info(`Watchdog: ${service.name} has been restored`); + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('watchdog-service-restored', { serviceKey, service: service.name }); + } + }); + + watchdogService.on('serviceFailed', (serviceKey, service) => { + log.error(`Watchdog: ${service.name} has failed after maximum retry attempts`); + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('watchdog-service-failed', { serviceKey, service: service.name }); + } + }); + + watchdogService.on('serviceRestarted', (serviceKey, service) => { + log.info(`Watchdog: ${service.name} has been restarted successfully`); + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('watchdog-service-restarted', { serviceKey, service: service.name }); + } + }); + + // Start the watchdog monitoring + watchdogService.start(); + + // Signal watchdog service that Docker setup is complete + watchdogService.signalSetupComplete(); + + sendStatus('Watchdog', 'Watchdog service started successfully', 'success'); + } catch (watchdogError) { + log.error('Error initializing Watchdog service:', watchdogError); + sendStatus('Watchdog', 'Watchdog service initialization failed', 'warning'); + } + + // Notify that Docker mode initialization is complete + sendStatus('System', 'Docker mode initialization complete', 'success'); + log.info('Docker mode service initialization completed'); + + } catch (error) { + log.error('Error during Docker mode service initialization:', error); + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('background-service-error', { + service: 'System', + error: `Docker mode initialization error: ${error.message}` + }); + } + } +} + +async function createMainWindow() { + if (mainWindow) return; + + const startupPreferences = getStartupPreferences(); + let shouldStartFullscreen = startupPreferences.startFullscreen; + let shouldStartMinimized = startupPreferences.startMinimized; + + log.info(`Creating main window with fullscreen: ${shouldStartFullscreen}, minimized: ${shouldStartMinimized}`); + + const loginItemSettings = typeof app.getLoginItemSettings === 'function' + ? app.getLoginItemSettings() + : null; + const launchedAtLogin = !!(loginItemSettings && (loginItemSettings.wasOpenedAtLogin || loginItemSettings.wasOpenedAsHidden)); + + mainWindow = new BrowserWindow({ + fullscreen: shouldStartFullscreen, + fullscreenable: true, + width: shouldStartFullscreen ? undefined : 1200, + height: shouldStartFullscreen ? undefined : 800, + webPreferences: { + preload: path.join(__dirname, 'preload.cjs'), + contextIsolation: true, + nodeIntegration: false, + webviewTag: true, + sandbox: false, + webSecurity: false, // Required for screen sharing in Electron + experimentalFeatures: true // Enable experimental web features + }, + show: false, + backgroundColor: '#0f0f23', // Dark background to match loading screen + frame: true + }); + + if (typeof mainWindow.setFullScreenable === 'function') { + mainWindow.setFullScreenable(true); + } + + // Apply minimized state if needed + if (shouldStartMinimized) { + if (process.platform === 'darwin') { + mainWindow.hide(); + } else { + mainWindow.minimize(); + } + } + + // Handle window minimize to tray + mainWindow.on('minimize', (event) => { + if (process.platform !== 'darwin') { + // On Windows/Linux, minimize to tray + event.preventDefault(); + mainWindow.hide(); + + // Show balloon notification if tray is available + if (tray && process.platform === 'win32') { + try { + tray.displayBalloon({ + iconType: 'info', + title: 'ClaraVerse', + content: 'ClaraVerse is still running in the background. Click the tray icon to restore.' + }); + } catch (error) { + log.warn('Failed to show balloon notification:', error); + } + } + } + }); + + // Handle window close to tray + mainWindow.on('close', (event) => { + if (!isQuitting) { + event.preventDefault(); + mainWindow.hide(); + + // Show balloon notification if tray is available + if (tray && process.platform === 'win32') { + try { + tray.displayBalloon({ + iconType: 'info', + title: 'ClaraVerse', + content: 'ClaraVerse is still running in the background. Click the tray icon to restore.' + }); + } catch (error) { + log.warn('Failed to show balloon notification:', error); + } + } + } + }); + + // Create and set the application menu + createAppMenu(mainWindow); + + // Setup remote server IPC handlers with stopAllLocalServices callback + setupRemoteServerIPC(mainWindow, stopAllLocalServices); + log.info('Remote server IPC handlers registered'); + + // Set security policies for webview, using the dynamic n8n port + mainWindow.webContents.session.setPermissionRequestHandler((webContents, permission, callback) => { + const url = webContents.getURL(); + const n8nPort = dockerSetup?.ports?.n8n; // Get the determined n8n port + + try { + const parsedUrl = new URL(url); + const hostname = parsedUrl.hostname; + + // Allow ALL permissions for Clara app on localhost/127.0.0.1 + if ((hostname === 'localhost' || hostname === '127.0.0.1') && !url.includes('/n8n')) { + log.info(`Granted '${permission}' permission for Clara app URL: ${url}`); + callback(true); + return; + } + + // Allow permissions for n8n service + if (n8nPort && (hostname === 'localhost' || hostname === '127.0.0.1') && + url.startsWith(`http://${hostname}:${n8nPort}`)) { + log.info(`Granted '${permission}' permission for n8n URL: ${url}`); + callback(true); + return; + } + } catch (error) { + // Fallback for file:// URLs or parsing errors + if (url.startsWith('file://')) { + log.info(`Granted '${permission}' permission for file:// URL`); + callback(true); + return; + } + } + + log.warn(`Blocked permission request '${permission}' for URL: ${url}`); + callback(false); + }); + + // CRITICAL FIX: Register ALL event listeners BEFORE loading the URL + // This prevents race conditions where the event fires before the listener is attached + // Wait for DOM content to be fully loaded before showing + mainWindow.webContents.once('dom-ready', () => { + log.info('Main window DOM ready, showing immediately (fast startup mode)'); + + // Show window immediately for fast startup (unless user requested start minimized) + if (mainWindow && !mainWindow.isDestroyed()) { + if (!shouldStartMinimized) { + if (process.platform === 'darwin' && app.dock && typeof app.dock.show === 'function') { + app.dock.show(); + } + log.info('Showing main window (fast startup)'); + mainWindow.show(); + mainWindow.focus(); + + // CRITICAL FIX: Force webContents to regain focus for input elements + // This prevents the input box click issue where window is focused but inputs don't work + if (mainWindow.webContents && !mainWindow.webContents.isDestroyed()) { + mainWindow.webContents.focus(); + } + } else { + log.info('Skipping initial show because startMinimized is enabled'); + } + } + + // Initialize auto-updater when window is ready + setupAutoUpdater(mainWindow); + }); + + if (launchedAtLogin && !shouldStartMinimized) { + setTimeout(() => { + if (mainWindow && !mainWindow.isDestroyed() && !mainWindow.isVisible()) { + log.info('Forcing window visible after login launch'); + mainWindow.show(); + mainWindow.focus(); + + // CRITICAL FIX: Force webContents focus after login launch + if (mainWindow.webContents && !mainWindow.webContents.isDestroyed()) { + setTimeout(() => { + if (mainWindow && !mainWindow.isDestroyed() && mainWindow.webContents) { + mainWindow.webContents.focus(); + } + }, 100); + } + } + }, 750); + } + + // Fallback: Show window when ready (in case dom-ready doesn't fire) + mainWindow.once('ready-to-show', () => { + log.info('Main window ready-to-show event fired'); + // Only show if not already shown by dom-ready handler + if (mainWindow && !mainWindow.isVisible() && !shouldStartMinimized) { + setTimeout(() => { + if (mainWindow && !mainWindow.isDestroyed() && !mainWindow.isVisible() && !shouldStartMinimized) { + log.info('Fallback: Showing main window via ready-to-show'); + mainWindow.show(); + + if (loadingScreen) { + loadingScreen.close(); + loadingScreen = null; + } + } + }, 3000); + } + }); + + // ADDITIONAL FIX: Handle window focus events to ensure webContents stays in sync + mainWindow.on('focus', () => { + if (mainWindow && !mainWindow.isDestroyed() && mainWindow.webContents && !mainWindow.webContents.isDestroyed()) { + // Force webContents to focus when window gains focus + // This solves input box click issues after minimize/restore or DevTools toggle + setTimeout(() => { + if (mainWindow && !mainWindow.isDestroyed() && mainWindow.webContents) { + mainWindow.webContents.focus(); + } + }, 100); + } + }); + + // ADDITIONAL FIX: Handle window restore (from minimize) to ensure inputs work + mainWindow.on('restore', () => { + if (mainWindow && !mainWindow.isDestroyed() && mainWindow.webContents && !mainWindow.webContents.isDestroyed()) { + setTimeout(() => { + if (mainWindow && !mainWindow.isDestroyed() && mainWindow.webContents) { + mainWindow.webContents.focus(); + } + }, 100); + } + }); + + mainWindow.on('closed', () => { + mainWindow = null; + }); + + // NOW load the URL after all event listeners are registered + // Development mode with hot reload + if (process.env.NODE_ENV === 'development') { + if (process.env.ELECTRON_HOT_RELOAD === 'true') { + // Hot reload mode + const devServerUrl = process.env.ELECTRON_START_URL || 'http://localhost:5173'; + + log.info('Loading development server with hot reload:', devServerUrl); + mainWindow.loadURL(devServerUrl).catch(err => { + log.error('Failed to load dev server:', err); + // Fallback to local file if dev server fails + mainWindow.loadFile(path.join(__dirname, '../dist/index.html')); + }); + + // Enable hot reload by watching the renderer process + mainWindow.webContents.on('did-fail-load', () => { + log.warn('Page failed to load, retrying...'); + setTimeout(() => { + mainWindow?.webContents.reload(); + }, 1000); + }); + } else { + // Development mode without hot reload - use built files + log.info('Loading development build from dist directory'); + mainWindow.loadFile(path.join(__dirname, '../dist/index.html')); + } + + // Open DevTools in both development modes + mainWindow.webContents.openDevTools(); + } else { + // Production mode - serve via loopback server so COOP/COEP headers are present + try { + const port = await ensureStaticServer(); + const prodUrl = `http://${STATIC_SERVER_HOST}:${port}/index.html`; + log.info(`Loading production build from ${prodUrl}`); + await mainWindow.loadURL(prodUrl); + } catch (error) { + log.error('Falling back to file:// load after static server failure:', error); + await mainWindow.loadFile(path.join(__dirname, '../dist/index.html')); + } + } +} + +// Initialize app when ready +app.whenReady().then(async () => { + // Request single instance lock to prevent multiple instances + // This prevents port conflicts and data corruption + const gotTheLock = app.requestSingleInstanceLock(); + + if (!gotTheLock) { + log.warn('Another instance is already running. Quitting this instance.'); + app.quit(); + return; + } + + // If someone tries to run a second instance, focus our window + app.on('second-instance', () => { + if (mainWindow) { + if (mainWindow.isMinimized()) mainWindow.restore(); + mainWindow.focus(); + + // CRITICAL FIX: Force webContents focus on second instance + if (mainWindow.webContents && !mainWindow.webContents.isDestroyed()) { + setTimeout(() => { + if (mainWindow && !mainWindow.isDestroyed() && mainWindow.webContents) { + mainWindow.webContents.focus(); + } + }, 100); + } + } + }); + + // Initialize isolated startup settings manager first + ensureStartupSettingsManager(); + log.info('🔒 Isolated startup settings manager initialized on app ready'); + + // SECURITY: Clean up any stored passwords from previous versions + await cleanupStoredPasswords(); + + await initialize(); + + // Create system tray + createTray(); + + // Register global shortcuts after app is ready + registerGlobalShortcuts(); + + log.info('Application initialization complete with isolated startup settings and global shortcuts registered'); +}); + +// Quit when all windows are closed +app.on('window-all-closed', async () => { + // If the app is quitting intentionally, proceed with cleanup + if (isQuitting) { + try { + // Clean up tray + if (tray && !tray.isDestroyed()) { + tray.destroy(); + tray = null; + } + + // Unregister global shortcuts when app is quitting + globalShortcut.unregisterAll(); + } catch (error) { + log.error('Error during window-all-closed cleanup:', error); + } + + if (staticServer) { + try { + staticServer.close(); + log.info('Loopback static server stopped'); + } catch (error) { + log.warn('Error shutting down loopback static server:', error); + } finally { + staticServer = null; + staticServerPort = null; + } + } + + // Stop watchdog service first + if (watchdogService) { + try { + log.info('Stopping watchdog service...'); + watchdogService.stop(); + } catch (error) { + log.error('Error stopping watchdog service:', error); + } + } + + // Stop scheduler service + if (schedulerService) { + try { + log.info('Stopping scheduler service...'); + await schedulerService.cleanup(); + } catch (error) { + log.error('Error stopping scheduler service:', error); + } + } + + // Stop widget service + if (widgetService) { + try { + log.info('Stopping widget service...'); + await widgetService.cleanup(); + } catch (error) { + log.error('Error stopping widget service:', error); + } + } + + // Save MCP server running state before stopping + if (mcpService) { + try { + log.info('Saving MCP server running state...'); + mcpService.saveRunningState(); + } catch (error) { + log.error('Error saving MCP server running state:', error); + } + } + + + // Stop all MCP servers + if (mcpService) { + try { + log.info('Stopping all MCP servers...'); + await mcpService.stopAllServers(); + } catch (error) { + log.error('Error stopping MCP servers:', error); + } + } + + // Stop ClaraCore service + if (centralServiceManager && centralServiceManager.services.has('claracore')) { + try { + log.info('Stopping ClaraCore service...'); + await centralServiceManager.stopService('claracore'); + log.info('✅ ClaraCore service stopped successfully'); + } catch (error) { + log.error('❌ Error stopping ClaraCore service:', error); + } + } + + // Stop Docker containers + if (dockerSetup) { + await dockerSetup.stop(); + } + + // Note: On macOS, we don't call app.quit() here because: + // 1. If quit was initiated from the exit button, app.quit() was already called in app-close handler + // 2. If quit was initiated from menu (Cmd+Q), app.quit() was already called + // 3. This prevents double-quit issues + // On Windows/Linux, we need to explicitly quit because the behavior is different + if (process.platform !== 'darwin') { + app.quit(); + } + } else { + // If not quitting intentionally, keep the app running in the tray + // On macOS, it's common to keep the app running when all windows are closed + if (process.platform === 'darwin') { + // Do nothing - keep app running + } else { + // On Windows/Linux, show a notification that the app is running in the tray + log.info('App minimized to system tray'); + } + } +}); + +// Handle app quit - ensure services are stopped +app.on('before-quit', async (event) => { + if (!isQuitting) { + log.info('App is quitting, setting isQuitting flag...'); + isQuitting = true; + } +}); + +// Additional cleanup on will-quit +let willQuitHandled = false; // Flag to prevent infinite loop +app.on('will-quit', async (event) => { + log.info('App will quit, ensuring all services are stopped...'); + + // Stop ClaraCore service explicitly (only once) + if (!willQuitHandled && centralServiceManager && centralServiceManager.services.has('claracore')) { + willQuitHandled = true; // Set flag to prevent re-entry + + try { + event.preventDefault(); // Prevent quit until cleanup is done + log.info('Stopping ClaraCore service on quit...'); + await centralServiceManager.stopService('claracore'); + log.info('✅ ClaraCore service stopped successfully'); + + // Now actually quit + setTimeout(() => app.quit(), 500); + } catch (error) { + log.error('❌ Error stopping ClaraCore service on quit:', error); + // Still quit even if cleanup fails + setTimeout(() => app.quit(), 500); + } + } +}); + +app.on('activate', async () => { + if (!mainWindow || mainWindow.isDestroyed()) { + await createMainWindow(); + return; + } + + if (mainWindow.isMinimized()) { + mainWindow.restore(); + } + + if (!mainWindow.isVisible()) { + mainWindow.show(); + } + + mainWindow.focus(); + + // CRITICAL FIX: Force webContents focus on app activate (macOS dock click) + if (mainWindow.webContents && !mainWindow.webContents.isDestroyed()) { + setTimeout(() => { + if (mainWindow && !mainWindow.isDestroyed() && mainWindow.webContents) { + mainWindow.webContents.focus(); + } + }, 100); + } +}); + +// Register startup settings handler +ipcMain.handle('set-startup-settings', async (event, settings) => { + // DEPRECATED: Redirect to isolated startup settings system + log.warn('⚠️ DEPRECATED: set-startup-settings called. Use startup-settings:update instead.'); + + try { + const manager = ensureStartupSettingsManager(); + + // Update using isolated system with implicit consent for backward compatibility + const result = await manager.updateSettings(settings); + + const shouldUpdateLoginItem = + settings.autoStart !== undefined || settings.startMinimized !== undefined; + if (shouldUpdateLoginItem && (process.platform === 'darwin' || process.platform === 'win32')) { + const openAtLogin = !result.isDevelopment && !!result.autoStart; + const openAsHidden = !!result.startMinimized; + + if (!openAtLogin) { + app.setLoginItemSettings({ + openAtLogin: false, + openAsHidden: false + }); + } else { + app.setLoginItemSettings({ + openAtLogin, + openAsHidden, + path: process.execPath, + args: [] + }); + } + } + + log.info('🔒 Startup settings updated via deprecated handler:', settings); + return { success: true }; + } catch (error) { + log.error('Error updating startup settings:', error); + return { success: false, error: error.message }; + } +}); + +ipcMain.handle('get-startup-settings', async () => { + // DEPRECATED: Redirect to isolated startup settings system + log.warn('⚠️ DEPRECATED: get-startup-settings called. Use startup-settings:get instead.'); + + try { + const settings = ensureStartupSettingsManager().readSettings(); + log.info('🔒 Startup settings retrieved via deprecated handler'); + + return settings; + } catch (error) { + log.error('Error reading isolated startup settings:', error); + return { + isDevelopment: process.env.NODE_ENV === 'development' || !app.isPackaged, + startFullscreen: false, + startMinimized: false, + autoStart: false, + checkUpdates: true, + restoreLastSession: true, + autoStartMCP: true + }; + } +}); + +// Register feature configuration IPC handlers +ipcMain.handle('get-feature-config', async () => { + try { + return FeatureSelectionScreen.getCurrentConfig(); + } catch (error) { + log.error('Error getting feature configuration:', error); + return null; + } +}); + +ipcMain.handle('update-feature-config', async (event, newConfig) => { + try { + const FeatureSelectionScreen = require('./featureSelection.cjs'); + const featureSelection = new FeatureSelectionScreen(); + + // Load current config + const currentConfig = featureSelection.loadConfig(); + + // Check if this is completing first-time setup + const wasFirstTime = currentConfig.firstTimeSetup === true; + + // Update with new selections + const updatedConfig = { + ...currentConfig, + selectedFeatures: { + claraCore: true, // Always enabled + ...newConfig + }, + firstTimeSetup: false, // Mark onboarding as complete + setupTimestamp: new Date().toISOString() + }; + + // Save the updated configuration + const success = featureSelection.saveConfig(updatedConfig); + + if (success) { + // Update global selected features + global.selectedFeatures = updatedConfig.selectedFeatures; + global.needsFeatureSelection = false; // User has completed onboarding + log.info('✅ Feature configuration updated:', updatedConfig.selectedFeatures); + + // If this was first-time setup completion, initialize services with user consent + if (wasFirstTime && newConfig.userConsentGiven) { + log.info('🎉 User completed onboarding - initializing selected services with consent'); + + // Send status update to UI + const sendStatusUpdate = (status, details = {}) => { + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('service-status-update', { status, ...details }); + } + }; + + // Initialize services with user's selections + if (dockerSetup) { + const isDockerAvailable = await dockerSetup.isDockerRunning(); + + if (isDockerAvailable) { + sendStatusUpdate('docker-available', { message: 'Starting selected services with Docker...' }); + await initializeServicesWithDocker(updatedConfig.selectedFeatures, sendStatusUpdate); + } else { + sendStatusUpdate('docker-not-available', { message: 'Starting selected services in lightweight mode...' }); + await initializeServicesWithoutDocker(updatedConfig.selectedFeatures, sendStatusUpdate); + } + + sendStatusUpdate('ready', { message: 'All selected services initialized' }); + } + } + } + + return success; + } catch (error) { + log.error('Error updating feature configuration:', error); + return false; + } +}); + +ipcMain.handle('reset-feature-config', async () => { + try { + const configPath = path.join(app.getPath('userData'), 'clara-features.yaml'); + if (fs.existsSync(configPath)) { + fs.unlinkSync(configPath); + log.info('Feature configuration reset successfully'); + return true; + } + return false; + } catch (error) { + log.error('Error resetting feature configuration:', error); + return false; + } +}); + +// Initialize electron-store for persistent configuration (using dynamic import for ES module) +let store = null; +let storeInitPromise = null; + +// Lazy initialize store when needed +async function getStore() { + if (store) return store; + if (storeInitPromise) return storeInitPromise; + + storeInitPromise = (async () => { + try { + const StoreModule = await import('electron-store'); + const Store = StoreModule.default; + store = new Store(); + log.info('📦 [Store] Initialized successfully'); + return store; + } catch (error) { + log.error('📦 [Store] Failed to initialize:', error); + throw error; + } + })(); + + return storeInitPromise; +} + +/** + * SECURITY: Clean up any passwords that may have been stored in previous versions + * This runs on app startup to ensure no passwords are persisted + */ +async function cleanupStoredPasswords() { + try { + const store = await getStore(); + + // Check and clean remoteServer config + const remoteServer = store.get('remoteServer'); + if (remoteServer && remoteServer.password) { + log.warn('🔒 [Security] Found password in remoteServer config - removing it'); + delete remoteServer.password; + store.set('remoteServer', remoteServer); + log.info('✅ [Security] Password removed from remoteServer config'); + } + + // Check and clean claraCoreRemote config (shouldn't have password but check anyway) + const claraCoreRemote = store.get('claraCoreRemote'); + if (claraCoreRemote && claraCoreRemote.password) { + log.warn('🔒 [Security] Found password in claraCoreRemote config - removing it'); + delete claraCoreRemote.password; + store.set('claraCoreRemote', claraCoreRemote); + log.info('✅ [Security] Password removed from claraCoreRemote config'); + } + + log.info('✅ [Security] Password cleanup complete'); + } catch (error) { + log.error('❌ [Security] Failed to cleanup stored passwords:', error); + } +} + +// electron-store IPC handlers for configuration persistence +ipcMain.handle('store:get', async (event, key) => { + try { + const store = await getStore(); + const value = store.get(key); + log.info(`📦 [Store] GET ${key}:`, value); + return value; + } catch (error) { + log.error(`Error getting store key ${key}:`, error); + return null; + } +}); + +ipcMain.handle('store:set', async (event, key, value) => { + try { + const store = await getStore(); + store.set(key, value); + log.info(`📦 [Store] SET ${key}:`, value); + return true; + } catch (error) { + log.error(`Error setting store key ${key}:`, error); + return false; + } +}); + +ipcMain.handle('store:delete', async (event, key) => { + try { + const store = await getStore(); + store.delete(key); + log.info(`📦 [Store] DELETE ${key}`); + return true; + } catch (error) { + log.error(`Error deleting store key ${key}:`, error); + return false; + } +}); + +ipcMain.handle('store:has', async (event, key) => { + try { + const store = await getStore(); + return store.has(key); + } catch (error) { + log.error(`Error checking store key ${key}:`, error); + return false; + } +}); + +ipcMain.handle('store:clear', async () => { + try { + const store = await getStore(); + store.clear(); + log.info('📦 [Store] CLEARED'); + return true; + } catch (error) { + log.error('Error clearing store:', error); + return false; + } +}); + +// Model Manager IPC handlers +ipcMain.handle('model-manager:search-civitai', async (event, { query, types, sort, apiKey, nsfw = false }) => { + try { + // Enhanced search with multiple strategies for better results + const searches = []; + + // Strategy 1: Exact query search + const exactUrl = new URL('https://civitai.com/api/v1/models'); + exactUrl.searchParams.set('limit', '50'); // Increased limit for better results + exactUrl.searchParams.set('query', query); + exactUrl.searchParams.set('sort', sort || 'Highest Rated'); + if (types && types.length > 0) { + exactUrl.searchParams.set('types', types.join(',')); + } + if (nsfw) { + exactUrl.searchParams.set('nsfw', 'true'); + } + + // Strategy 2: Tag-based search (if query looks like it could be tags) + const tagUrl = new URL('https://civitai.com/api/v1/models'); + tagUrl.searchParams.set('limit', '30'); + tagUrl.searchParams.set('tag', query); + tagUrl.searchParams.set('sort', sort || 'Highest Rated'); + if (types && types.length > 0) { + tagUrl.searchParams.set('types', types.join(',')); + } + if (nsfw) { + tagUrl.searchParams.set('nsfw', 'true'); + } + + // Strategy 3: Username search (if query looks like a username) + let usernameUrl = null; + if (query && !query.includes(' ') && query.length > 2) { + usernameUrl = new URL('https://civitai.com/api/v1/models'); + usernameUrl.searchParams.set('limit', '20'); + usernameUrl.searchParams.set('username', query); + usernameUrl.searchParams.set('sort', sort || 'Highest Rated'); + if (types && types.length > 0) { + usernameUrl.searchParams.set('types', types.join(',')); + } + if (nsfw) { + usernameUrl.searchParams.set('nsfw', 'true'); + } + } + + // Add API key for authenticated requests if available + const headers = { + 'User-Agent': 'Clara-AI-Assistant/1.0', + 'Content-Type': 'application/json' + }; + if (apiKey) { + headers['Authorization'] = `Bearer ${apiKey}`; + } + + // Execute searches in parallel + const searchPromises = [ + fetch(exactUrl.toString(), { headers }).then(r => r.json()), + fetch(tagUrl.toString(), { headers }).then(r => r.json()).catch(() => ({ items: [] })) + ]; + + if (usernameUrl) { + searchPromises.push( + fetch(usernameUrl.toString(), { headers }).then(r => r.json()).catch(() => ({ items: [] })) + ); + } + + const results = await Promise.all(searchPromises); + + // Combine and deduplicate results + const allItems = []; + const seenIds = new Set(); + + results.forEach((result, index) => { + if (result.items) { + result.items.forEach(item => { + if (!seenIds.has(item.id)) { + seenIds.add(item.id); + // Add relevance score based on search strategy + item._relevanceScore = index === 0 ? 10 : (index === 1 ? 7 : 5); + allItems.push(item); + } + }); + } + }); + + // Enhanced sorting with relevance and popularity + allItems.sort((a, b) => { + // Primary sort by relevance score + if (a._relevanceScore !== b._relevanceScore) { + return b._relevanceScore - a._relevanceScore; + } + + // Secondary sort by the requested sort order + if (sort === 'Most Downloaded') { + return (b.stats?.downloadCount || 0) - (a.stats?.downloadCount || 0); + } else if (sort === 'Newest') { + return new Date(b.createdAt || 0) - new Date(a.createdAt || 0); + } else { // Highest Rated + return (b.stats?.rating || 0) - (a.stats?.rating || 0); + } + }); + + return { + items: allItems.slice(0, 60), // Return top 60 results + metadata: { + totalItems: allItems.length, + searchStrategies: results.length, + hasApiKey: !!apiKey + } + }; + } catch (error) { + log.error('Error searching CivitAI models:', error); + throw error; + } +}); + +ipcMain.handle('model-manager:search-huggingface', async (event, { query, modelType, author }) => { + try { + const url = new URL('https://huggingface.co/api/models'); + url.searchParams.set('limit', '20'); + url.searchParams.set('search', query); + if (modelType) { + url.searchParams.set('filter', `library:${modelType}`); + } + if (author) { + url.searchParams.set('author', author); + } + + const response = await fetch(url.toString()); + const data = await response.json(); + return data; + } catch (error) { + log.error('Error searching Hugging Face models:', error); + throw error; + } +}); + +ipcMain.handle('model-manager:download-model', async (event, { url, filename, modelType, source }) => { + return new Promise((resolve, reject) => { + try { + const modelsDir = path.join(app.getPath('userData'), 'models', modelType); + if (!fs.existsSync(modelsDir)) { + fs.mkdirSync(modelsDir, { recursive: true }); + } + + const filePath = path.join(modelsDir, filename); + const file = fs.createWriteStream(filePath); + const client = url.startsWith('https:') ? https : http; + + // Add headers for different sources + const headers = {}; + if (source === 'huggingface') { + // For Hugging Face, we might need auth headers + headers['User-Agent'] = 'Clara-AI-Assistant/1.0'; + } + + const request = client.get(url, { headers }, (response) => { + if (response.statusCode !== 200) { + reject(new Error(`Failed to download: ${response.statusCode}`)); + return; + } + + const totalSize = parseInt(response.headers['content-length']); + let downloadedSize = 0; + + response.on('data', (chunk) => { + downloadedSize += chunk.length; + const progress = (downloadedSize / totalSize) * 100; + event.sender.send('model-download-progress', { + filename, + progress: Math.round(progress), + downloadedSize, + totalSize + }); + }); + + response.pipe(file); + + file.on('finish', () => { + file.close(() => { + log.info(`Model downloaded successfully: ${filename}`); + resolve({ success: true, path: filePath }); + }); + }); + }); + + request.on('error', (error) => { + fs.unlink(filePath, () => {}); // Delete partial file + reject(error); + }); + } catch (error) { + log.error('Error downloading model:', error); + reject(error); + } + }); +}); + +ipcMain.handle('model-manager:get-local-models', async () => { + try { + const modelsDir = path.join(app.getPath('userData'), 'models'); + if (!fs.existsSync(modelsDir)) { + return {}; + } + + const models = {}; + const modelTypes = fs.readdirSync(modelsDir, { withFileTypes: true }) + .filter(dirent => dirent.isDirectory()) + .map(dirent => dirent.name); + + for (const type of modelTypes) { + const typePath = path.join(modelsDir, type); + const files = fs.readdirSync(typePath).map(filename => { + const filePath = path.join(typePath, filename); + const stats = fs.statSync(filePath); + return { + name: filename, + size: stats.size, + modified: stats.mtime, + path: filePath + }; + }); + models[type] = files; + } + + return models; + } catch (error) { + log.error('Error getting local models:', error); + return {}; + } +}); + +ipcMain.handle('model-manager:delete-local-model', async (event, { modelType, filename }) => { + try { + const filePath = path.join(app.getPath('userData'), 'models', modelType, filename); + if (fs.existsSync(filePath)) { + fs.unlinkSync(filePath); + log.info(`Model deleted: ${filename}`); + return { success: true }; + } + return { success: false, error: 'File not found' }; + } catch (error) { + log.error('Error deleting model:', error); + throw error; + } +}); + +ipcMain.handle('model-manager:save-api-keys', async (event, keys) => { + try { + const settingsPath = path.join(app.getPath('userData'), 'model-manager-settings.json'); + fs.writeFileSync(settingsPath, JSON.stringify(keys, null, 2)); + log.info('API keys saved successfully'); + return { success: true }; + } catch (error) { + log.error('Error saving API keys:', error); + throw error; + } +}); + +ipcMain.handle('model-manager:get-api-keys', async () => { + try { + const settingsPath = path.join(app.getPath('userData'), 'model-manager-settings.json'); + if (fs.existsSync(settingsPath)) { + const keys = JSON.parse(fs.readFileSync(settingsPath, 'utf8')); + return keys; + } + return {}; + } catch (error) { + log.error('Error reading API keys:', error); + return {}; + } +}); + +// ComfyUI Model Download Handler - downloads directly to ComfyUI model directories +ipcMain.handle('comfyui-model-manager:download-model', async (event, { url, filename, modelType, source, apiKey }) => { + return new Promise((resolve, reject) => { + try { + // Map model types to ComfyUI directory structure + const modelTypeMapping = { + 'checkpoint': 'checkpoints', + 'lora': 'loras', + 'vae': 'vae', + 'controlnet': 'controlnet', + 'upscaler': 'upscale_models', + 'embedding': 'embeddings', + 'textualinversion': 'embeddings', // CivitAI uses this term + 'hypernetwork': 'hypernetworks', + 'style': 'style_models', + 't2i_adapter': 't2i_adapter', + 'clip': 'clip', + 'unet': 'unet' + }; + + const comfyuiDir = modelTypeMapping[modelType] || 'checkpoints'; + + // Get the ComfyUI models directory - prefer WSL2 path if available + let comfyuiModelsDir; + + try { + // Try to use WSL2 path for better performance + const os = require('os'); + if (os.platform() === 'win32') { + const { execSync } = require('child_process'); + const wslList = execSync('wsl -l -v', { encoding: 'utf8' }); + const distributions = wslList.split('\n') + .filter(line => line.includes('Running')) + .map(line => line.trim().split(/\s+/)[0]) + .filter(dist => dist && dist !== 'NAME'); + + if (distributions.length > 0) { + const distro = distributions[0]; + let wslUser = 'root'; + try { + wslUser = execSync(`wsl -d ${distro} whoami`, { encoding: 'utf8' }).trim(); + } catch (error) { + // Use root as fallback + } + + // Use WSL2 path + comfyuiModelsDir = `\\\\wsl.localhost\\${distro}\\home\\${wslUser}\\comfyui_models\\${comfyuiDir}`; + log.info(`Using WSL2 path for model download: ${comfyuiModelsDir}`); + } else { + throw new Error('No running WSL2 distributions found'); + } + } else { + throw new Error('Not on Windows'); + } + } catch (error) { + // Fallback to Windows path + comfyuiModelsDir = path.join(app.getPath('userData'), 'comfyui_models', comfyuiDir); + log.info(`Using Windows path for model download: ${comfyuiModelsDir}`); + } + + // Ensure directory exists + if (!fs.existsSync(comfyuiModelsDir)) { + fs.mkdirSync(comfyuiModelsDir, { recursive: true }); + log.info(`Created ComfyUI model directory: ${comfyuiModelsDir}`); + } + + const filePath = path.join(comfyuiModelsDir, filename); + + // Check if file already exists + if (fs.existsSync(filePath)) { + log.warn(`File already exists: ${filename}`); + resolve({ success: false, error: 'File already exists', path: filePath }); + return; + } + + const file = fs.createWriteStream(filePath); + const client = url.startsWith('https:') ? https : http; + + // Add headers for different sources + const headers = { + 'User-Agent': 'Clara-AI-Assistant/1.0', + 'Accept': '*/*', + 'Accept-Encoding': 'gzip, deflate, br', + 'Connection': 'keep-alive' + }; + + if (source === 'civitai') { + // CivitAI specific headers + headers['Referer'] = 'https://civitai.com/'; + if (apiKey) { + headers['Authorization'] = `Bearer ${apiKey}`; + } + } else if (source === 'huggingface' && apiKey) { + headers['Authorization'] = `Bearer ${apiKey}`; + } + + log.info(`Starting download: ${filename} to ${comfyuiDir} directory`); + log.info(`Download URL: ${url}`); + log.info(`File path: ${filePath}`); + log.info(`Source: ${source}, API Key provided: ${!!apiKey}`); + + // Validate URL + try { + new URL(url); + } catch (urlError) { + reject(new Error(`Invalid download URL: ${url}`)); + return; + } + + const makeRequest = (requestUrl, redirectCount = 0) => { + if (redirectCount > 5) { + file.close(); + fs.unlink(filePath, () => {}); + reject(new Error('Too many redirects')); + return; + } + + const requestClient = requestUrl.startsWith('https:') ? https : http; + const request = requestClient.get(requestUrl, { headers }, (response) => { + // Handle all redirect status codes + if ([301, 302, 303, 307, 308].includes(response.statusCode)) { + const redirectUrl = response.headers.location; + if (!redirectUrl) { + file.close(); + fs.unlink(filePath, () => {}); + reject(new Error(`Redirect response missing location header`)); + return; + } + + log.info(`Following ${response.statusCode} redirect to: ${redirectUrl}`); + + // Handle relative URLs + let fullRedirectUrl = redirectUrl; + if (redirectUrl.startsWith('/')) { + const urlObj = new URL(requestUrl); + fullRedirectUrl = `${urlObj.protocol}//${urlObj.host}${redirectUrl}`; + } else if (!redirectUrl.startsWith('http')) { + const urlObj = new URL(requestUrl); + fullRedirectUrl = `${urlObj.protocol}//${urlObj.host}/${redirectUrl}`; + } + + // Make new request to redirect URL + makeRequest(fullRedirectUrl, redirectCount + 1); + return; + } + + if (response.statusCode !== 200) { + file.close(); + fs.unlink(filePath, () => {}); + log.error(`Download failed with status ${response.statusCode}: ${response.statusMessage}`); + log.error(`Response headers:`, response.headers); + reject(new Error(`Failed to download: ${response.statusCode} ${response.statusMessage}`)); + return; + } + + log.info(`Download started successfully for ${filename}`); + log.info(`Content-Length: ${response.headers['content-length'] || 'Unknown'}`); + + handleDownloadResponse(response); + }); + + request.on('error', (error) => { + file.close(); + fs.unlink(filePath, () => {}); + reject(new Error(`Request failed: ${error.message}`)); + }); + + request.setTimeout(30000, () => { + request.destroy(); + file.close(); + fs.unlink(filePath, () => {}); + reject(new Error('Download timeout')); + }); + }; + + makeRequest(url); + + function handleDownloadResponse(response) { + const totalSize = parseInt(response.headers['content-length']) || 0; + let downloadedSize = 0; + const startTime = Date.now(); + + response.on('data', (chunk) => { + downloadedSize += chunk.length; + const progress = totalSize > 0 ? (downloadedSize / totalSize) * 100 : 0; + const elapsed = (Date.now() - startTime) / 1000; + const speed = downloadedSize / elapsed; + const remaining = totalSize > 0 ? (totalSize - downloadedSize) / speed : 0; + + // Send progress update + event.sender.send('comfyui-model-download-progress', { + filename, + progress: Math.round(progress), + downloadedSize, + totalSize, + speed: formatBytes(speed) + '/s', + eta: remaining > 0 ? `${Math.round(remaining)}s` : 'Unknown' + }); + }); + + response.pipe(file); + + file.on('finish', () => { + file.close(() => { + log.info(`ComfyUI model downloaded successfully: ${filename} to ${comfyuiDir}`); + + // Send completion event + event.sender.send('comfyui-model-download-complete', { + filename, + modelType: comfyuiDir, + path: filePath, + size: fs.statSync(filePath).size + }); + + resolve({ + success: true, + path: filePath, + modelType: comfyuiDir, + size: fs.statSync(filePath).size + }); + }); + }); + + file.on('error', (error) => { + fs.unlink(filePath, () => {}); + reject(new Error(`File write error: ${error.message}`)); + }); + } + + } catch (error) { + log.error('Error downloading ComfyUI model:', error); + reject(error); + } + }); +}); + +// Get ComfyUI models organized by type +ipcMain.handle('comfyui-model-manager:get-local-models', async () => { + try { + const comfyuiModelsDir = path.join(app.getPath('userData'), 'comfyui_models'); + if (!fs.existsSync(comfyuiModelsDir)) { + return {}; + } + + const models = {}; + const modelDirs = [ + 'checkpoints', 'loras', 'vae', 'controlnet', 'upscale_models', + 'embeddings', 'hypernetworks', 'style_models', 't2i_adapter', 'clip', 'unet' + ]; + + for (const dir of modelDirs) { + const dirPath = path.join(comfyuiModelsDir, dir); + if (fs.existsSync(dirPath)) { + try { + const files = fs.readdirSync(dirPath) + .filter(file => { + // Filter for model files (safetensors, ckpt, pt, pth, bin) + const ext = path.extname(file).toLowerCase(); + return ['.safetensors', '.ckpt', '.pt', '.pth', '.bin'].includes(ext); + }) + .map(filename => { + const filePath = path.join(dirPath, filename); + const stats = fs.statSync(filePath); + return { + name: filename, + size: stats.size, + modified: stats.mtime, + path: filePath, + type: dir + }; + }); + models[dir] = files; + } catch (error) { + log.warn(`Error reading directory ${dir}:`, error); + models[dir] = []; + } + } else { + models[dir] = []; + } + } + + return models; + } catch (error) { + log.error('Error getting ComfyUI local models:', error); + return {}; + } +}); + +// Delete ComfyUI model +ipcMain.handle('comfyui-model-manager:delete-model', async (event, { modelType, filename }) => { + try { + const filePath = path.join(app.getPath('userData'), 'comfyui_models', modelType, filename); + if (fs.existsSync(filePath)) { + fs.unlinkSync(filePath); + log.info(`ComfyUI model deleted: ${filename} from ${modelType}`); + return { success: true }; + } + return { success: false, error: 'File not found' }; + } catch (error) { + log.error('Error deleting ComfyUI model:', error); + throw error; + } +}); + +// Get ComfyUI models directory info +ipcMain.handle('comfyui-model-manager:get-models-dir', async () => { + try { + const comfyuiModelsDir = path.join(app.getPath('userData'), 'comfyui_models'); + return { + path: comfyuiModelsDir, + exists: fs.existsSync(comfyuiModelsDir) + }; + } catch (error) { + log.error('Error getting ComfyUI models directory:', error); + return { path: '', exists: false }; + } +}); + +// ============================================== +// ComfyUI Output Images Management +// ============================================== + +// List ComfyUI output images +ipcMain.handle('comfyui:list-output-images', async () => { + try { + const os = require('os'); + const outputDir = path.join(os.homedir(), '.clara', 'comfyui-data', 'outputs'); + + if (!fs.existsSync(outputDir)) { + log.info('ComfyUI outputs directory does not exist:', outputDir); + return []; + } + + const files = fs.readdirSync(outputDir) + .filter(file => /\.(png|jpg|jpeg|webp|gif)$/i.test(file)) + .map(filename => { + const filePath = path.join(outputDir, filename); + const stats = fs.statSync(filePath); + + // Try to extract prompt from filename if it follows ComfyUI pattern + let prompt = 'ComfyUI Generated Image'; + const promptMatch = filename.match(/^(.+?)_\d+_?\d*\.(png|jpg|jpeg|webp|gif)$/i); + if (promptMatch) { + prompt = promptMatch[1].replace(/_/g, ' ').trim(); + } + + return { + id: `comfyui-${filename}`, + name: filename, + path: filePath, + size: stats.size, + modified: stats.mtime.toISOString(), + created: stats.birthtime.toISOString(), + prompt: prompt, + source: 'comfyui', + url: `file://${filePath}`, + // Convert to base64 for web display + dataUrl: null // Will be populated on demand + }; + }) + .sort((a, b) => new Date(b.modified).getTime() - new Date(a.modified).getTime()); // newest first + + log.info(`Found ${files.length} ComfyUI output images`); + return files; + } catch (error) { + log.error('Error listing ComfyUI output images:', error); + return []; + } +}); + +// Get ComfyUI image as base64 data URL +ipcMain.handle('comfyui:get-image-data', async (event, imagePath) => { + try { + if (!fs.existsSync(imagePath)) { + throw new Error('Image file not found'); + } + + const imageBuffer = fs.readFileSync(imagePath); + const ext = path.extname(imagePath).toLowerCase().slice(1); + const mimeType = ext === 'jpg' ? 'jpeg' : ext; + const base64 = imageBuffer.toString('base64'); + const dataUrl = `data:image/${mimeType};base64,${base64}`; + + return dataUrl; + } catch (error) { + log.error('Error reading ComfyUI image:', error); + throw error; + } +}); + +// Watch ComfyUI outputs directory for changes +let comfyuiOutputWatcher = null; + +ipcMain.handle('comfyui:start-output-watcher', async (event) => { + try { + // Stop existing watcher if running + if (comfyuiOutputWatcher) { + comfyuiOutputWatcher.close(); + } + + const os = require('os'); + const outputDir = path.join(os.homedir(), '.clara', 'comfyui-data', 'outputs'); + + if (!fs.existsSync(outputDir)) { + fs.mkdirSync(outputDir, { recursive: true }); + } + + const chokidar = require('chokidar'); + comfyuiOutputWatcher = chokidar.watch(outputDir, { + ignored: /^\./, + persistent: true, + ignoreInitial: true + }); + + comfyuiOutputWatcher.on('add', (filePath) => { + if (/\.(png|jpg|jpeg|webp|gif)$/i.test(filePath)) { + const filename = path.basename(filePath); + const stats = fs.statSync(filePath); + + // Extract prompt from filename + let prompt = 'ComfyUI Generated Image'; + const promptMatch = filename.match(/^(.+?)_\d+_?\d*\.(png|jpg|jpeg|webp|gif)$/i); + if (promptMatch) { + prompt = promptMatch[1].replace(/_/g, ' ').trim(); + } + + const imageInfo = { + id: `comfyui-${filename}`, + name: filename, + path: filePath, + size: stats.size, + modified: stats.mtime.toISOString(), + created: stats.birthtime.toISOString(), + prompt: prompt, + source: 'comfyui' + }; + + event.sender.send('comfyui:new-output-image', imageInfo); + log.info('New ComfyUI image detected:', filename); + } + }); + + log.info('ComfyUI output watcher started'); + return { success: true }; + } catch (error) { + log.error('Error starting ComfyUI output watcher:', error); + throw error; + } +}); + +ipcMain.handle('comfyui:stop-output-watcher', async () => { + try { + if (comfyuiOutputWatcher) { + comfyuiOutputWatcher.close(); + comfyuiOutputWatcher = null; + log.info('ComfyUI output watcher stopped'); + } + return { success: true }; + } catch (error) { + log.error('Error stopping ComfyUI output watcher:', error); + throw error; + } +}); + +// Delete ComfyUI output image +ipcMain.handle('comfyui:delete-output-image', async (event, imagePath) => { + try { + if (fs.existsSync(imagePath)) { + fs.unlinkSync(imagePath); + log.info('ComfyUI output image deleted:', imagePath); + return { success: true }; + } + return { success: false, error: 'File not found' }; + } catch (error) { + log.error('Error deleting ComfyUI output image:', error); + throw error; + } +}); + +// ============================================== +// ComfyUI Internal Model Management Service +// ============================================== + +// Get models stored inside ComfyUI container +ipcMain.handle('comfyui-internal:list-models', async (event, category = 'checkpoints') => { + try { + if (!comfyUIModelService) { + throw new Error('ComfyUI Model Service not initialized'); + } + + const models = await comfyUIModelService.listInstalledModels(category); + return { success: true, models }; + } catch (error) { + log.error(`Error listing ComfyUI ${category} models:`, error); + return { success: false, error: error.message, models: [] }; + } +}); + +// Get ComfyUI container storage information +ipcMain.handle('comfyui-internal:get-storage-info', async () => { + try { + if (!comfyUIModelService) { + throw new Error('ComfyUI Model Service not initialized'); + } + + const storageInfo = await comfyUIModelService.getStorageInfo(); + return { success: true, storage: storageInfo }; + } catch (error) { + log.error('Error getting ComfyUI storage info:', error); + return { success: false, error: error.message, storage: null }; + } +}); + +// Download and install model to ComfyUI container +ipcMain.handle('comfyui-internal:download-model', async (event, { url, filename, category = 'checkpoints', apiKey, source }) => { + try { + if (!comfyUIModelService) { + throw new Error('ComfyUI Model Service not initialized'); + } + + log.info(`Starting ComfyUI model download: ${filename} (${category}) from ${url}`); + log.info(`API key provided: ${!!apiKey}, Source: ${source || 'unknown'}`); + + // Set up progress forwarding + const progressHandler = (progressData) => { + event.sender.send('comfyui-internal-download-progress', progressData); + }; + + // Set up event forwarding + const eventHandlers = { + 'download:start': (data) => event.sender.send('comfyui-internal-download-start', data), + 'download:complete': (data) => event.sender.send('comfyui-internal-download-complete', data), + 'download:error': (data) => event.sender.send('comfyui-internal-download-error', data), + 'install:start': (data) => event.sender.send('comfyui-internal-install-start', data), + 'install:complete': (data) => event.sender.send('comfyui-internal-install-complete', data), + 'install:error': (data) => event.sender.send('comfyui-internal-install-error', data) + }; + + // Attach event listeners + Object.entries(eventHandlers).forEach(([eventName, handler]) => { + comfyUIModelService.on(eventName, handler); + }); + + try { + // Prepare options for download + const downloadOptions = { + apiKey, + source: source || (url.includes('civitai.com') ? 'civitai' : url.includes('huggingface.co') ? 'huggingface' : 'unknown') + }; + + const result = await comfyUIModelService.downloadAndInstallModel(url, filename, category, progressHandler, downloadOptions); + + // Clean up event listeners + Object.entries(eventHandlers).forEach(([eventName, handler]) => { + comfyUIModelService.removeListener(eventName, handler); + }); + + return result; + } catch (error) { + // Clean up event listeners on error + Object.entries(eventHandlers).forEach(([eventName, handler]) => { + comfyUIModelService.removeListener(eventName, handler); + }); + throw error; + } + + } catch (error) { + log.error('Error downloading ComfyUI model to container:', error); + return { + success: false, + filename, + category, + error: error.message + }; + } +}); + +// Remove model from ComfyUI container +ipcMain.handle('comfyui-internal:remove-model', async (event, { filename, category = 'checkpoints' }) => { + try { + if (!comfyUIModelService) { + throw new Error('ComfyUI Model Service not initialized'); + } + + const result = await comfyUIModelService.removeModel(filename, category); + log.info(`Removed ComfyUI model: ${filename} from ${category}`); + return result; + } catch (error) { + log.error('Error removing ComfyUI model from container:', error); + return { success: false, error: error.message }; + } +}); + +// Get ComfyUI model management status +ipcMain.handle('comfyui-internal:get-status', async () => { + try { + if (!comfyUIModelService) { + throw new Error('ComfyUI Model Service not initialized'); + } + + const status = await comfyUIModelService.getStatus(); + return { success: true, status }; + } catch (error) { + log.error('Error getting ComfyUI service status:', error); + return { success: false, error: error.message, status: null }; + } +}); + +// Search for models from external repositories +ipcMain.handle('comfyui-internal:search-models', async (event, { query, source = 'huggingface', category = 'checkpoints' }) => { + try { + if (!comfyUIModelService) { + throw new Error('ComfyUI Model Service not initialized'); + } + + const results = await comfyUIModelService.searchModels(query, source, category); + return { success: true, results }; + } catch (error) { + log.error('Error searching ComfyUI models:', error); + return { success: false, error: error.message, results: null }; + } +}); + +// Backup models from container to host +ipcMain.handle('comfyui-internal:backup-models', async (event, { category = 'checkpoints', backupPath }) => { + try { + if (!comfyUIModelService) { + throw new Error('ComfyUI Model Service not initialized'); + } + + // Use user data directory if no backup path specified + if (!backupPath) { + backupPath = path.join(app.getPath('userData'), 'comfyui_backups'); + if (!fs.existsSync(backupPath)) { + fs.mkdirSync(backupPath, { recursive: true }); + } + } + + const result = await comfyUIModelService.backupModels(category, backupPath); + log.info(`ComfyUI models backed up: ${category} to ${result.backupFile}`); + return result; + } catch (error) { + log.error('Error backing up ComfyUI models:', error); + return { success: false, error: error.message }; + } +}); + +// ============================================== +// Enhanced Local Model Management +// ============================================== + +// List locally stored persistent models +ipcMain.handle('comfyui-local:list-models', async (event, category = 'checkpoints') => { + try { + // Ensure ComfyUI Model Service is initialized + if (!comfyUIModelService) { + log.info('🎨 Initializing ComfyUI Model Service for list models request'); + comfyUIModelService = new ComfyUIModelService(); + } + + const models = await comfyUIModelService.listLocalModels(category); + return { success: true, models }; + } catch (error) { + log.error(`Error listing local ComfyUI ${category} models:`, error); + return { success: false, error: error.message, models: [] }; + } +}); + +// Download model to local storage (persistent) +ipcMain.handle('comfyui-local:download-model', async (event, { url, filename, category = 'checkpoints', apiKey, source }) => { + try { + // Ensure ComfyUI Model Service is initialized for downloads + if (!comfyUIModelService) { + log.info('🎨 Initializing ComfyUI Model Service for download request'); + try { + comfyUIModelService = new ComfyUIModelService(); + log.info('✅ ComfyUI Model Service initialized successfully'); + } catch (initError) { + log.error('❌ Failed to initialize ComfyUI Model Service:', initError); + throw new Error(`Failed to initialize ComfyUI Model Service: ${initError.message}`); + } + } + + log.info(`Starting local ComfyUI model download: ${filename} (${category}) from ${url}`); + log.info(`API key provided: ${!!apiKey}, Source: ${source || 'unknown'}`); + + // Set up progress forwarding - fix the parameter format + const progressHandler = (progress, downloadedSize, totalSize) => { + const progressData = { + filename, + progress: Math.round(progress), + downloadedSize, + totalSize, + speed: downloadedSize > 0 ? `${(downloadedSize / 1024 / 1024).toFixed(1)} MB/s` : '0 MB/s', + eta: totalSize > 0 && downloadedSize > 0 ? `${Math.round((totalSize - downloadedSize) / (downloadedSize / 1000))}s` : 'Unknown' + }; + event.sender.send('comfyui-local-download-progress', progressData); + }; + + // Set up event forwarding + const eventHandlers = { + 'download:start': (data) => event.sender.send('comfyui-local-download-start', data), + 'download:complete': (data) => event.sender.send('comfyui-local-download-complete', data), + 'download:error': (data) => event.sender.send('comfyui-local-download-error', data), + 'download:progress': (data) => { + // Also handle progress events from the service + const progressData = { + filename: data.filename || filename, + progress: Math.round(data.progress || 0), + downloadedSize: data.downloadedSize || 0, + totalSize: data.totalSize || 0, + speed: data.speed || '0 MB/s', + eta: data.eta || 'Unknown' + }; + event.sender.send('comfyui-local-download-progress', progressData); + } + }; + + // Attach event listeners + Object.entries(eventHandlers).forEach(([eventName, handler]) => { + comfyUIModelService.on(eventName, handler); + }); + + try { + // Prepare options for download + const downloadOptions = { + apiKey, + source: source || (url.includes('civitai.com') ? 'civitai' : url.includes('huggingface.co') ? 'huggingface' : 'unknown') + }; + + const result = await comfyUIModelService.downloadModel(url, filename, category, progressHandler, 0, downloadOptions); + + // Clean up event listeners + Object.entries(eventHandlers).forEach(([eventName, handler]) => { + comfyUIModelService.removeListener(eventName, handler); + }); + + return { success: true, ...result }; + } catch (error) { + // Clean up event listeners on error + Object.entries(eventHandlers).forEach(([eventName, handler]) => { + comfyUIModelService.removeListener(eventName, handler); + }); + throw error; + } + + } catch (error) { + log.error('Error downloading ComfyUI model to local storage:', error); + log.error('Error details:', { + filename, + category, + url: url.substring(0, 100) + '...', + error: error.message, + stack: error.stack + }); + + // Send error event to frontend + event.sender.send('comfyui-local-download-error', { + filename, + category, + error: error.message + }); + + return { + success: false, + filename, + category, + error: error.message + }; + } +}); + +// Delete local persistent model +ipcMain.handle('comfyui-local:delete-model', async (event, { filename, category = 'checkpoints' }) => { + try { + // Ensure ComfyUI Model Service is initialized + if (!comfyUIModelService) { + log.info('🎨 Initializing ComfyUI Model Service for delete model request'); + comfyUIModelService = new ComfyUIModelService(); + } + + const result = await comfyUIModelService.deleteLocalModel(filename, category); + log.info(`Deleted local ComfyUI model: ${filename} from ${category}`); + return result; + } catch (error) { + log.error('Error deleting local ComfyUI model:', error); + return { success: false, error: error.message }; + } +}); + +// Import external model file to persistent storage +ipcMain.handle('comfyui-local:import-model', async (event, { externalPath, filename, category = 'checkpoints' }) => { + try { + // Ensure ComfyUI Model Service is initialized + if (!comfyUIModelService) { + log.info('🎨 Initializing ComfyUI Model Service for import model request'); + comfyUIModelService = new ComfyUIModelService(); + } + + const result = await comfyUIModelService.importExternalModel(externalPath, filename, category); + log.info(`Imported external ComfyUI model: ${filename} to ${category}`); + return result; + } catch (error) { + log.error('Error importing external ComfyUI model:', error); + return { success: false, error: error.message }; + } +}); + +// Get enhanced storage information (local + container) +ipcMain.handle('comfyui-local:get-storage-info', async () => { + try { + // Ensure ComfyUI Model Service is initialized + if (!comfyUIModelService) { + log.info('🎨 Initializing ComfyUI Model Service for storage info request'); + comfyUIModelService = new ComfyUIModelService(); + } + + const storageInfo = await comfyUIModelService.getEnhancedStorageInfo(); + return { success: true, storage: storageInfo }; + } catch (error) { + log.error('Error getting enhanced ComfyUI storage info:', error); + return { success: false, error: error.message, storage: null }; + } +}); + +// ============================================================================ +// Netlify OAuth IPC Handlers +// ============================================================================ + +let netlifyOAuthHandler = null; + +ipcMain.handle('netlify-oauth:authenticate', async (event, authUrl) => { + try { + if (!netlifyOAuthHandler) { + netlifyOAuthHandler = new NetlifyOAuthHandler(); + } + + const accessToken = await netlifyOAuthHandler.authenticate(authUrl); + return { success: true, accessToken }; + } catch (error) { + log.error('Netlify OAuth error:', error); + return { success: false, error: error.message }; + } +}); + +ipcMain.handle('netlify-oauth:cancel', async () => { + try { + if (netlifyOAuthHandler) { + netlifyOAuthHandler.cancel(); + } + return { success: true }; + } catch (error) { + log.error('Error canceling Netlify OAuth:', error); + return { success: false, error: error.message }; + } +}); + +// ============================================================================ + +// Find the initializeServicesInBackground function and add central service manager integration +async function initializeServicesInBackground() { + try { + log.info('Starting remaining services initialization (Docker mode)...'); + + // Send initialization status to renderer if main window is ready + const sendStatus = (service, status, type = 'info') => { + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('background-service-status', { service, status, type }); + } + log.info(`[Docker Mode] ${service}: ${status}`); + }; + + // NEW: Start central service manager services first + if (centralServiceManager) { + try { + sendStatus('System', 'Starting service management system...', 'info'); + log.info('🎯 Starting central service manager services...'); + + // Update service states based on Docker container status + if (dockerSetup) { + await updateCentralServiceManagerWithDockerStatus(); + } + + // Start manual services through central manager + await startManualServicesInCentralManager(); + + sendStatus('System', 'Service management system started', 'success'); + } catch (error) { + log.error('❌ Error starting central service manager services:', error); + sendStatus('System', 'Service management system startup failed', 'warning'); + } + } + + // Initialize MCP service in background + sendStatus('MCP', 'Initializing MCP service...', 'info'); + try { + // Initialize MCP service if not already initialized + if (!mcpService) { + mcpService = new MCPService(); + } + sendStatus('MCP', 'MCP service initialized', 'success'); + + // Update central service manager with MCP status + if (centralServiceManager) { + updateServiceStateInCentralManager('mcp', 'running', { + type: 'service', + startTime: Date.now(), + healthCheck: () => mcpService && Object.keys(mcpService.servers).length > 0 + }); + } + + // Check startup settings for MCP auto-start + sendStatus('MCP', 'Checking startup settings...', 'info'); + let shouldAutoStartMCP = true; // Default to true for backward compatibility + + try { + const { autoStartMCP } = getStartupPreferences(); + shouldAutoStartMCP = autoStartMCP; + } catch (settingsError) { + log.warn('Error reading startup settings for MCP auto-start:', settingsError); + // Default to true on error to maintain existing behavior + } + + if (shouldAutoStartMCP) { + // Auto-start previously running servers + sendStatus('MCP', 'Restoring MCP servers...', 'info'); + try { + const restoreResults = await mcpService.startPreviouslyRunningServers(); + const successCount = restoreResults.filter(r => r.success).length; + const totalCount = restoreResults.length; + + if (totalCount > 0) { + sendStatus('MCP', `Restored ${successCount}/${totalCount} MCP servers`, successCount === totalCount ? 'success' : 'warning'); + } else { + sendStatus('MCP', 'No MCP servers to restore', 'info'); + } + global.mcpServersRestored = true; // Mark as restored to prevent duplicate restoration + } catch (restoreError) { + log.error('Error restoring MCP servers:', restoreError); + sendStatus('MCP', 'Failed to restore some MCP servers', 'warning'); + } + } else { + sendStatus('MCP', 'MCP auto-start disabled in settings', 'info'); + log.info('MCP auto-start is disabled in startup settings'); + global.mcpServersRestored = true; // Mark as "restored" to prevent later attempts + } + } catch (mcpError) { + log.error('Error initializing MCP service:', mcpError); + sendStatus('MCP', 'MCP service initialization failed', 'warning'); + if (centralServiceManager) { + updateServiceStateInCentralManager('mcp', 'error', null); + } + } + + // Initialize MCP HTTP Proxy service for browser support + sendStatus('MCP Proxy', 'Starting MCP HTTP proxy...', 'info'); + try { + if (mcpService) { + const MCPProxyService = require('./mcpProxyService.cjs'); + const mcpProxyService = new MCPProxyService(mcpService); + await mcpProxyService.start(8092); + sendStatus('MCP Proxy', 'MCP HTTP proxy started on port 8092', 'success'); + log.info('✅ MCP HTTP proxy service started successfully'); + } else { + log.warn('⚠️ MCP service not available, skipping proxy start'); + sendStatus('MCP Proxy', 'Skipped (MCP service not available)', 'warning'); + } + } catch (proxyError) { + log.error('Error starting MCP HTTP proxy:', proxyError); + sendStatus('MCP Proxy', 'MCP proxy failed to start', 'warning'); + } + + // Initialize Watchdog service in background (with Docker support) + sendStatus('Watchdog', 'Initializing Watchdog service...', 'info'); + try { + watchdogService = new WatchdogService(dockerSetup,mcpService, ipcLogger); + + // Set up event listeners for watchdog events + watchdogService.on('serviceRestored', (serviceKey, service) => { + log.info(`Watchdog: ${service.name} has been restored`); + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('watchdog-service-restored', { serviceKey, service: service.name }); + } + }); + + watchdogService.on('serviceFailed', (serviceKey, service) => { + log.error(`Watchdog: ${service.name} has failed after maximum retry attempts`); + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('watchdog-service-failed', { serviceKey, service: service.name }); + } + }); + + watchdogService.on('serviceRestarted', (serviceKey, service) => { + log.info(`Watchdog: ${service.name} has been restarted successfully`); + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('watchdog-service-restarted', { serviceKey, service: service.name }); + } + }); + + // Start the watchdog monitoring + watchdogService.start(); + + // Signal watchdog service that Docker setup is complete + watchdogService.signalSetupComplete(); + + sendStatus('Watchdog', 'Watchdog service started successfully', 'success'); + } catch (watchdogError) { + log.error('Error initializing Watchdog service:', watchdogError); + sendStatus('Watchdog', 'Watchdog service initialization failed', 'warning'); + } + + // Notify that Docker mode initialization is complete + sendStatus('System', 'Docker mode initialization complete', 'success'); + log.info('Docker mode service initialization completed'); + + } catch (error) { + log.error('Error during Docker mode service initialization:', error); + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.webContents.send('background-service-error', { + service: 'System', + error: `Docker mode initialization error: ${error.message}` + }); + } + } +} + +// NEW: Helper function to update central service manager with Docker container status +async function updateCentralServiceManagerWithDockerStatus() { + if (!dockerSetup || !centralServiceManager) return; + + try { + log.info('🔄 Updating central service manager with Docker container status...'); + + // Check Docker daemon status + const dockerRunning = await dockerSetup.isDockerRunning(); + if (dockerRunning) { + updateServiceStateInCentralManager('docker', 'running', { + type: 'docker-daemon', + startTime: Date.now(), + healthCheck: () => dockerSetup.isDockerRunning() + }); + } else { + updateServiceStateInCentralManager('docker', 'stopped', null); + } + + // Check individual container status + const containerServices = ['python-backend', 'n8n', 'comfyui']; + + for (const serviceName of containerServices) { + try { + const containerName = `clara_${serviceName.replace('-backend', '')}`; + const container = dockerSetup.docker.getContainer(containerName); + const containerInfo = await container.inspect(); + + if (containerInfo.State.Running) { + const serviceUrl = getServiceUrlFromContainer(serviceName, containerInfo); + log.info(`🔗 Detected Docker service URL for ${serviceName}: ${serviceUrl}`); + updateServiceStateInCentralManager(serviceName, 'running', { + type: 'docker-container', + containerName: containerName, + startTime: Date.now(), + url: serviceUrl, + healthCheck: () => checkContainerHealth(containerName) + }); + } else { + updateServiceStateInCentralManager(serviceName, 'stopped', null); + } + } catch (error) { + // Container not found or error - mark as stopped + updateServiceStateInCentralManager(serviceName, 'stopped', null); + log.debug(`Container ${serviceName} not found or not running`); + } + } + + log.info('✅ Central service manager updated with Docker status'); + } catch (error) { + log.error('❌ Error updating central service manager with Docker status:', error); + } +} + +// NEW: Helper function to start manual services in central manager +async function startManualServicesInCentralManager() { + if (!centralServiceManager || !serviceConfigManager) return; + + try { + log.info('🔄 Starting manual services in central service manager...'); + + if (serviceConfigManager && typeof serviceConfigManager.getAllServiceConfigs === 'function') { + const allConfigs = serviceConfigManager.getAllServiceConfigs(); + + for (const [serviceName, config] of Object.entries(allConfigs)) { + if (config.mode === 'manual' && config.url) { + try { + await centralServiceManager.startService(serviceName); + log.info(`✅ Manual service ${serviceName} started via central manager`); + } catch (error) { + log.error(`❌ Failed to start manual service ${serviceName}:`, error); + } + } + } + } else { + log.warn('Service config manager not available, skipping manual services startup'); + } + + log.info('✅ Manual services startup completed'); + } catch (error) { + log.error('❌ Error starting manual services:', error); + } +} + +// NEW: Helper function to update service state in central manager +function updateServiceStateInCentralManager(serviceName, state, instance) { + if (!centralServiceManager) return; + + try { + const service = centralServiceManager.services.get(serviceName); + if (service) { + service.state = state; + service.instance = instance; + service.lastHealthCheck = Date.now(); + + if (instance && instance.url) { + service.serviceUrl = instance.url; + log.debug(`🎯 Set serviceUrl for ${serviceName}: ${instance.url}`); + } + + centralServiceManager.serviceStates.set(serviceName, state); + + log.debug(`📊 Updated ${serviceName} state to ${state} in central manager`); + } + } catch (error) { + log.error(`❌ Error updating service state for ${serviceName}:`, error); + } +} + +// NEW: Helper function to get service URL from container info +function getServiceUrlFromContainer(serviceName, containerInfo) { + try { + const ports = containerInfo.NetworkSettings.Ports; + + // Service-specific port mapping + const servicePortMap = { + 'python-backend': '5001', + 'n8n': '5678', + 'comfyui': '8188' + }; + + const targetPort = servicePortMap[serviceName]; + if (!targetPort) return null; + + const portKey = `${targetPort}/tcp`; + if (ports[portKey] && ports[portKey][0]) { + const hostPort = ports[portKey][0].HostPort; + return `http://localhost:${hostPort}`; + } + + return `http://localhost:${targetPort}`; + } catch (error) { + log.error(`Error getting service URL for ${serviceName}:`, error); + return null; + } +} + +// NEW: Helper function to check container health +async function checkContainerHealth(containerName) { + if (!dockerSetup) return false; + + try { + const container = dockerSetup.docker.getContainer(containerName); + const containerInfo = await container.inspect(); + return containerInfo.State.Running; + } catch (error) { + return false; + } +} + +// Register global shortcuts for quick access +function registerGlobalShortcuts() { + try { + // Clear any existing shortcuts to avoid conflicts + globalShortcut.unregisterAll(); + + // Define shortcuts based on platform + const shortcuts = process.platform === 'darwin' + ? ['Option+Ctrl+Space'] + : ['Ctrl+Alt+Space']; + + // Debounce variables to prevent multiple rapid triggers + let lastTriggerTime = 0; + const debounceDelay = 500; // 500ms debounce + + shortcuts.forEach(shortcut => { + const ret = globalShortcut.register(shortcut, () => { + const now = Date.now(); + + // Check if we're within the debounce period + if (now - lastTriggerTime < debounceDelay) { + log.info(`Global shortcut ${shortcut} debounced - too soon after last trigger`); + return; + } + + lastTriggerTime = now; + log.info(`Global shortcut ${shortcut} pressed - bringing Clara to foreground`); + + // Bring window to foreground + if (mainWindow) { + if (mainWindow.isMinimized()) { + mainWindow.restore(); + } + + // Focus and show the window + mainWindow.focus(); + mainWindow.show(); + + // Send message to renderer to start new chat + mainWindow.webContents.send('trigger-new-chat'); + } else { + log.warn('Main window not available for global shortcut'); + } + }); + + if (!ret) { + log.error(`Failed to register global shortcut: ${shortcut}`); + } else { + log.info(`Successfully registered global shortcut: ${shortcut}`); + } + }); + + log.info(`Global shortcuts registered for platform: ${process.platform}`); + } catch (error) { + log.error('Error registering global shortcuts:', error); + } +} + +// Add tray creation function +function createTray() { + if (tray) return; + + try { + // Try to use the actual logo file first + const possibleLogoPaths = [ + path.join(__dirname, 'assets', 'tray-icon.png'), + path.join(__dirname, '../public/logo.png'), + path.join(__dirname, '../src/assets/logo.png'), + path.join(__dirname, '../assets/icons/logo.png'), + path.join(__dirname, '../assets/icons/png/logo.png') + ]; + + let trayIcon; + let logoFound = false; + + // Try to find and use the actual logo + for (const logoPath of possibleLogoPaths) { + if (fs.existsSync(logoPath)) { + try { + trayIcon = nativeImage.createFromPath(logoPath); + + // Resize for tray - different sizes for different platforms + if (process.platform === 'darwin') { + // macOS prefers 16x16 for tray icons + trayIcon = trayIcon.resize({ width: 16, height: 16 }); + // Set as template image for proper macOS styling + trayIcon.setTemplateImage(true); + } else if (process.platform === 'win32') { + // Windows prefers 16x16 or 32x32 + trayIcon = trayIcon.resize({ width: 16, height: 16 }); + } else { + // Linux typically uses 22x22 or 24x24 + trayIcon = trayIcon.resize({ width: 22, height: 22 }); + } + + logoFound = true; + log.info(`Using logo from: ${logoPath}`); + break; + } catch (error) { + log.warn(`Failed to load logo from ${logoPath}:`, error); + } + } + } + + // Fallback to programmatic icon if logo not found + if (!logoFound) { + log.info('Logo file not found, creating programmatic icon'); + const iconSize = process.platform === 'darwin' ? 16 : (process.platform === 'win32' ? 16 : 22); + + if (process.platform === 'darwin') { + // For macOS, create a simple template icon (must be black/transparent for template) + const canvas = ` + + + C + + `; + trayIcon = nativeImage.createFromDataURL(`data:image/svg+xml;base64,${Buffer.from(canvas).toString('base64')}`); + trayIcon.setTemplateImage(true); + } else { + // For Windows/Linux, create a colored icon matching ClaraVerse brand colors + const canvas = ` + + + + + + + + + C + + `; + trayIcon = nativeImage.createFromDataURL(`data:image/svg+xml;base64,${Buffer.from(canvas).toString('base64')}`); + } + } + + // Create the tray + tray = new Tray(trayIcon); + + // Set tooltip + tray.setToolTip('ClaraVerse'); + + // Create context menu + const contextMenu = Menu.buildFromTemplate([ + { + label: 'Show ClaraVerse', + click: () => { + try { + if (mainWindow && !mainWindow.isDestroyed()) { + if (mainWindow.isMinimized()) { + mainWindow.restore(); + } + mainWindow.show(); + mainWindow.focus(); + + // CRITICAL FIX: Force webContents focus when showing from tray menu + if (mainWindow.webContents && !mainWindow.webContents.isDestroyed()) { + setTimeout(() => { + if (mainWindow && !mainWindow.isDestroyed() && mainWindow.webContents) { + mainWindow.webContents.focus(); + } + }, 100); + } + } else if (!isQuitting) { + createMainWindow(); + } + } catch (error) { + log.error('Error showing window from tray:', error); + } + } + }, + { + label: 'Hide ClaraVerse', + click: () => { + try { + if (mainWindow && !mainWindow.isDestroyed()) { + mainWindow.hide(); + } + } catch (error) { + log.error('Error hiding window from tray:', error); + } + } + }, + { type: 'separator' }, + { + label: 'Quit', + click: () => { + try { + isQuitting = true; + app.quit(); + } catch (error) { + log.error('Error quitting from tray:', error); + process.exit(0); // Force exit as last resort + } + } + } + ]); + + tray.setContextMenu(contextMenu); + + // Handle tray click + tray.on('click', () => { + try { + if (mainWindow && !mainWindow.isDestroyed()) { + if (mainWindow.isVisible()) { + mainWindow.hide(); + } else { + if (mainWindow.isMinimized()) { + mainWindow.restore(); + } + mainWindow.show(); + mainWindow.focus(); + + // CRITICAL FIX: Force webContents focus on tray click + if (mainWindow.webContents && !mainWindow.webContents.isDestroyed()) { + setTimeout(() => { + if (mainWindow && !mainWindow.isDestroyed() && mainWindow.webContents) { + mainWindow.webContents.focus(); + } + }, 100); + } + } + } else if (!isQuitting) { + createMainWindow(); + } + } catch (error) { + log.error('Error handling tray click:', error); + } + }); + + // Handle double-click on tray (Windows/Linux) + tray.on('double-click', () => { + try { + if (mainWindow && !mainWindow.isDestroyed()) { + if (mainWindow.isMinimized()) { + mainWindow.restore(); + } + mainWindow.show(); + mainWindow.focus(); + + // CRITICAL FIX: Force webContents focus on tray double-click + if (mainWindow.webContents && !mainWindow.webContents.isDestroyed()) { + setTimeout(() => { + if (mainWindow && !mainWindow.isDestroyed() && mainWindow.webContents) { + mainWindow.webContents.focus(); + } + }, 100); + } + } else if (!isQuitting) { + createMainWindow(); + } + } catch (error) { + log.error('Error handling tray double-click:', error); + } + }); + + log.info('System tray created successfully'); + } catch (error) { + log.error('Error creating system tray:', error); + } +} \ No newline at end of file diff --git a/electron/mcpProxyService.cjs b/electron/mcpProxyService.cjs new file mode 100644 index 00000000..450a1220 --- /dev/null +++ b/electron/mcpProxyService.cjs @@ -0,0 +1,448 @@ +const express = require('express'); +const cors = require('cors'); +const log = require('electron-log'); + +/** + * MCP HTTP Proxy Service + * + * Provides HTTP/REST API access to MCP (Model Context Protocol) functionality. + * This service bridges the gap between browser-based clients and the Node.js-based + * MCP service, allowing web applications to use MCP tools without Electron. + * + * Architecture: + * Browser Client → HTTP API (port 8092) → MCPService → MCP Servers + * + * Key Features: + * - Full MCP protocol support (server management, tool execution, discovery) + * - CORS enabled for localhost + * - No authentication (trusted localhost environment) + * - Health check endpoint for monitoring + * - Auto-start with Electron app + */ +class MCPProxyService { + constructor(mcpService) { + this.mcpService = mcpService; + this.app = express(); + this.server = null; + this.port = 8092; + this.isRunning = false; + + this.setupMiddleware(); + this.setupRoutes(); + } + + setupMiddleware() { + // Enable CORS for localhost (browser access) + this.app.use(cors({ + origin: [ + 'http://localhost:3000', + 'http://localhost:5173', + 'http://127.0.0.1:3000', + 'http://127.0.0.1:5173', + 'http://127.0.0.1:37117', // Loopback server port + /^http:\/\/localhost:\d+$/, // Any localhost port + /^http:\/\/127\.0\.0\.1:\d+$/ // Any 127.0.0.1 port + ], + credentials: true + })); + + // Parse JSON bodies + this.app.use(express.json({ limit: '50mb' })); + + // Request logging + this.app.use((req, res, next) => { + log.info(`[MCP Proxy] ${req.method} ${req.path}`); + next(); + }); + + // Error handler + this.app.use((err, req, res, next) => { + log.error('[MCP Proxy] Error:', err); + res.status(500).json({ + success: false, + error: err.message || 'Internal server error' + }); + }); + } + + setupRoutes() { + // Health check + this.app.get('/health', (req, res) => { + res.json({ + success: true, + status: 'healthy', + service: 'mcp-proxy', + port: this.port, + timestamp: new Date().toISOString() + }); + }); + + // Get all servers + this.app.get('/api/servers', async (req, res) => { + try { + const servers = this.mcpService.getAllServers(); + res.json({ + success: true, + servers + }); + } catch (error) { + log.error('[MCP Proxy] Error getting servers:', error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Get server status + this.app.get('/api/servers/:name', async (req, res) => { + try { + const { name } = req.params; + const status = this.mcpService.getServerStatus(name); + res.json({ + success: true, + status + }); + } catch (error) { + log.error(`[MCP Proxy] Error getting server status for ${req.params.name}:`, error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Add server + this.app.post('/api/servers', async (req, res) => { + try { + const serverConfig = req.body; + const result = await this.mcpService.addServer(serverConfig); + res.json({ + success: true, + result + }); + } catch (error) { + log.error('[MCP Proxy] Error adding server:', error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Update server + this.app.put('/api/servers/:name', async (req, res) => { + try { + const { name } = req.params; + const updates = req.body; + const result = await this.mcpService.updateServer(name, updates); + res.json({ + success: true, + result + }); + } catch (error) { + log.error(`[MCP Proxy] Error updating server ${req.params.name}:`, error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Remove server + this.app.delete('/api/servers/:name', async (req, res) => { + try { + const { name } = req.params; + await this.mcpService.removeServer(name); + res.json({ + success: true + }); + } catch (error) { + log.error(`[MCP Proxy] Error removing server ${req.params.name}:`, error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Start server + this.app.post('/api/servers/:name/start', async (req, res) => { + try { + const { name } = req.params; + await this.mcpService.startServer(name); + res.json({ + success: true + }); + } catch (error) { + log.error(`[MCP Proxy] Error starting server ${req.params.name}:`, error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Stop server + this.app.post('/api/servers/:name/stop', async (req, res) => { + try { + const { name } = req.params; + await this.mcpService.stopServer(name); + res.json({ + success: true + }); + } catch (error) { + log.error(`[MCP Proxy] Error stopping server ${req.params.name}:`, error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Restart server + this.app.post('/api/servers/:name/restart', async (req, res) => { + try { + const { name } = req.params; + await this.mcpService.restartServer(name); + res.json({ + success: true + }); + } catch (error) { + log.error(`[MCP Proxy] Error restarting server ${req.params.name}:`, error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Test server + this.app.post('/api/servers/:name/test', async (req, res) => { + try { + const { name } = req.params; + const result = await this.mcpService.testServer(name); + res.json({ + success: true, + result + }); + } catch (error) { + log.error(`[MCP Proxy] Error testing server ${req.params.name}:`, error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Execute tool call + this.app.post('/api/tools/execute', async (req, res) => { + try { + const toolCall = req.body; + const result = await this.mcpService.executeToolCall(toolCall); + res.json({ + success: true, + result + }); + } catch (error) { + log.error('[MCP Proxy] Error executing tool call:', error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Get server templates + this.app.get('/api/templates', async (req, res) => { + try { + const templates = this.mcpService.getServerTemplates(); + res.json({ + success: true, + templates + }); + } catch (error) { + log.error('[MCP Proxy] Error getting templates:', error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Start all enabled servers + this.app.post('/api/servers/start-all-enabled', async (req, res) => { + try { + await this.mcpService.startAllEnabledServers(); + res.json({ + success: true + }); + } catch (error) { + log.error('[MCP Proxy] Error starting all enabled servers:', error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Stop all servers + this.app.post('/api/servers/stop-all', async (req, res) => { + try { + await this.mcpService.stopAllServers(); + res.json({ + success: true + }); + } catch (error) { + log.error('[MCP Proxy] Error stopping all servers:', error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Start previously running servers + this.app.post('/api/servers/start-previously-running', async (req, res) => { + try { + await this.mcpService.startPreviouslyRunningServers(); + res.json({ + success: true + }); + } catch (error) { + log.error('[MCP Proxy] Error starting previously running servers:', error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Save running state + this.app.post('/api/servers/save-running-state', async (req, res) => { + try { + this.mcpService.saveRunningState(); + res.json({ + success: true + }); + } catch (error) { + log.error('[MCP Proxy] Error saving running state:', error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Import Claude config + this.app.post('/api/import-claude-config', async (req, res) => { + try { + const { configPath } = req.body; + const result = await this.mcpService.importFromClaudeConfig(configPath); + res.json({ + success: true, + result + }); + } catch (error) { + log.error('[MCP Proxy] Error importing Claude config:', error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + + // Diagnose Node installation + this.app.get('/api/diagnose-node', async (req, res) => { + try { + const diagnosis = await this.mcpService.diagnoseNodeInstallation(); + res.json({ + success: true, + diagnosis + }); + } catch (error) { + log.error('[MCP Proxy] Error diagnosing Node:', error); + res.status(500).json({ + success: false, + error: error.message + }); + } + }); + } + + /** + * Start the HTTP proxy server + */ + async start(port = 8092) { + if (this.isRunning) { + log.warn('[MCP Proxy] Server is already running'); + return; + } + + this.port = port; + + return new Promise((resolve, reject) => { + try { + this.server = this.app.listen(this.port, '127.0.0.1', () => { + this.isRunning = true; + log.info(`[MCP Proxy] Server started on http://127.0.0.1:${this.port}`); + resolve({ + port: this.port, + url: `http://127.0.0.1:${this.port}`, + healthCheck: async () => { + try { + const response = await fetch(`http://127.0.0.1:${this.port}/health`); + return response.ok; + } catch (error) { + return false; + } + } + }); + }); + + this.server.on('error', (error) => { + log.error('[MCP Proxy] Server error:', error); + if (error.code === 'EADDRINUSE') { + reject(new Error(`Port ${this.port} is already in use`)); + } else { + reject(error); + } + }); + } catch (error) { + log.error('[MCP Proxy] Failed to start server:', error); + reject(error); + } + }); + } + + /** + * Stop the HTTP proxy server + */ + async stop() { + if (!this.isRunning || !this.server) { + log.warn('[MCP Proxy] Server is not running'); + return; + } + + return new Promise((resolve) => { + this.server.close(() => { + this.isRunning = false; + log.info('[MCP Proxy] Server stopped'); + resolve(); + }); + }); + } + + /** + * Get service status + */ + getStatus() { + return { + isRunning: this.isRunning, + port: this.port, + url: this.isRunning ? `http://127.0.0.1:${this.port}` : null + }; + } +} + +module.exports = MCPProxyService; diff --git a/electron/mcpService.cjs b/electron/mcpService.cjs new file mode 100644 index 00000000..7f91f2ff --- /dev/null +++ b/electron/mcpService.cjs @@ -0,0 +1,1887 @@ +const { spawn } = require('child_process'); +const path = require('path'); +const fs = require('fs'); +const log = require('electron-log'); +const { app } = require('electron'); +const os = require('os'); + +class MCPService { + constructor() { + this.servers = new Map(); + this.configPath = path.join(app.getPath('userData'), 'mcp_config.json'); + this.loadConfig(); + } + + loadConfig() { + try { + if (fs.existsSync(this.configPath)) { + const configData = fs.readFileSync(this.configPath, 'utf8'); + this.config = JSON.parse(configData); + } else { + this.config = { + mcpServers: {}, + lastRunningServers: [] // Track which servers were running when app was closed + }; + this.saveConfig(); + } + + // Ensure lastRunningServers exists for backward compatibility + if (!this.config.lastRunningServers) { + this.config.lastRunningServers = []; + } + + // Schedule Clara's MCP server check for next tick to avoid sync/async issues + setImmediate(() => { + this.ensureClaraMCPExists().catch(error => { + log.error('Failed to ensure Clara MCP server exists during config load:', error); + }); + }); + } catch (error) { + log.error('Error loading MCP config:', error); + this.config = { + mcpServers: {}, + lastRunningServers: [] + }; + // Schedule Clara's MCP server check even after an error + setImmediate(() => { + this.ensureClaraMCPExists().catch(ensureError => { + log.error('Failed to ensure Clara MCP server exists after config load error:', ensureError); + }); + }); + } + } + + saveConfig() { + try { + fs.writeFileSync(this.configPath, JSON.stringify(this.config, null, 2)); + } catch (error) { + log.error('Error saving MCP config:', error); + } + } + + // Ensure Clara's Python MCP server always exists + async ensureClaraMCPExists() { + try { + // Check if Clara's Python MCP server exists + if (!this.config.mcpServers['python-mcp']) { + log.info('Clara\'s Python MCP server missing, restoring it...'); + + try { + // Get the resolved executable path + const executablePath = this.resolveBundledExecutablePath('python-mcp-server'); + + this.config.mcpServers['python-mcp'] = { + type: 'stdio', + command: executablePath, + args: [], + env: {}, + description: 'Bundled Python MCP Server (Clara Native) - Always Available', + enabled: true, + createdAt: new Date().toISOString() + }; + + this.saveConfig(); + log.info('Clara\'s Python MCP server restored successfully'); + } catch (pathError) { + log.warn('Failed to resolve bundled executable path, Clara MCP server may not work:', pathError); + // Still create the entry so the system knows it should exist + this.config.mcpServers['python-mcp'] = { + type: 'stdio', + command: 'python-mcp-server', // Fallback to the command name + args: [], + env: {}, + description: 'Bundled Python MCP Server (Clara Native) - Always Available (Path Unresolved)', + enabled: false, // Disable since path couldn't be resolved + createdAt: new Date().toISOString() + }; + this.saveConfig(); + log.info('Clara\'s Python MCP server entry created but disabled due to path resolution failure'); + } + } + } catch (error) { + log.error('Error ensuring Clara\'s MCP server exists:', error); + } + } + + async addServer(serverConfig) { + const { name, type, command, args, env, description, url, headers } = serverConfig; + + if (this.config.mcpServers[name]) { + throw new Error(`MCP server '${name}' already exists`); + } + + const serverType = type || 'stdio'; + + // Validate required fields based on server type + if (serverType === 'remote') { + if (!url) { + throw new Error('URL is required for remote MCP servers'); + } + } else if (serverType === 'stdio') { + if (!command) { + throw new Error('Command is required for stdio MCP servers'); + } + } + + this.config.mcpServers[name] = { + type: serverType, + command, + args: args || [], + env: env || {}, + url, + headers: headers || {}, + description: description || '', + enabled: true, + createdAt: new Date().toISOString() + }; + + this.saveConfig(); + log.info(`Added MCP server: ${name} (type: ${serverType})`); + return true; + } + + async removeServer(name) { + if (!this.config.mcpServers[name]) { + throw new Error(`MCP server '${name}' not found`); + } + + // Prevent deletion of Clara's core MCP server (python-mcp) + if (name === 'python-mcp') { + log.warn(`Attempted to delete Clara's core MCP server (${name}) - operation blocked`); + throw new Error("Clara's Python MCP server cannot be deleted as it is a system-required component."); + } + + // Stop the server if it's running + await this.stopServer(name); + + delete this.config.mcpServers[name]; + this.saveConfig(); + log.info(`Removed MCP server: ${name}`); + return true; + } + + async updateServer(name, updates) { + if (!this.config.mcpServers[name]) { + throw new Error(`MCP server '${name}' not found`); + } + + // Stop the server if it's running + const wasRunning = this.servers.has(name); + if (wasRunning) { + await this.stopServer(name); + } + + this.config.mcpServers[name] = { + ...this.config.mcpServers[name], + ...updates, + updatedAt: new Date().toISOString() + }; + + this.saveConfig(); + + // Restart if it was running + if (wasRunning && this.config.mcpServers[name].enabled) { + await this.startServer(name); + } + + log.info(`Updated MCP server: ${name}`); + return true; + } + + // Helper method to resolve bundled executable paths + resolveBundledExecutablePath(command) { + // Check if this is a bundled Python MCP server request + if (command === 'python-mcp-server') { + let executableName; + switch (os.platform()) { + case 'win32': + executableName = 'python-mcp-server-windows.exe'; + break; + case 'darwin': + // For macOS, choose based on architecture + if (os.arch() === 'arm64') { + executableName = 'python-mcp-server-mac-arm64'; + } else if (os.arch() === 'x64') { + executableName = 'python-mcp-server-mac-intel'; + } else { + // Fallback to universal binary + executableName = 'python-mcp-server-mac-universal'; + } + break; + case 'linux': + executableName = 'python-mcp-server-linux'; + break; + default: + throw new Error(`Unsupported platform: ${os.platform()}`); + } + + // Multiple paths to try (in order of preference) + const pathsToTry = []; + + // 1. Production: electron app resources + if (process.resourcesPath) { + pathsToTry.push(path.join(process.resourcesPath, 'electron', 'services', executableName)); + pathsToTry.push(path.join(process.resourcesPath, 'clara-mcp', executableName)); + } + + // 2. Development: relative to electron directory + pathsToTry.push(path.join(__dirname, 'services', executableName)); + pathsToTry.push(path.join(__dirname, '..', 'clara-mcp', executableName)); + + // 3. Development: relative to project root + const projectRoot = path.resolve(__dirname, '..'); + pathsToTry.push(path.join(projectRoot, 'clara-mcp', executableName)); + pathsToTry.push(path.join(projectRoot, 'electron', 'services', executableName)); + + // Check which path exists + for (const tryPath of pathsToTry) { + if (fs.existsSync(tryPath)) { + log.info(`Resolved bundled executable path: ${command} -> ${tryPath}`); + return tryPath; + } + } + + // If none found, log all tried paths for debugging + log.error(`MCP server binary not found. Tried paths: ${pathsToTry.join(', ')}`); + throw new Error(`MCP server binary not found. Tried ${pathsToTry.length} paths.`); + } + + // Return original command for non-bundled executables + return command; + } + + // Helper method to get enhanced PATH with common Node.js installation locations + getEnhancedPath() { + const currentPath = process.env.PATH || ''; + const homedir = os.homedir(); + + // check for nvm versions + let nvmNodePath = null; + try { + const versionsDir = path.join(homedir, '.nvm/versions/node'); + const versions = fs.readdirSync(versionsDir); + if (versions.length > 0) { + // Assuming the user is using the first installed version + nvmNodePath = path.join(versionsDir, versions[0], 'bin'); + } + } catch (err) { + // ignore if not found + } + + // Common Node.js installation paths + + const commonNodePaths = [ + '/usr/local/bin', + '/opt/homebrew/bin', + '/usr/bin', + nvmNodePath, + path.join(homedir, '.volta/bin'), + path.join(homedir, '.fnm/current/bin'), + path.join(homedir, 'n/bin'), + '/usr/local/node/bin', + '/opt/node/bin' + ]; + + // Filter existing paths and add them to PATH + const existingPaths = commonNodePaths.filter(nodePath => { + try { + return fs.existsSync(nodePath); + } catch (error) { + return false; + } + }); + + // Combine current PATH with existing Node.js paths + const allPaths = [currentPath, ...existingPaths].filter(Boolean); + return allPaths.join(path.delimiter); + } + + // Helper method to check if a command exists + async commandExists(command) { + return new Promise((resolve) => { + const testProcess = spawn(command, ['--version'], { + stdio: 'ignore', + shell: process.platform === 'win32', + env: { + ...process.env, + PATH: this.getEnhancedPath() + } + }); + + testProcess.on('error', () => resolve(false)); + testProcess.on('exit', (code) => resolve(code === 0)); + }); + } + + // Diagnose Node.js installation + async diagnoseNodeInstallation() { + const enhancedPath = this.getEnhancedPath(); + const pathDirs = enhancedPath.split(path.delimiter); + + const diagnosis = { + nodeAvailable: false, + npmAvailable: false, + npxAvailable: false, + nodePath: null, + npmPath: null, + npxPath: null, + pathDirs: pathDirs, + suggestions: [] + }; + + // Check for node, npm, and npx + for (const dir of pathDirs) { + if (!diagnosis.nodeAvailable) { + const nodePath = path.join(dir, process.platform === 'win32' ? 'node.exe' : 'node'); + if (fs.existsSync(nodePath)) { + diagnosis.nodeAvailable = true; + diagnosis.nodePath = nodePath; + } + } + + if (!diagnosis.npmAvailable) { + const npmPath = path.join(dir, process.platform === 'win32' ? 'npm.cmd' : 'npm'); + if (fs.existsSync(npmPath)) { + diagnosis.npmAvailable = true; + diagnosis.npmPath = npmPath; + } + } + + if (!diagnosis.npxAvailable) { + const npxPath = path.join(dir, process.platform === 'win32' ? 'npx.cmd' : 'npx'); + if (fs.existsSync(npxPath)) { + diagnosis.npxAvailable = true; + diagnosis.npxPath = npxPath; + } + } + } + + // Generate suggestions + if (!diagnosis.nodeAvailable) { + diagnosis.suggestions.push('Node.js is not installed or not found in PATH. Please install Node.js from https://nodejs.org/'); + } + + if (!diagnosis.npmAvailable) { + diagnosis.suggestions.push('npm is not available. It should come with Node.js installation.'); + } + + if (!diagnosis.npxAvailable) { + diagnosis.suggestions.push('npx is not available. It should come with npm 5.2.0 or later.'); + } + + if (diagnosis.nodeAvailable && diagnosis.npmAvailable && diagnosis.npxAvailable) { + diagnosis.suggestions.push('Node.js, npm, and npx are all available and should work correctly.'); + } + + return diagnosis; + } + + async startServer(name) { + const serverConfig = this.config.mcpServers[name]; + if (!serverConfig) { + throw new Error(`MCP server '${name}' not found`); + } + + if (this.servers.has(name)) { + throw new Error(`MCP server '${name}' is already running`); + } + + try { + // Handle remote servers differently + if (serverConfig.type === 'remote') { + log.info(`Connecting to remote MCP server: ${name} at ${serverConfig.url}`); + + // Test the connection to the remote server + const response = await fetch(serverConfig.url, { + method: 'GET', + headers: serverConfig.headers || {} + }); + + if (!response.ok) { + throw new Error(`Remote server not accessible: HTTP ${response.status}: ${response.statusText}`); + } + + const serverInfo = { + name, + config: serverConfig, + startedAt: new Date(), + status: 'running', + type: 'remote' + }; + + this.servers.set(name, serverInfo); + log.info(`Connected to remote MCP server: ${name}`); + return serverInfo; + } + + // Handle stdio servers (existing logic) + const { command, args = [], env = {} } = serverConfig; + + // Resolve bundled executable paths + const resolvedCommand = this.resolveBundledExecutablePath(command); + + // Check if command exists before trying to start (skip for bundled executables) + if ((command === 'npx' || command === 'npm' || command === 'node') && command === resolvedCommand) { + const commandAvailable = await this.commandExists(command); + if (!commandAvailable) { + throw new Error(`Command '${command}' not found. Please ensure Node.js and npm are properly installed and available in your PATH.`); + } + } + + // Merge environment variables with enhanced PATH + const processEnv = { + ...process.env, + PATH: this.getEnhancedPath(), + ...env, + // Add writable paths for Clara MCP workspace and SearXNG config + CLARA_MCP_WORKSPACE: path.join(app.getPath('userData'), 'mcp_workspace'), + CLARA_SEARXNG_CONFIG_DIR: path.join(app.getPath('userData'), 'searxng-config') + }; + + log.info(`Starting MCP server: ${name} with command: ${resolvedCommand} ${args.join(' ')}`); + log.info(`Using PATH: ${processEnv.PATH}`); + + const serverProcess = spawn(resolvedCommand, args, { + env: processEnv, + stdio: ['pipe', 'pipe', 'pipe'], + shell: process.platform === 'win32' + }); + + const serverInfo = { + process: serverProcess, + name, + config: serverConfig, + startedAt: new Date(), + status: 'starting', + type: 'stdio' + }; + + this.servers.set(name, serverInfo); + + // Handle process events + serverProcess.on('spawn', () => { + log.info(`MCP server '${name}' spawned successfully`); + serverInfo.status = 'running'; + serverInfo.initialized = false; // Reset initialization status for new process + }); + + serverProcess.on('error', (error) => { + log.error(`MCP server '${name}' error:`, error); + serverInfo.status = 'error'; + + // Provide more helpful error messages + if (error.code === 'ENOENT') { + serverInfo.error = `Command '${command}' not found. Please ensure Node.js and npm are properly installed.`; + } else { + serverInfo.error = error.message; + } + }); + + serverProcess.on('exit', (code, signal) => { + log.info(`MCP server '${name}' exited with code ${code}, signal ${signal}`); + this.servers.delete(name); + }); + + // Handle stdout/stderr + serverProcess.stdout.on('data', (data) => { + log.debug(`MCP server '${name}' stdout:`, data.toString()); + }); + + serverProcess.stderr.on('data', (data) => { + log.debug(`MCP server '${name}' stderr:`, data.toString()); + }); + + return serverInfo; + } catch (error) { + log.error(`Error starting MCP server '${name}':`, error); + throw error; + } + } + + async stopServer(name) { + const serverInfo = this.servers.get(name); + if (!serverInfo) { + return false; + } + + try { + log.info(`Stopping MCP server: ${name}`); + + // Handle remote servers differently + if (serverInfo.type === 'remote') { + this.servers.delete(name); + log.info(`Disconnected from remote MCP server: ${name}`); + return true; + } + + // Handle stdio servers (existing logic) + // Send SIGTERM first + serverInfo.process.kill('SIGTERM'); + + // Wait a bit, then force kill if needed + setTimeout(() => { + if (this.servers.has(name)) { + log.warn(`Force killing MCP server: ${name}`); + serverInfo.process.kill('SIGKILL'); + } + }, 5000); + + this.servers.delete(name); + return true; + } catch (error) { + log.error(`Error stopping MCP server '${name}':`, error); + throw error; + } + } + + async restartServer(name) { + await this.stopServer(name); + // Wait a bit before restarting + await new Promise(resolve => setTimeout(resolve, 1000)); + return await this.startServer(name); + } + + getServerStatus(name) { + const serverInfo = this.servers.get(name); + const config = this.config.mcpServers[name]; + + if (!config) { + return null; + } + + return { + name, + config, + isRunning: !!serverInfo, + status: serverInfo?.status || 'stopped', + startedAt: serverInfo?.startedAt, + error: serverInfo?.error, + pid: serverInfo?.process?.pid + }; + } + + getAllServers() { + // Ensure Clara's MCP server exists before returning server list (async, non-blocking) + this.ensureClaraMCPExists().catch(error => { + log.error('Failed to ensure Clara MCP server exists in getAllServers:', error); + }); + + const servers = []; + + for (const [name, config] of Object.entries(this.config.mcpServers)) { + const serverInfo = this.servers.get(name); + servers.push({ + name, + config, + isRunning: !!serverInfo, + status: serverInfo?.status || 'stopped', + startedAt: serverInfo?.startedAt, + error: serverInfo?.error, + pid: serverInfo?.process?.pid + }); + } + + return servers; + } + + async startAllEnabledServers() { + const results = []; + + for (const [name, config] of Object.entries(this.config.mcpServers)) { + if (config.enabled) { + try { + await this.startServer(name); + results.push({ name, success: true }); + } catch (error) { + log.error(`Failed to start MCP server '${name}':`, error); + results.push({ name, success: false, error: error.message }); + } + } + } + + return results; + } + + async stopAllServers() { + const results = []; + + for (const name of this.servers.keys()) { + try { + await this.stopServer(name); + results.push({ name, success: true }); + } catch (error) { + log.error(`Failed to stop MCP server '${name}':`, error); + results.push({ name, success: false, error: error.message }); + } + } + + return results; + } + + // Test server connection + async testServer(name) { + try { + const serverConfig = this.config.mcpServers[name]; + if (!serverConfig) { + throw new Error(`MCP server '${name}' not found`); + } + + // For remote servers, try to connect + if (serverConfig.type === 'remote') { + const response = await fetch(serverConfig.url, { + method: 'GET', + headers: serverConfig.headers || {} + }); + + if (!response.ok) { + throw new Error(`HTTP ${response.status}: ${response.statusText}`); + } + + return { success: true, message: 'Remote server is accessible' }; + } + + // For stdio servers, check if command exists + const { command } = serverConfig; + const resolvedCommand = this.resolveBundledExecutablePath(command); + + // For bundled executables, check file existence instead of version command + if (command !== resolvedCommand) { + // This is a bundled executable, check if file exists + if (fs.existsSync(resolvedCommand)) { + return { success: true, message: 'Bundled executable is available' }; + } else { + return { success: false, error: `Bundled executable not found at: ${resolvedCommand}` }; + } + } + + return new Promise((resolve) => { + const testProcess = spawn(resolvedCommand, ['--version'], { + stdio: 'ignore', + shell: process.platform === 'win32', + env: { + ...process.env, + PATH: this.getEnhancedPath() + } + }); + + testProcess.on('error', (error) => { + if (error.code === 'ENOENT') { + resolve({ success: false, error: `Command '${resolvedCommand}' not found. Please ensure Node.js and npm are properly installed.` }); + } else { + resolve({ success: false, error: error.message }); + } + }); + + testProcess.on('exit', (code) => { + if (code === 0) { + resolve({ success: true, message: 'Command is available' }); + } else { + resolve({ success: false, error: `Command exited with code ${code}` }); + } + }); + }); + } catch (error) { + return { success: false, error: error.message }; + } + } + + // Get available MCP server templates + getServerTemplates() { + return [ + // System & File Management + { + name: 'filesystem', + displayName: 'File System', + description: 'Access and manipulate files and directories', + command: 'npx', + args: ['@modelcontextprotocol/server-filesystem', '/path/to/directory'], + type: 'stdio', + category: 'System', + icon: 'fas fa-folder', + popularity: 'high' + }, + { + name: 'hyper-shell', + displayName: 'Shell Access', + description: 'Secure shell and OS-level command execution', + command: 'npx', + args: ['hyper-mcp-shell'], + type: 'stdio', + category: 'System', + icon: 'fas fa-terminal', + popularity: 'medium' + }, + + // Development & Git + { + name: 'git', + displayName: 'Git Repository', + description: 'Git repository operations and history', + command: 'npx', + args: ['@modelcontextprotocol/server-git', '/path/to/repo'], + type: 'stdio', + category: 'Development', + icon: 'fab fa-git-alt', + popularity: 'high' + }, + { + name: 'github', + displayName: 'GitHub', + description: 'GitHub repository and issue management', + command: 'npx', + args: ['@modelcontextprotocol/server-github'], + type: 'stdio', + category: 'Development', + icon: 'fab fa-github', + popularity: 'high', + env: { + GITHUB_PERSONAL_ACCESS_TOKEN: 'your-github-token' + } + }, + { + name: 'gitlab', + displayName: 'GitLab', + description: 'GitLab project and pipeline management', + command: 'npx', + args: ['@modelcontextprotocol/server-gitlab'], + type: 'stdio', + category: 'Development', + icon: 'fab fa-gitlab', + popularity: 'medium', + env: { + GITLAB_PERSONAL_ACCESS_TOKEN: 'your-gitlab-token', + GITLAB_URL: 'https://gitlab.com' + } + }, + + // Databases + { + name: 'sqlite', + displayName: 'SQLite Database', + description: 'Query and manipulate SQLite databases', + command: 'npx', + args: ['@modelcontextprotocol/server-sqlite', '/path/to/database.db'], + type: 'stdio', + category: 'Database', + icon: 'fas fa-database', + popularity: 'high' + }, + { + name: 'postgres', + displayName: 'PostgreSQL Database', + description: 'Connect to PostgreSQL databases', + command: 'npx', + args: ['@modelcontextprotocol/server-postgres'], + type: 'stdio', + category: 'Database', + icon: 'fas fa-server', + popularity: 'high', + env: { + POSTGRES_CONNECTION_STRING: 'postgresql://user:password@localhost:5432/dbname' + } + }, + { + name: 'mysql', + displayName: 'MySQL Database', + description: 'Connect to MySQL/MariaDB databases', + command: 'npx', + args: ['@modelcontextprotocol/server-mysql'], + type: 'stdio', + category: 'Database', + icon: 'fas fa-hdd', + popularity: 'medium', + env: { + MYSQL_CONNECTION_STRING: 'mysql://user:password@localhost:3306/dbname' + } + }, + + // Web & APIs + { + name: 'puppeteer', + displayName: 'Web Scraping', + description: 'Web scraping and browser automation', + command: 'npx', + args: ['@modelcontextprotocol/server-puppeteer'], + type: 'stdio', + category: 'Web', + icon: 'fas fa-spider', + popularity: 'high' + }, + { + name: 'playwright', + displayName: 'Playwright Automation', + description: 'Cross-browser automation and testing', + command: 'npx', + args: ['playwright-mcp-server'], + type: 'stdio', + category: 'Web', + icon: 'fas fa-theater-masks', + popularity: 'medium' + }, + { + name: 'fetch', + displayName: 'HTTP Requests', + description: 'Make HTTP requests and API calls', + command: 'npx', + args: ['@modelcontextprotocol/server-fetch'], + type: 'stdio', + category: 'Web', + icon: 'fas fa-globe', + popularity: 'high' + }, + + // Search Engines + { + name: 'brave-search', + displayName: 'Brave Search', + description: 'Search the web using Brave Search API', + command: 'npx', + args: ['@modelcontextprotocol/server-brave-search'], + type: 'stdio', + category: 'Search', + icon: 'fas fa-shield-alt', + popularity: 'high', + env: { + BRAVE_API_KEY: 'your-brave-api-key' + } + }, + { + name: 'searxng', + displayName: 'SearxNG Search', + description: 'Privacy-focused meta-search through SearxNG', + command: 'npx', + args: ['mcp-searxng'], + type: 'stdio', + category: 'Search', + icon: 'fas fa-search', + popularity: 'medium', + env: { + SEARXNG_URL: 'http://localhost:8080' + } + }, + { + name: 'google-search', + displayName: 'Google Search', + description: 'Search Google with custom search engine', + command: 'npx', + args: ['@modelcontextprotocol/server-google-search'], + type: 'stdio', + category: 'Search', + icon: 'fab fa-google', + popularity: 'high', + env: { + GOOGLE_API_KEY: 'your-google-api-key', + GOOGLE_SEARCH_ENGINE_ID: 'your-search-engine-id' + } + }, + + // AI & Reasoning + { + name: 'memory', + displayName: 'Persistent Memory', + description: 'Knowledge-graph based long-term memory store', + command: 'npx', + args: ['@modelcontextprotocol/server-memory'], + type: 'stdio', + category: 'AI', + icon: 'fas fa-brain', + popularity: 'high' + }, + { + name: 'sequential-thinking', + displayName: 'Sequential Thinking', + description: 'Structured multi-step reasoning tools', + command: 'npx', + args: ['@modelcontextprotocol/server-sequential-thinking'], + type: 'stdio', + category: 'AI', + icon: 'fas fa-sitemap', + popularity: 'medium' + }, + + // Communication & Collaboration + { + name: 'slack', + displayName: 'Slack', + description: 'Slack workspace integration', + command: 'npx', + args: ['@modelcontextprotocol/server-slack'], + type: 'stdio', + category: 'Communication', + icon: 'fab fa-slack', + popularity: 'high', + env: { + SLACK_BOT_TOKEN: 'your-slack-bot-token' + } + }, + { + name: 'discord', + displayName: 'Discord', + description: 'Discord server and message management', + command: 'npx', + args: ['discord-mcp-server'], + type: 'stdio', + category: 'Communication', + icon: 'fab fa-discord', + popularity: 'medium', + env: { + DISCORD_BOT_TOKEN: 'your-discord-bot-token' + } + }, + { + name: 'notion', + displayName: 'Notion', + description: 'Notion workspace and page management', + command: 'npx', + args: ['@modelcontextprotocol/server-notion'], + type: 'stdio', + category: 'Communication', + icon: 'fas fa-sticky-note', + popularity: 'high', + env: { + NOTION_API_KEY: 'your-notion-api-key' + } + }, + + // Remote & Custom + { + name: 'remote-server', + displayName: 'Remote MCP Server', + description: 'Connect to a remote MCP server via HTTP', + type: 'remote', + url: 'http://localhost:3000/mcp', + headers: {}, + category: 'Remote', + icon: 'fas fa-network-wired', + popularity: 'low' + }, + + // Additional Popular Servers (No API Keys Required) + { + name: 'calculator', + displayName: 'Calculator', + description: 'Precise numerical calculations and math operations', + command: 'npx', + args: ['mcp-server-calculator'], + type: 'stdio', + category: 'Utilities', + icon: 'fas fa-calculator', + popularity: 'high' + }, + { + name: 'redis', + displayName: 'Redis Cache', + description: 'In-memory caching and key-value operations', + command: 'npx', + args: ['mcp-server-redis'], + type: 'stdio', + category: 'Database', + icon: 'fas fa-memory', + popularity: 'high', + env: { + REDIS_URL: 'redis://localhost:6379' + } + }, + { + name: 'docker', + displayName: 'Docker Manager', + description: 'Manage containers, images, volumes, and networks', + command: 'npx', + args: ['mcp-server-docker'], + type: 'stdio', + category: 'Development', + icon: 'fab fa-docker', + popularity: 'high' + }, + { + name: 'kubernetes', + displayName: 'Kubernetes', + description: 'Connect to Kubernetes cluster and manage resources', + command: 'npx', + args: ['mcp-server-kubernetes'], + type: 'stdio', + category: 'Development', + icon: 'fas fa-dharmachakra', + popularity: 'high', + env: { + KUBECONFIG: '~/.kube/config' + } + }, + { + name: 'pandoc', + displayName: 'Document Converter', + description: 'Convert between various document formats using Pandoc', + command: 'npx', + args: ['mcp-pandoc'], + type: 'stdio', + category: 'Productivity', + icon: 'fas fa-file-export', + popularity: 'high' + }, + { + name: 'everything-search', + displayName: 'Everything Search', + description: 'Fast file searching capabilities across Windows/macOS/Linux', + command: 'npx', + args: ['mcp-everything-search'], + type: 'stdio', + category: 'Utilities', + icon: 'fas fa-search-plus', + popularity: 'high' + }, + { + name: 'obsidian', + displayName: 'Obsidian Notes', + description: 'Read and search through your Obsidian vault notes', + command: 'npx', + args: ['mcp-obsidian'], + type: 'stdio', + category: 'Productivity', + icon: 'fas fa-sticky-note', + popularity: 'high', + env: { + OBSIDIAN_VAULT_PATH: '/path/to/obsidian/vault' + } + }, + { + name: 'open-meteo', + displayName: 'Weather Data', + description: 'Weather forecasts and climate data (free API)', + command: 'npx', + args: ['mcp-weather'], + type: 'stdio', + category: 'Utilities', + icon: 'fas fa-cloud-sun', + popularity: 'high' + }, + { + name: 'wikipedia', + displayName: 'Wikipedia', + description: 'Access and search Wikipedia articles', + command: 'npx', + args: ['wikipedia-mcp'], + type: 'stdio', + category: 'Knowledge', + icon: 'fab fa-wikipedia-w', + popularity: 'high' + }, + { + name: 'shell-commands', + displayName: 'Shell Commands', + description: 'Run shell commands and scripts securely', + command: 'npx', + args: ['mcp-server-commands'], + type: 'stdio', + category: 'System', + icon: 'fas fa-terminal', + popularity: 'high' + }, + { + name: 'ssh', + displayName: 'SSH Remote Access', + description: 'Execute SSH commands remotely and transfer files', + command: 'npx', + args: ['ssh-mcp-server'], + type: 'stdio', + category: 'System', + icon: 'fas fa-server', + popularity: 'medium', + env: { + SSH_HOST: 'hostname', + SSH_USER: 'username', + SSH_PRIVATE_KEY_PATH: '/path/to/private/key' + } + }, + { + name: 'json-tools', + displayName: 'JSON Tools', + description: 'JSON handling and processing with JSONPath queries', + command: 'npx', + args: ['json-mcp-server'], + type: 'stdio', + category: 'Utilities', + icon: 'fas fa-code', + popularity: 'medium' + }, + { + name: 'pdf-tools', + displayName: 'PDF Tools', + description: 'Read, search, and manipulate PDF files', + command: 'npx', + args: ['mcp-pdf-reader'], + type: 'stdio', + category: 'Productivity', + icon: 'fas fa-file-pdf', + popularity: 'medium' + }, + { + name: 'csv-editor', + displayName: 'CSV Editor', + description: 'Comprehensive CSV processing and data manipulation', + command: 'npx', + args: ['csv-editor'], + type: 'stdio', + category: 'Productivity', + icon: 'fas fa-table', + popularity: 'medium' + }, + { + name: 'time-utils', + displayName: 'Date/Time Utils', + description: 'Date and time operations and calculations', + command: 'npx', + args: ['mcp-datetime'], + type: 'stdio', + category: 'Utilities', + icon: 'fas fa-clock', + popularity: 'medium' + }, + { + name: 'markdown-docs', + displayName: 'Markdown Docs', + description: 'Access and manage local documentation files', + command: 'npx', + args: ['docs-mcp'], + type: 'stdio', + category: 'Productivity', + icon: 'fab fa-markdown', + popularity: 'medium', + env: { + DOCS_PATH: '/path/to/docs' + } + }, + { + name: 'code-analysis', + displayName: 'Code Analysis', + description: 'Code context provider and analysis tools', + command: 'npx', + args: ['code-context-provider-mcp'], + type: 'stdio', + category: 'Development', + icon: 'fas fa-code-branch', + popularity: 'medium' + }, + { + name: 'fast-filesystem', + displayName: 'Fast Filesystem', + description: 'Advanced filesystem operations with large file handling', + command: 'npx', + args: ['fast-filesystem-mcp'], + type: 'stdio', + category: 'System', + icon: 'fas fa-folder-open', + popularity: 'medium' + }, + { + name: 'basic-memory', + displayName: 'Basic Memory', + description: 'Local-first knowledge management with semantic search', + command: 'npx', + args: ['basic-memory'], + type: 'stdio', + category: 'AI', + icon: 'fas fa-brain', + popularity: 'medium' + }, + { + name: 'random-number', + displayName: 'Random Generator', + description: 'Random number and data generation utilities', + command: 'npx', + args: ['random-number-mcp'], + type: 'stdio', + category: 'Utilities', + icon: 'fas fa-dice', + popularity: 'low' + }, + { + name: 'browsermcp', + displayName: 'Browser MCP', + description: 'Control browser actions and automation - needs https://browsermcp.io/install extension', + command: 'npx', + args: ['@browsermcp/mcp@latest'], + type: 'stdio', + category: 'Browser', + icon: 'fas fa-chrome', + popularity: 'high' + }, + { + name: 'terminator-mcp-agent', + displayName: 'Terminator MCP Agent', + description: 'Control your PC directly from LLM', + command: 'npx', + args: ['-y', 'terminator-mcp-agent@latest'], + type: 'stdio', + category: 'System', + icon: 'fas fa-terminal', + popularity: 'high' + }, + { + name: 'blender', + displayName: 'Blender MCP', + description: 'Control and manage Blender instances - needs uvx and bit of a setup (https://github.com/ahujasid/blender-mcp)', + command: 'uvx', + args: ['blender-mcp'], + type: 'stdio', + category: '3D', + icon: 'fas fa-cube', + popularity: 'medium' + }, + { + name: 'desktop-commander', + displayName: 'Desktop Commander', + description: 'All-in-one development tool for managing your desktop environment', + command: 'npx', + args: ['-y', '@wonderwhy-er/desktop-commander@latest'], + type: 'stdio', + category: 'Productivity', + icon: 'fas fa-desktop', + popularity: 'high' + }, + { + name: 'mcp-docker-portal', + displayName: 'MCP Docker Portal', + description: 'Manage MCP instances in Docker Desktop', + command: 'docker', + args: [ + 'mcp', + 'gateway', + 'run' + ], + type: 'stdio', + category: 'System', + icon: 'fas fa-docker', + popularity: 'high' + } + ]; + } + + // Import servers from Claude Desktop config + async importFromClaudeConfig(claudeConfigPath) { + try { + if (!fs.existsSync(claudeConfigPath)) { + throw new Error('Claude config file not found'); + } + + const claudeConfig = JSON.parse(fs.readFileSync(claudeConfigPath, 'utf8')); + const mcpServers = claudeConfig.mcpServers || {}; + + let imported = 0; + const errors = []; + + for (const [name, config] of Object.entries(mcpServers)) { + try { + if (!this.config.mcpServers[name]) { + this.config.mcpServers[name] = { + type: 'stdio', + command: config.command, + args: config.args || [], + env: config.env || {}, + description: `Imported from Claude Desktop`, + enabled: true, + createdAt: new Date().toISOString() + }; + imported++; + } + } catch (error) { + errors.push({ name, error: error.message }); + } + } + + if (imported > 0) { + this.saveConfig(); + } + + return { imported, errors }; + } catch (error) { + throw new Error(`Failed to import Claude config: ${error.message}`); + } + } + + // Execute MCP tool call + async executeToolCall(toolCall) { + try { + const { server: serverName, name: toolName, arguments: args, callId } = toolCall; + + // Get the server info + const serverInfo = this.servers.get(serverName); + if (!serverInfo || serverInfo.status !== 'running') { + return { + callId, + success: false, + error: `Server ${serverName} is not running` + }; + } + + // Handle special MCP protocol methods + if (toolName === 'tools/list') { + return await this.listToolsFromServer(serverName, callId); + } + + // Handle remote servers differently + if (serverInfo.type === 'remote') { + return await this.executeRemoteToolCall(serverInfo, toolName, args, callId); + } + + // **CRITICAL FIX: Ensure MCP server is properly initialized before executing tools** + if (!serverInfo.initialized) { + log.info(`[${serverName}] Server not initialized, performing MCP handshake first...`); + try { + await this.initializeMCPServer(serverName); + log.info(`[${serverName}] MCP initialization completed successfully`); + } catch (initError) { + log.error(`[${serverName}] MCP initialization failed:`, initError); + return { + callId, + success: false, + error: `MCP initialization failed: ${initError.message}` + }; + } + } + + // For stdio servers, use the existing implementation + const mcpRequest = { + jsonrpc: '2.0', + id: callId, + method: 'tools/call', + params: { + name: toolName, + arguments: args + } + }; + + // Send the request to the MCP server via stdin + const requestString = JSON.stringify(mcpRequest) + '\n'; + + log.info(`[${serverName}] Sending MCP request:`, mcpRequest); + log.info(`[${serverName}] Request string:`, requestString); + + return new Promise((resolve) => { + let responseData = ''; + let timeoutId; + + // Set up response handler + const onData = (data) => { + responseData += data.toString(); + + // Log raw response data for debugging + log.info(`[${serverName}] Raw response data:`, data.toString()); + log.info(`[${serverName}] Accumulated responseData:`, responseData); + + // Try to parse JSON response + try { + const lines = responseData.split('\n').filter(line => line.trim()); + log.info(`[${serverName}] Split into ${lines.length} lines:`, lines); + + for (const line of lines) { + // Only try to parse lines that look like JSON (start with { or [) + const trimmedLine = line.trim(); + log.info(`[${serverName}] Processing line:`, trimmedLine); + + if (!trimmedLine.startsWith('{') && !trimmedLine.startsWith('[')) { + log.info(`[${serverName}] Skipping non-JSON line:`, trimmedLine); + continue; + } + + log.info(`[${serverName}] Attempting to parse JSON line:`, trimmedLine); + + try { + const response = JSON.parse(trimmedLine); + log.info(`[${serverName}] Successfully parsed JSON:`, response); + + if (response.id === callId) { + log.info(`[${serverName}] Found matching response for callId:`, callId); + // Clean up + clearTimeout(timeoutId); + serverInfo.process.stdout.off('data', onData); + + if (response.error) { + log.error(`[${serverName}] MCP server returned error:`, response.error); + resolve({ + callId, + success: false, + error: response.error.message || 'MCP tool execution failed' + }); + } else { + log.info(`[${serverName}] MCP server returned success:`, response.result); + resolve({ + callId, + success: true, + content: response.result?.content || [{ type: 'text', text: JSON.stringify(response.result) }], + metadata: { + server: serverName, + tool: toolName, + executedAt: new Date().toISOString() + } + }); + } + return; + } else { + log.info(`[${serverName}] Response ID ${response.id} doesn't match expected ${callId}`); + } + } catch (lineParseError) { + // Skip malformed lines and continue + log.error(`[${serverName}] JSON parse error for line:`, trimmedLine, 'Error:', lineParseError.message); + continue; + } + } + } catch (parseError) { + // Log parsing errors but continue waiting for more data + log.error(`[${serverName}] Overall JSON parsing error:`, parseError.message, 'ResponseData:', responseData); + } + }; + + // Set up timeout + timeoutId = setTimeout(() => { + log.error(`[${serverName}] MCP tool execution timeout after 60 seconds`); + log.error(`[${serverName}] Final responseData:`, responseData); + serverInfo.process.stdout.off('data', onData); + resolve({ + callId, + success: false, + error: 'MCP tool execution timeout' + }); + }, 60000); // 60 second timeout + + // Listen for response + serverInfo.process.stdout.on('data', onData); + + // Send the request + try { + log.info(`[${serverName}] Writing request to stdin...`); + serverInfo.process.stdin.write(requestString); + log.info(`[${serverName}] Request sent successfully`); + } catch (writeError) { + log.error(`[${serverName}] Failed to write request:`, writeError); + clearTimeout(timeoutId); + serverInfo.process.stdout.off('data', onData); + resolve({ + callId, + success: false, + error: `Failed to send request to MCP server: ${writeError.message}` + }); + } + }); + + } catch (error) { + log.error(`Error executing MCP tool call:`, error); + return { + callId: toolCall.callId, + success: false, + error: error.message || 'Unknown error occurred' + }; + } + } + + // Initialize MCP server with proper handshake + async initializeMCPServer(serverName) { + const serverInfo = this.servers.get(serverName); + if (!serverInfo || serverInfo.status !== 'running') { + throw new Error(`Server ${serverName} is not running`); + } + + // Skip initialization for remote servers or if already initialized + if (serverInfo.type === 'remote' || serverInfo.initialized) { + return true; + } + + log.info(`[${serverName}] Starting MCP initialization handshake...`); + + return new Promise((resolve, reject) => { + let initResponseReceived = false; + const initCallId = `init-${Date.now()}`; + + // Step 1: Send initialize request + const initRequest = { + jsonrpc: '2.0', + id: initCallId, + method: 'initialize', + params: { + protocolVersion: '2024-11-05', + capabilities: { + tools: {}, + resources: {} + }, + clientInfo: { + name: 'ClaraVerse', + version: '1.0.0' + } + } + }; + + const timeout = setTimeout(() => { + if (!initResponseReceived) { + serverInfo.process.stdout.off('data', onInitData); + reject(new Error(`MCP initialization timeout for ${serverName}`)); + } + }, 10000); // 10 second timeout for initialization + + const onInitData = (data) => { + const responseData = data.toString(); + log.info(`[${serverName}] Init response data:`, responseData); + + try { + const lines = responseData.split('\n').filter(line => line.trim()); + for (const line of lines) { + const trimmedLine = line.trim(); + if (!trimmedLine.startsWith('{')) continue; + + try { + const response = JSON.parse(trimmedLine); + if (response.id === initCallId) { + clearTimeout(timeout); + serverInfo.process.stdout.off('data', onInitData); + initResponseReceived = true; + + if (response.error) { + log.error(`[${serverName}] MCP initialization failed:`, response.error); + reject(new Error(`MCP initialization failed: ${response.error.message}`)); + return; + } + + log.info(`[${serverName}] MCP initialization successful:`, response.result); + + // Step 2: Send initialized notification + const initializedNotification = { + jsonrpc: '2.0', + method: 'notifications/initialized', + params: {} + }; + + try { + serverInfo.process.stdin.write(JSON.stringify(initializedNotification) + '\n'); + log.info(`[${serverName}] Sent initialized notification`); + + // Mark as initialized + serverInfo.initialized = true; + resolve(true); + } catch (notifyError) { + log.error(`[${serverName}] Failed to send initialized notification:`, notifyError); + reject(notifyError); + } + return; + } + } catch (lineParseError) { + log.debug(`[${serverName}] Skipping non-JSON line during init:`, trimmedLine); + continue; + } + } + } catch (parseError) { + log.debug(`[${serverName}] Parse error during initialization:`, parseError.message); + } + }; + + // Listen for initialization response + serverInfo.process.stdout.on('data', onInitData); + + // Send initialization request + try { + const requestString = JSON.stringify(initRequest) + '\n'; + log.info(`[${serverName}] Sending MCP initialize request:`, initRequest); + serverInfo.process.stdin.write(requestString); + } catch (writeError) { + clearTimeout(timeout); + serverInfo.process.stdout.off('data', onInitData); + reject(new Error(`Failed to send initialize request: ${writeError.message}`)); + } + }); + } + + // List tools from an MCP server + async listToolsFromServer(serverName, callId) { + try { + const serverInfo = this.servers.get(serverName); + if (!serverInfo || serverInfo.status !== 'running') { + return { + callId, + success: false, + error: `Server ${serverName} is not running` + }; + } + + // Handle remote servers differently + if (serverInfo.type === 'remote') { + const mcpRequest = { + jsonrpc: '2.0', + id: callId, + method: 'tools/list', + params: {} + }; + + log.info(`[${serverName}] Sending remote tools/list request:`, mcpRequest); + + const response = await fetch(serverInfo.config.url, { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + ...serverInfo.config.headers + }, + body: JSON.stringify(mcpRequest) + }); + + if (!response.ok) { + throw new Error(`HTTP ${response.status}: ${response.statusText}`); + } + + const responseData = await response.json(); + log.info(`[${serverName}] Remote tools/list response:`, responseData); + + if (responseData.error) { + return { + callId, + success: false, + error: responseData.error.message || 'Failed to list tools' + }; + } + + const tools = responseData.result?.tools || []; + return { + callId, + success: true, + content: [{ + type: 'json', + text: JSON.stringify(tools), + data: tools + }], + metadata: { + server: serverName, + tool: 'tools/list', + executedAt: new Date().toISOString(), + type: 'remote' + } + }; + } + + // **CRITICAL FIX: Ensure MCP server is properly initialized before sending tools/list** + if (!serverInfo.initialized) { + log.info(`[${serverName}] Server not initialized, performing MCP handshake first...`); + try { + await this.initializeMCPServer(serverName); + log.info(`[${serverName}] MCP initialization completed successfully`); + } catch (initError) { + log.error(`[${serverName}] MCP initialization failed:`, initError); + return { + callId, + success: false, + error: `MCP initialization failed: ${initError.message}` + }; + } + } + + // Handle stdio servers (existing logic) + const mcpRequest = { + jsonrpc: '2.0', + id: callId, + method: 'tools/list', + params: {} + }; + + const requestString = JSON.stringify(mcpRequest) + '\n'; + + return new Promise((resolve) => { + let responseData = ''; + let timeoutId; + + const onData = (data) => { + responseData += data.toString(); + + try { + const lines = responseData.split('\n').filter(line => line.trim()); + for (const line of lines) { + // Only try to parse lines that look like JSON (start with { or [) + const trimmedLine = line.trim(); + if (!trimmedLine.startsWith('{') && !trimmedLine.startsWith('[')) { + log.debug(`Skipping non-JSON line from ${serverName}:`, trimmedLine); + continue; + } + + try { + const response = JSON.parse(trimmedLine); + if (response.id === callId) { + clearTimeout(timeoutId); + serverInfo.process.stdout.off('data', onData); + + if (response.error) { + resolve({ + callId, + success: false, + error: response.error.message || 'Failed to list tools' + }); + } else { + // Return the tools list in a format the frontend expects + const tools = response.result?.tools || []; + resolve({ + callId, + success: true, + content: [{ + type: 'json', + text: JSON.stringify(tools), + data: tools + }], + metadata: { + server: serverName, + tool: 'tools/list', + executedAt: new Date().toISOString(), + type: 'stdio' + } + }); + } + return; + } + } catch (lineParseError) { + log.debug(`Skipping malformed JSON line from ${serverName}:`, trimmedLine); + continue; + } + } + } catch (parseError) { + log.debug(`JSON parsing error for ${serverName}:`, parseError.message); + } + }; + + timeoutId = setTimeout(() => { + serverInfo.process.stdout.off('data', onData); + resolve({ + callId, + success: false, + error: 'Tool listing timeout (waited 60s)' + }); + }, 60000); // 60 second timeout for all MCP servers + + serverInfo.process.stdout.on('data', onData); + + try { + serverInfo.process.stdin.write(requestString); + } catch (writeError) { + clearTimeout(timeoutId); + serverInfo.process.stdout.off('data', onData); + resolve({ + callId, + success: false, + error: `Failed to send tools/list request: ${writeError.message}` + }); + } + }); + + } catch (error) { + log.error(`Error listing tools from MCP server ${serverName}:`, error); + return { + callId, + success: false, + error: error.message || 'Unknown error occurred' + }; + } + } + + // Execute tool call on remote MCP server + async executeRemoteToolCall(serverInfo, toolName, args, callId) { + try { + const { config } = serverInfo; + const mcpRequest = { + jsonrpc: '2.0', + id: callId, + method: 'tools/call', + params: { + name: toolName, + arguments: args + } + }; + + log.info(`[${serverInfo.name}] Sending remote MCP request:`, mcpRequest); + + const response = await fetch(config.url, { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + ...config.headers + }, + body: JSON.stringify(mcpRequest) + }); + + if (!response.ok) { + throw new Error(`HTTP ${response.status}: ${response.statusText}`); + } + + const responseData = await response.json(); + log.info(`[${serverInfo.name}] Remote MCP response:`, responseData); + + if (responseData.error) { + return { + callId, + success: false, + error: responseData.error.message || 'Remote MCP tool execution failed' + }; + } + + return { + callId, + success: true, + content: responseData.result?.content || [{ type: 'text', text: JSON.stringify(responseData.result) }], + metadata: { + server: serverInfo.name, + tool: toolName, + executedAt: new Date().toISOString(), + type: 'remote' + } + }; + + } catch (error) { + log.error(`Error executing remote MCP tool call:`, error); + return { + callId, + success: false, + error: error.message || 'Remote MCP tool execution failed' + }; + } + } + + /** + * Save the current running state of all servers + */ + saveRunningState() { + try { + this.config.lastRunningServers = Array.from(this.servers.keys()); + this.saveConfig(); + log.info(`Saved running state: ${this.config.lastRunningServers.length} servers were running`); + } catch (error) { + log.error('Error saving running state:', error); + } + } + + /** + * Start all servers that were running when the app was last closed + */ + async startPreviouslyRunningServers() { + const results = []; + const previouslyRunning = this.config.lastRunningServers || []; + + log.info(`Attempting to restore ${previouslyRunning.length} previously running servers:`, previouslyRunning); + + for (const serverName of previouslyRunning) { + // Check if the server still exists in config + if (!this.config.mcpServers[serverName]) { + log.warn(`Previously running server '${serverName}' no longer exists in config`); + continue; + } + + // Check if the server is enabled + if (!this.config.mcpServers[serverName].enabled) { + log.info(`Previously running server '${serverName}' is now disabled, skipping`); + continue; + } + + try { + await this.startServer(serverName); + results.push({ name: serverName, success: true }); + log.info(`Successfully restored server: ${serverName}`); + } catch (error) { + log.error(`Failed to restore MCP server '${serverName}':`, error); + results.push({ name: serverName, success: false, error: error.message }); + } + } + + log.info(`Restored ${results.filter(r => r.success).length}/${previouslyRunning.length} previously running servers`); + return results; + } +} + +module.exports = MCPService; \ No newline at end of file diff --git a/electron/mcpService.cjs.backup b/electron/mcpService.cjs.backup new file mode 100644 index 00000000..0cc2ab52 --- /dev/null +++ b/electron/mcpService.cjs.backup @@ -0,0 +1,692 @@ +const { spawn } = require('child_process'); +const path = require('path'); +const fs = require('fs'); +const log = require('electron-log'); +const { app } = require('electron'); + +class MCPService { + constructor() { + this.servers = new Map(); + this.configPath = path.join(app.getPath('userData'), 'mcp_config.json'); + this.loadConfig(); + } + + loadConfig() { + try { + if (fs.existsSync(this.configPath)) { + const configData = fs.readFileSync(this.configPath, 'utf8'); + this.config = JSON.parse(configData); + } else { + this.config = { + mcpServers: {} + }; + this.saveConfig(); + } + } catch (error) { + log.error('Error loading MCP config:', error); + this.config = { + mcpServers: {} + }; + } + } + + saveConfig() { + try { + fs.writeFileSync(this.configPath, JSON.stringify(this.config, null, 2)); + } catch (error) { + log.error('Error saving MCP config:', error); + } + } + + async addServer(serverConfig) { + const { name, type, command, args, env, description } = serverConfig; + + if (this.config.mcpServers[name]) { + throw new Error(`MCP server '${name}' already exists`); + } + + this.config.mcpServers[name] = { + type: type || 'stdio', + command, + args: args || [], + env: env || {}, + description: description || '', + enabled: true, + createdAt: new Date().toISOString() + }; + + this.saveConfig(); + log.info(`Added MCP server: ${name}`); + return true; + } + + async removeServer(name) { + if (!this.config.mcpServers[name]) { + throw new Error(`MCP server '${name}' not found`); + } + + // Stop the server if it's running + await this.stopServer(name); + + delete this.config.mcpServers[name]; + this.saveConfig(); + log.info(`Removed MCP server: ${name}`); + return true; + } + + async updateServer(name, updates) { + if (!this.config.mcpServers[name]) { + throw new Error(`MCP server '${name}' not found`); + } + + // Stop the server if it's running + const wasRunning = this.servers.has(name); + if (wasRunning) { + await this.stopServer(name); + } + + this.config.mcpServers[name] = { + ...this.config.mcpServers[name], + ...updates, + updatedAt: new Date().toISOString() + }; + + this.saveConfig(); + + // Restart if it was running + if (wasRunning && this.config.mcpServers[name].enabled) { + await this.startServer(name); + } + + log.info(`Updated MCP server: ${name}`); + return true; + } + + async startServer(name) { + const serverConfig = this.config.mcpServers[name]; + if (!serverConfig) { + throw new Error(`MCP server '${name}' not found`); + } + + if (this.servers.has(name)) { + throw new Error(`MCP server '${name}' is already running`); + } + + try { + const { command, args = [], env = {} } = serverConfig; + + // Merge environment variables + const processEnv = { + ...process.env, + ...env + }; + + log.info(`Starting MCP server: ${name} with command: ${command} ${args.join(' ')}`); + + const serverProcess = spawn(command, args, { + env: processEnv, + stdio: ['pipe', 'pipe', 'pipe'], + shell: process.platform === 'win32' + }); + + const serverInfo = { + process: serverProcess, + name, + config: serverConfig, + startedAt: new Date(), + status: 'starting' + }; + + this.servers.set(name, serverInfo); + + // Handle process events + serverProcess.on('spawn', () => { + log.info(`MCP server '${name}' spawned successfully`); + serverInfo.status = 'running'; + }); + + serverProcess.on('error', (error) => { + log.error(`MCP server '${name}' error:`, error); + serverInfo.status = 'error'; + serverInfo.error = error.message; + }); + + serverProcess.on('exit', (code, signal) => { + log.info(`MCP server '${name}' exited with code ${code}, signal ${signal}`); + this.servers.delete(name); + }); + + // Handle stdout/stderr + serverProcess.stdout.on('data', (data) => { + log.debug(`MCP server '${name}' stdout:`, data.toString()); + }); + + serverProcess.stderr.on('data', (data) => { + log.debug(`MCP server '${name}' stderr:`, data.toString()); + }); + + return serverInfo; + } catch (error) { + log.error(`Error starting MCP server '${name}':`, error); + throw error; + } + } + + async stopServer(name) { + const serverInfo = this.servers.get(name); + if (!serverInfo) { + return false; + } + + try { + log.info(`Stopping MCP server: ${name}`); + + // Send SIGTERM first + serverInfo.process.kill('SIGTERM'); + + // Wait a bit, then force kill if needed + setTimeout(() => { + if (this.servers.has(name)) { + log.warn(`Force killing MCP server: ${name}`); + serverInfo.process.kill('SIGKILL'); + } + }, 5000); + + this.servers.delete(name); + return true; + } catch (error) { + log.error(`Error stopping MCP server '${name}':`, error); + throw error; + } + } + + async restartServer(name) { + await this.stopServer(name); + // Wait a bit before restarting + await new Promise(resolve => setTimeout(resolve, 1000)); + return await this.startServer(name); + } + + getServerStatus(name) { + const serverInfo = this.servers.get(name); + const config = this.config.mcpServers[name]; + + if (!config) { + return null; + } + + return { + name, + config, + isRunning: !!serverInfo, + status: serverInfo?.status || 'stopped', + startedAt: serverInfo?.startedAt, + error: serverInfo?.error, + pid: serverInfo?.process?.pid + }; + } + + getAllServers() { + const servers = []; + + for (const [name, config] of Object.entries(this.config.mcpServers)) { + const serverInfo = this.servers.get(name); + servers.push({ + name, + config, + isRunning: !!serverInfo, + status: serverInfo?.status || 'stopped', + startedAt: serverInfo?.startedAt, + error: serverInfo?.error, + pid: serverInfo?.process?.pid + }); + } + + return servers; + } + + async startAllEnabledServers() { + const results = []; + + for (const [name, config] of Object.entries(this.config.mcpServers)) { + if (config.enabled) { + try { + await this.startServer(name); + results.push({ name, success: true }); + } catch (error) { + log.error(`Failed to start MCP server '${name}':`, error); + results.push({ name, success: false, error: error.message }); + } + } + } + + return results; + } + + async stopAllServers() { + const results = []; + + for (const name of this.servers.keys()) { + try { + await this.stopServer(name); + results.push({ name, success: true }); + } catch (error) { + log.error(`Failed to stop MCP server '${name}':`, error); + results.push({ name, success: false, error: error.message }); + } + } + + return results; + } + + // Test server connection + async testServer(name) { + try { + const serverConfig = this.config.mcpServers[name]; + if (!serverConfig) { + throw new Error(`MCP server '${name}' not found`); + } + + // For remote servers, try to connect + if (serverConfig.type === 'remote') { + const response = await fetch(serverConfig.url, { + method: 'GET', + headers: serverConfig.headers || {} + }); + + if (!response.ok) { + throw new Error(`HTTP ${response.status}: ${response.statusText}`); + } + + return { success: true, message: 'Remote server is accessible' }; + } + + // For stdio servers, check if command exists + const { command } = serverConfig; + return new Promise((resolve) => { + const testProcess = spawn(command, ['--version'], { + stdio: 'ignore', + shell: process.platform === 'win32' + }); + + testProcess.on('error', (error) => { + resolve({ success: false, error: error.message }); + }); + + testProcess.on('exit', (code) => { + if (code === 0) { + resolve({ success: true, message: 'Command is available' }); + } else { + resolve({ success: false, error: `Command exited with code ${code}` }); + } + }); + }); + } catch (error) { + return { success: false, error: error.message }; + } + } + + // Get available MCP server templates + getServerTemplates() { + return [ + { + name: 'filesystem', + displayName: 'File System', + description: 'Access and manipulate files and directories', + command: 'npx', + args: ['@modelcontextprotocol/server-filesystem', '/path/to/directory'], + type: 'stdio', + category: 'System' + }, + { + name: 'git', + displayName: 'Git Repository', + description: 'Git repository operations and history', + command: 'npx', + args: ['@modelcontextprotocol/server-git', '/path/to/repo'], + type: 'stdio', + category: 'Development' + }, + { + name: 'sqlite', + displayName: 'SQLite Database', + description: 'Query and manipulate SQLite databases', + command: 'npx', + args: ['@modelcontextprotocol/server-sqlite', '/path/to/database.db'], + type: 'stdio', + category: 'Database' + }, + { + name: 'postgres', + displayName: 'PostgreSQL Database', + description: 'Connect to PostgreSQL databases', + command: 'npx', + args: ['@modelcontextprotocol/server-postgres'], + type: 'stdio', + category: 'Database', + env: { + POSTGRES_CONNECTION_STRING: 'postgresql://user:password@localhost:5432/dbname' + } + }, + { + name: 'puppeteer', + displayName: 'Web Scraping', + description: 'Web scraping and browser automation', + command: 'npx', + args: ['@modelcontextprotocol/server-puppeteer'], + type: 'stdio', + category: 'Web' + }, + { + name: 'brave-search', + displayName: 'Brave Search', + description: 'Search the web using Brave Search API', + command: 'npx', + args: ['@modelcontextprotocol/server-brave-search'], + type: 'stdio', + category: 'Search', + env: { + BRAVE_API_KEY: 'your-brave-api-key' + } + }, + { + name: 'github', + displayName: 'GitHub', + description: 'GitHub repository and issue management', + command: 'npx', + args: ['@modelcontextprotocol/server-github'], + type: 'stdio', + category: 'Development', + env: { + GITHUB_PERSONAL_ACCESS_TOKEN: 'your-github-token' + } + }, + { + name: 'slack', + displayName: 'Slack', + description: 'Slack workspace integration', + command: 'npx', + args: ['@modelcontextprotocol/server-slack'], + type: 'stdio', + category: 'Communication', + env: { + SLACK_BOT_TOKEN: 'your-slack-bot-token' + } + } + ]; + } + + // Import servers from Claude Desktop config + async importFromClaudeConfig(claudeConfigPath) { + try { + if (!fs.existsSync(claudeConfigPath)) { + throw new Error('Claude config file not found'); + } + + const claudeConfig = JSON.parse(fs.readFileSync(claudeConfigPath, 'utf8')); + const mcpServers = claudeConfig.mcpServers || {}; + + let imported = 0; + const errors = []; + + for (const [name, config] of Object.entries(mcpServers)) { + try { + if (!this.config.mcpServers[name]) { + this.config.mcpServers[name] = { + type: 'stdio', + command: config.command, + args: config.args || [], + env: config.env || {}, + description: `Imported from Claude Desktop`, + enabled: true, + createdAt: new Date().toISOString() + }; + imported++; + } + } catch (error) { + errors.push({ name, error: error.message }); + } + } + + if (imported > 0) { + this.saveConfig(); + } + + return { imported, errors }; + } catch (error) { + throw new Error(`Failed to import Claude config: ${error.message}`); + } + } + + // Execute MCP tool call + async executeToolCall(toolCall) { + try { + const { server: serverName, name: toolName, arguments: args, callId } = toolCall; + + // Get the server info + const serverInfo = this.servers.get(serverName); + if (!serverInfo || serverInfo.status !== 'running') { + return { + callId, + success: false, + error: `Server ${serverName} is not running` + }; + } + + // Handle special MCP protocol methods + if (toolName === 'tools/list') { + return await this.listToolsFromServer(serverName, callId); + } + + // For now, we'll implement basic MCP protocol communication + // This is a simplified implementation - a full MCP client would handle the complete protocol + const mcpRequest = { + jsonrpc: '2.0', + id: callId, + method: 'tools/call', + params: { + name: toolName, + arguments: args + } + }; + + // Send the request to the MCP server via stdin + const requestString = JSON.stringify(mcpRequest) + '\n'; + + return new Promise((resolve) => { + let responseData = ''; + let timeoutId; + + // Set up response handler + const onData = (data) => { + responseData += data.toString(); + + // Try to parse JSON response + try { + const lines = responseData.split('\n').filter(line => line.trim()); + for (const line of lines) { + try { + const response = JSON.parse(line); + if (response.id === callId) { + // Clean up + clearTimeout(timeoutId); + serverInfo.process.stdout.off('data', onData); + + if (response.error) { + resolve({ + callId, + success: false, + error: response.error.message || 'MCP tool execution failed' + }); + } else { + resolve({ + callId, + success: true, + content: response.result?.content || [{ type: 'text', text: JSON.stringify(response.result) }], + metadata: { + server: serverName, + tool: toolName, + executedAt: new Date().toISOString() + } + }); + } + return; + } + } catch (lineParseError) { + // Skip malformed lines and continue + log.debug(`Skipping malformed JSON line from ${serverName}:`, line); + continue; + } + } + } catch (parseError) { + // Log parsing errors but continue waiting for more data + log.debug(`JSON parsing error for ${serverName}:`, parseError.message); + } + }; + + // Set up timeout + timeoutId = setTimeout(() => { + serverInfo.process.stdout.off('data', onData); + resolve({ + callId, + success: false, + error: 'MCP tool execution timeout' + }); + }, 30000); // 30 second timeout + + // Listen for response + serverInfo.process.stdout.on('data', onData); + + // Send the request + try { + serverInfo.process.stdin.write(requestString); + } catch (writeError) { + clearTimeout(timeoutId); + serverInfo.process.stdout.off('data', onData); + resolve({ + callId, + success: false, + error: `Failed to send request to MCP server: ${writeError.message}` + }); + } + }); + + } catch (error) { + log.error(`Error executing MCP tool call:`, error); + return { + callId: toolCall.callId, + success: false, + error: error.message || 'Unknown error occurred' + }; + } + } + + // List tools from an MCP server + async listToolsFromServer(serverName, callId) { + try { + const serverInfo = this.servers.get(serverName); + if (!serverInfo || serverInfo.status !== 'running') { + return { + callId, + success: false, + error: `Server ${serverName} is not running` + }; + } + + const mcpRequest = { + jsonrpc: '2.0', + id: callId, + method: 'tools/list', + params: {} + }; + + const requestString = JSON.stringify(mcpRequest) + '\n'; + + return new Promise((resolve) => { + let responseData = ''; + let timeoutId; + + const onData = (data) => { + responseData += data.toString(); + + try { + const lines = responseData.split('\n').filter(line => line.trim()); + for (const line of lines) { + try { + const response = JSON.parse(line); + if (response.id === callId) { + clearTimeout(timeoutId); + serverInfo.process.stdout.off('data', onData); + + if (response.error) { + resolve({ + callId, + success: false, + error: response.error.message || 'Failed to list tools' + }); + } else { + // Return the tools list in a format the frontend expects + const tools = response.result?.tools || []; + resolve({ + callId, + success: true, + content: [{ + type: 'json', + text: JSON.stringify(tools), + data: tools + }], + metadata: { + server: serverName, + tool: 'tools/list', + executedAt: new Date().toISOString() + } + }); + } + return; + } + } catch (lineParseError) { + log.debug(`Skipping malformed JSON line from ${serverName}:`, line); + continue; + } + } + } catch (parseError) { + log.debug(`JSON parsing error for ${serverName}:`, parseError.message); + } + }; + + timeoutId = setTimeout(() => { + serverInfo.process.stdout.off('data', onData); + resolve({ + callId, + success: false, + error: 'Tool listing timeout' + }); + }, 10000); // 10 second timeout for listing + + serverInfo.process.stdout.on('data', onData); + + try { + serverInfo.process.stdin.write(requestString); + } catch (writeError) { + clearTimeout(timeoutId); + serverInfo.process.stdout.off('data', onData); + resolve({ + callId, + success: false, + error: `Failed to send tools/list request: ${writeError.message}` + }); + } + }); + + } catch (error) { + log.error(`Error listing tools from MCP server ${serverName}:`, error); + return { + callId, + success: false, + error: error.message || 'Unknown error occurred' + }; + } + } +} + +module.exports = MCPService; \ No newline at end of file diff --git a/electron/menu.cjs b/electron/menu.cjs new file mode 100644 index 00000000..a4a4afe4 --- /dev/null +++ b/electron/menu.cjs @@ -0,0 +1,161 @@ +const { app, Menu, shell } = require('electron'); +const { checkForUpdates } = require('./updateService.cjs'); + +function createAppMenu(mainWindow) { + const isMac = process.platform === 'darwin'; + + const template = [ + // App menu (macOS only) + ...(isMac ? [{ + label: app.name, + submenu: [ + { role: 'about' }, + { type: 'separator' }, + { + label: 'Check for Updates...', + click: () => checkForUpdates() + }, + { type: 'separator' }, + { role: 'services' }, + { type: 'separator' }, + { role: 'hide' }, + { role: 'hideOthers' }, + { role: 'unhide' }, + { type: 'separator' }, + { role: 'quit' } + ] + }] : []), + + // File menu + { + label: 'File', + submenu: [ + ...(isMac ? [] : [ + { + label: 'Check for Updates...', + click: () => checkForUpdates() + }, + { type: 'separator' } + ]), + { role: 'quit' } + ] + }, + + // Edit menu + { + label: 'Edit', + submenu: [ + { role: 'undo' }, + { role: 'redo' }, + { type: 'separator' }, + { role: 'cut' }, + { role: 'copy' }, + { role: 'paste' }, + ...(isMac ? [ + { role: 'pasteAndMatchStyle' }, + { role: 'delete' }, + { role: 'selectAll' }, + { type: 'separator' }, + { + label: 'Speech', + submenu: [ + { role: 'startSpeaking' }, + { role: 'stopSpeaking' } + ] + } + ] : [ + { role: 'delete' }, + { type: 'separator' }, + { role: 'selectAll' } + ]) + ] + }, + + // View menu + { + label: 'View', + submenu: [ + { role: 'reload' }, + { role: 'forceReload' }, + { role: 'toggleDevTools' }, + { type: 'separator' }, + { role: 'resetZoom' }, + { role: 'zoomIn' }, + { role: 'zoomOut' }, + { type: 'separator' }, + { role: 'togglefullscreen' } + ] + }, + + // Window menu + { + label: 'Window', + submenu: [ + { role: 'minimize' }, + { role: 'zoom' }, + ...(isMac ? [ + { type: 'separator' }, + { role: 'front' }, + { type: 'separator' }, + { role: 'window' } + ] : [ + { role: 'close' } + ]) + ] + }, + + // Help menu + { + role: 'help', + submenu: [ + { + label: 'Learn More', + click: async () => { + await shell.openExternal('https://github.com/badboysm890/ClaraVerse'); + } + }, + { + label: 'Documentation', + click: async () => { + await shell.openExternal('https://github.com/badboysm890/ClaraVerse#readme'); + } + }, + { + label: 'Report an Issue', + click: async () => { + await shell.openExternal('https://github.com/badboysm890/ClaraVerse/issues'); + } + }, + { type: 'separator' }, + { + label: 'Check for Updates...', + click: () => checkForUpdates() + }, + { + label: 'About Clara', + click: () => { + const version = app.getVersion(); + const electronVersion = process.versions.electron; + const nodeVersion = process.versions.node; + const message = `Clara Version: ${version}\nElectron: ${electronVersion}\nNode.js: ${nodeVersion}`; + + const { dialog } = require('electron'); + dialog.showMessageBox(mainWindow, { + title: 'About Clara', + message: 'Clara - Privacy-first, client-side AI assistant', + detail: message, + buttons: ['OK'] + }); + } + } + ] + } + ]; + + const menu = Menu.buildFromTemplate(template); + Menu.setApplicationMenu(menu); + + return menu; +} + +module.exports = { createAppMenu }; \ No newline at end of file diff --git a/electron/netlifyOAuthHandler.cjs b/electron/netlifyOAuthHandler.cjs new file mode 100644 index 00000000..a834b044 --- /dev/null +++ b/electron/netlifyOAuthHandler.cjs @@ -0,0 +1,125 @@ +/** + * Netlify OAuth Handler for Electron + * Handles OAuth flow using BrowserWindow instead of popup + */ + +const { BrowserWindow } = require('electron'); +const url = require('url'); + +class NetlifyOAuthHandler { + constructor() { + this.authWindow = null; + } + + /** + * Start OAuth flow + * @param {string} authUrl - The Netlify OAuth URL + * @returns {Promise} - Access token + */ + async authenticate(authUrl) { + return new Promise((resolve, reject) => { + // Close existing auth window if any + if (this.authWindow) { + this.authWindow.close(); + this.authWindow = null; + } + + // Create auth window + this.authWindow = new BrowserWindow({ + width: 600, + height: 700, + show: false, + webPreferences: { + nodeIntegration: false, + contextIsolation: true, + enableRemoteModule: false + }, + title: 'Connect to Netlify', + autoHideMenuBar: true + }); + + // Load the OAuth URL + this.authWindow.loadURL(authUrl); + + // Show window when ready + this.authWindow.once('ready-to-show', () => { + this.authWindow.show(); + }); + + // Handle navigation to extract token + this.authWindow.webContents.on('will-redirect', (event, redirectUrl) => { + this.handleCallback(redirectUrl, resolve, reject); + }); + + // Also check on did-navigate for some OAuth providers + this.authWindow.webContents.on('did-navigate', (event, navigateUrl) => { + this.handleCallback(navigateUrl, resolve, reject); + }); + + // Handle window closed + this.authWindow.on('closed', () => { + this.authWindow = null; + reject(new Error('OAuth window was closed before authentication completed')); + }); + + // Handle errors + this.authWindow.webContents.on('did-fail-load', (event, errorCode, errorDescription) => { + console.error('OAuth window failed to load:', errorCode, errorDescription); + }); + }); + } + + /** + * Handle OAuth callback URL + */ + handleCallback(callbackUrl, resolve, reject) { + const parsedUrl = url.parse(callbackUrl, true); + + // Check if this is the callback URL + if (callbackUrl.includes('/oauth-netlify-callback.html') || + parsedUrl.hash || + callbackUrl.includes('access_token')) { + + // Extract hash fragment (Netlify uses implicit flow with #access_token) + const hash = parsedUrl.hash ? parsedUrl.hash.substring(1) : ''; + + if (hash) { + const params = new URLSearchParams(hash); + const accessToken = params.get('access_token'); + const error = params.get('error'); + const errorDescription = params.get('error_description'); + + if (error) { + this.cleanup(); + reject(new Error(errorDescription || error)); + return; + } + + if (accessToken) { + this.cleanup(); + resolve(accessToken); + return; + } + } + } + } + + /** + * Cleanup auth window + */ + cleanup() { + if (this.authWindow) { + this.authWindow.close(); + this.authWindow = null; + } + } + + /** + * Cancel ongoing authentication + */ + cancel() { + this.cleanup(); + } +} + +module.exports = NetlifyOAuthHandler; diff --git a/electron/networkServiceManager.cjs b/electron/networkServiceManager.cjs new file mode 100644 index 00000000..6272a66f --- /dev/null +++ b/electron/networkServiceManager.cjs @@ -0,0 +1,199 @@ +const { app, webContents } = require('electron'); +const log = require('electron-log'); + +/** + * Network Service Manager - Prevents UI refreshes during service crashes + * Handles network service crash recovery without forcing renderer restarts + */ +class NetworkServiceManager { + constructor() { + this.crashCount = 0; + this.maxRetries = 3; + this.retryDelay = 1000; + this.isRecovering = false; + this.statePreservation = new Map(); + + this.setupNetworkServiceHandlers(); + } + + setupNetworkServiceHandlers() { + // Monitor for network service crashes + app.on('child-process-gone', (event, details) => { + if (details.type === 'Utility' && details.name === 'network.mojom.NetworkService') { + log.warn('🔄 Network service crashed, attempting graceful recovery...'); + this.handleNetworkServiceCrash(details); + } + }); + + // Preserve React state before potential crash + app.on('web-contents-created', (event, contents) => { + contents.on('render-process-gone', (crashEvent, details) => { + if (details.reason === 'crashed' || details.reason === 'abnormal-exit') { + this.preserveRendererState(contents); + } + }); + }); + } + + async handleNetworkServiceCrash(details) { + if (this.isRecovering) return; + + this.isRecovering = true; + this.crashCount++; + + try { + log.info(`🚑 Network service recovery attempt ${this.crashCount}/${this.maxRetries}`); + + // Prevent automatic renderer restart + const allWebContents = webContents.getAllWebContents(); + allWebContents.forEach(contents => { + if (contents.getType() === 'window') { + // Preserve current state + this.preserveWebContentsState(contents); + + // Disable automatic reload on network failure + contents.setWindowOpenHandler(() => ({ + action: 'allow', + overrideBrowserWindowOptions: { + webPreferences: { + webSecurity: false // Temporary for recovery + } + } + })); + } + }); + + // Wait for network service to recover + await this.waitForNetworkRecovery(); + + // Restore preserved state without full reload + await this.restorePreservedState(); + + log.info('✅ Network service recovered without UI refresh'); + + } catch (error) { + log.error('❌ Network service recovery failed:', error); + + if (this.crashCount >= this.maxRetries) { + log.warn('🔥 Max recovery attempts reached, allowing normal crash handling'); + return; + } + } finally { + this.isRecovering = false; + } + } + + preserveRendererState(contents) { + try { + // Inject state preservation script before potential crash + contents.executeJavaScriptInIsolatedWorld(999, [{ + code: ` + // Preserve React component state + if (window.React && window.React.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED) { + const fiber = document.querySelector('#root')?._reactInternalFiber || + document.querySelector('#root')?._reactInternals; + if (fiber) { + window.__PRESERVED_REACT_STATE__ = { + timestamp: Date.now(), + location: window.location.href, + reactState: 'preserved' + }; + } + } + + // Preserve form data and user inputs + const formData = {}; + document.querySelectorAll('input, textarea, select').forEach((elem, index) => { + if (elem.value) { + formData[\`input_\${index}\`] = { + type: elem.type, + value: elem.value, + name: elem.name, + id: elem.id + }; + } + }); + window.__PRESERVED_FORM_DATA__ = formData; + + 'state-preserved' + ` + }]); + } catch (error) { + log.warn('Failed to preserve renderer state:', error); + } + } + + preserveWebContentsState(contents) { + const webContentsId = contents.id; + this.statePreservation.set(webContentsId, { + url: contents.getURL(), + title: contents.getTitle(), + timestamp: Date.now() + }); + } + + async waitForNetworkRecovery() { + return new Promise((resolve) => { + let attempts = 0; + const maxAttempts = 10; + + const checkNetwork = () => { + attempts++; + + // Simple network connectivity test + require('dns').lookup('google.com', (err) => { + if (!err || attempts >= maxAttempts) { + resolve(); + } else { + setTimeout(checkNetwork, this.retryDelay); + } + }); + }; + + checkNetwork(); + }); + } + + async restorePreservedState() { + const allWebContents = webContents.getAllWebContents(); + + for (const contents of allWebContents) { + if (contents.getType() === 'window') { + try { + // Restore preserved state without reload + await contents.executeJavaScript(` + // Restore form data if preserved + if (window.__PRESERVED_FORM_DATA__) { + const formData = window.__PRESERVED_FORM_DATA__; + Object.values(formData).forEach(field => { + const elem = field.id ? document.getElementById(field.id) : + field.name ? document.querySelector(\`[name="\${field.name}"]\`) : + null; + if (elem && elem.type === field.type) { + elem.value = field.value; + } + }); + delete window.__PRESERVED_FORM_DATA__; + } + + // Signal successful recovery to React app + if (window.electron && window.electron.networkRecovered) { + window.electron.networkRecovered(); + } + + console.log('🔄 State restored after network service recovery'); + `); + } catch (error) { + log.warn('Failed to restore state for webContents:', error); + } + } + } + } + + // Reset crash counter after successful stable period + resetCrashCount() { + this.crashCount = 0; + } +} + +module.exports = NetworkServiceManager; diff --git a/electron/platformManager.cjs b/electron/platformManager.cjs new file mode 100644 index 00000000..a939c6ec --- /dev/null +++ b/electron/platformManager.cjs @@ -0,0 +1,1484 @@ +const os = require('os'); +const path = require('path'); +const fs = require('fs').promises; +const fsSync = require('fs'); +const log = require('electron-log'); +const yaml = require('js-yaml'); +const { app } = require('electron'); + +/** + * Platform Manager for handling cross-platform binary distribution + * Supports both pre-built binaries and just-in-time compilation + * Now includes comprehensive system resource validation + */ +class PlatformManager { + constructor(baseDir) { + this.baseDir = baseDir; + this.platformInfo = this.detectPlatformInfo(); + this.supportedPlatforms = this.getSupportedPlatforms(); + + // System resources configuration + this.systemResourcesConfigPath = path.join(app.getPath('userData'), 'clara-system-config.yaml'); + this.systemRequirements = { + minimum: { + ramGB: 8, + cpuCores: 4, + diskSpaceGB: 10 + }, + recommended: { + ramGB: 16, + cpuCores: 8, + diskSpaceGB: 50 + } + }; + + // OS version requirements + this.osRequirements = { + linux: { + minimumKernel: '4.4.0', + recommendedKernel: '5.4.0', + description: 'Linux Kernel 4.4+ (Ubuntu 16.04+, CentOS 7+, RHEL 7+)' + }, + darwin: { + minimumVersion: '20.0.0', // macOS Big Sur 11.0 + recommendedVersion: '21.0.0', // macOS Monterey 12.0 + description: 'macOS Big Sur 11.0 or later' + }, + win32: { + minimumBuild: 19041, // Windows 10 Build 19041 (2004 May 2020 Update) + recommendedBuild: 22000, // Windows 11 + description: 'Windows 10 Build 19041+ (May 2020 Update) or Windows 11' + } + }; + + this.systemResourcesInfo = null; + this.osCompatibilityInfo = null; + } + + detectPlatformInfo() { + const platform = os.platform(); + const arch = os.arch(); + + return { + platform, + arch, + platformDir: this.getPlatformDirectory(platform, arch), + isWindows: platform === 'win32', + isMac: platform === 'darwin', + isLinux: platform === 'linux', + osRelease: os.release(), + osType: os.type() + }; + } + + /** + * Comprehensive OS version validation + * Checks if the current OS version meets minimum requirements + */ + async validateOSCompatibility() { + log.info('🔍 Starting OS version compatibility validation...'); + + try { + const osInfo = await this.getDetailedOSInfo(); + const compatibility = this.evaluateOSCompatibility(osInfo); + + log.info('✅ OS compatibility validation completed'); + log.info(`🖥️ OS: ${osInfo.displayName}`); + log.info(`📊 Version: ${osInfo.version}`); + log.info(`✅ Compatible: ${compatibility.isSupported ? 'Yes' : 'No'}`); + + if (!compatibility.isSupported) { + log.error('❌ OS version not supported!'); + compatibility.issues.forEach(issue => log.error(` • ${issue}`)); + } + + return compatibility; + } catch (error) { + log.error('❌ OS compatibility validation failed:', error); + throw error; + } + } + + /** + * Get detailed OS information including version detection + */ + async getDetailedOSInfo() { + const platform = this.platformInfo.platform; + + let osInfo = { + platform, + arch: this.platformInfo.arch, + type: this.platformInfo.osType, + release: this.platformInfo.osRelease, + version: 'Unknown', + displayName: 'Unknown OS', + buildNumber: null, + kernelVersion: null, + codeName: null + }; + + try { + switch (platform) { + case 'linux': + osInfo = await this.getLinuxOSInfo(osInfo); + break; + case 'darwin': + osInfo = await this.getMacOSInfo(osInfo); + break; + case 'win32': + osInfo = await this.getWindowsOSInfo(osInfo); + break; + default: + osInfo.displayName = `Unsupported Platform: ${platform}`; + } + } catch (error) { + log.warn('⚠️ Failed to get detailed OS info, using fallback:', error.message); + osInfo.displayName = `${platform} ${osInfo.release}`; + osInfo.version = osInfo.release; + } + + this.osCompatibilityInfo = osInfo; + return osInfo; + } + + /** + * Get detailed Linux OS information + */ + async getLinuxOSInfo(baseInfo) { + try { + const { spawn } = require('child_process'); + + // Get kernel version + const kernelVersion = baseInfo.release; + + // Try to get distribution info + let distributionInfo = await this.getLinuxDistributionInfo(); + + return { + ...baseInfo, + kernelVersion, + version: kernelVersion, + displayName: distributionInfo.name || 'Linux', + codeName: distributionInfo.codeName, + distributionVersion: distributionInfo.version + }; + } catch (error) { + log.warn('⚠️ Failed to get Linux distribution info:', error.message); + return { + ...baseInfo, + kernelVersion: baseInfo.release, + version: baseInfo.release, + displayName: `Linux ${baseInfo.release}` + }; + } + } + + /** + * Get Linux distribution information + */ + async getLinuxDistributionInfo() { + try { + const { spawn } = require('child_process'); + + return new Promise((resolve) => { + // Try lsb_release first + const lsbRelease = spawn('lsb_release', ['-a'], { stdio: 'pipe' }); + let output = ''; + + lsbRelease.stdout.on('data', (data) => { + output += data.toString(); + }); + + lsbRelease.on('close', (code) => { + if (code === 0 && output) { + const lines = output.split('\n'); + const info = {}; + + lines.forEach(line => { + if (line.includes('Distributor ID:')) { + info.name = line.split(':')[1]?.trim(); + } else if (line.includes('Release:')) { + info.version = line.split(':')[1]?.trim(); + } else if (line.includes('Codename:')) { + info.codeName = line.split(':')[1]?.trim(); + } + }); + + if (info.name) { + return resolve(info); + } + } + + // Fallback to /etc/os-release + this.readOSReleaseFile().then(resolve).catch(() => resolve({})); + }); + + lsbRelease.on('error', () => { + // Fallback to /etc/os-release + this.readOSReleaseFile().then(resolve).catch(() => resolve({})); + }); + }); + } catch (error) { + return {}; + } + } + + /** + * Read /etc/os-release file + */ + async readOSReleaseFile() { + try { + const content = await fs.readFile('/etc/os-release', 'utf8'); + const lines = content.split('\n'); + const info = {}; + + lines.forEach(line => { + const [key, value] = line.split('='); + if (key && value) { + const cleanValue = value.replace(/['"]/g, ''); + if (key === 'NAME') info.name = cleanValue; + if (key === 'VERSION') info.version = cleanValue; + if (key === 'VERSION_CODENAME') info.codeName = cleanValue; + } + }); + + return info; + } catch (error) { + return {}; + } + } + + /** + * Get detailed macOS information + */ + async getMacOSInfo(baseInfo) { + try { + const { spawn } = require('child_process'); + + return new Promise((resolve) => { + const swVers = spawn('sw_vers', [], { stdio: 'pipe' }); + let output = ''; + + swVers.stdout.on('data', (data) => { + output += data.toString(); + }); + + swVers.on('close', (code) => { + try { + if (code === 0 && output) { + const lines = output.split('\n'); + let productName = 'macOS'; + let productVersion = baseInfo.release; + let buildVersion = null; + + lines.forEach(line => { + if (line.includes('ProductName:')) { + productName = line.split(':')[1]?.trim() || productName; + } else if (line.includes('ProductVersion:')) { + productVersion = line.split(':')[1]?.trim() || productVersion; + } else if (line.includes('BuildVersion:')) { + buildVersion = line.split(':')[1]?.trim(); + } + }); + + return resolve({ + ...baseInfo, + version: productVersion, + displayName: `${productName} ${productVersion}`, + buildNumber: buildVersion, + codeName: this.getMacOSCodeName(productVersion) + }); + } + } catch (parseError) { + log.warn('⚠️ Failed to parse macOS version info:', parseError.message); + } + + // Fallback + resolve({ + ...baseInfo, + version: baseInfo.release, + displayName: `macOS ${baseInfo.release}` + }); + }); + + swVers.on('error', () => { + resolve({ + ...baseInfo, + version: baseInfo.release, + displayName: `macOS ${baseInfo.release}` + }); + }); + }); + } catch (error) { + return { + ...baseInfo, + version: baseInfo.release, + displayName: `macOS ${baseInfo.release}` + }; + } + } + + /** + * Get macOS code name from version + */ + getMacOSCodeName(version) { + const versionMap = { + '14': 'Sonoma', + '13': 'Ventura', + '12': 'Monterey', + '11': 'Big Sur', + '10.15': 'Catalina', + '10.14': 'Mojave', + '10.13': 'High Sierra', + '10.12': 'Sierra' + }; + + const majorVersion = version.split('.')[0]; + return versionMap[majorVersion] || versionMap[version] || 'Unknown'; + } + + /** + * Get detailed Windows OS information + */ + async getWindowsOSInfo(baseInfo) { + try { + const { spawn } = require('child_process'); + + return new Promise((resolve) => { + // Use PowerShell to get detailed Windows info + const psCommand = ` + $os = Get-WmiObject -Class Win32_OperatingSystem; + $version = [System.Environment]::OSVersion.Version; + Write-Output "Caption: $($os.Caption)"; + Write-Output "Version: $($os.Version)"; + Write-Output "BuildNumber: $($os.BuildNumber)"; + Write-Output "OSArchitecture: $($os.OSArchitecture)"; + Write-Output "ServicePack: $($os.ServicePackMajorVersion).$($os.ServicePackMinorVersion)"; + `; + + const powershell = spawn('powershell', ['-Command', psCommand], { stdio: 'pipe' }); + let output = ''; + + powershell.stdout.on('data', (data) => { + output += data.toString(); + }); + + powershell.on('close', (code) => { + try { + if (code === 0 && output) { + const lines = output.split('\n'); + let caption = 'Windows'; + let version = baseInfo.release; + let buildNumber = null; + + lines.forEach(line => { + if (line.includes('Caption:')) { + caption = line.split(':')[1]?.trim() || caption; + } else if (line.includes('Version:')) { + version = line.split(':')[1]?.trim() || version; + } else if (line.includes('BuildNumber:')) { + buildNumber = parseInt(line.split(':')[1]?.trim()) || null; + } + }); + + return resolve({ + ...baseInfo, + version, + displayName: caption, + buildNumber, + codeName: this.getWindowsCodeName(buildNumber) + }); + } + } catch (parseError) { + log.warn('⚠️ Failed to parse Windows version info:', parseError.message); + } + + // Fallback + resolve({ + ...baseInfo, + version: baseInfo.release, + displayName: `Windows ${baseInfo.release}`, + buildNumber: this.extractWindowsBuildFromRelease(baseInfo.release) + }); + }); + + powershell.on('error', () => { + resolve({ + ...baseInfo, + version: baseInfo.release, + displayName: `Windows ${baseInfo.release}`, + buildNumber: this.extractWindowsBuildFromRelease(baseInfo.release) + }); + }); + }); + } catch (error) { + return { + ...baseInfo, + version: baseInfo.release, + displayName: `Windows ${baseInfo.release}`, + buildNumber: this.extractWindowsBuildFromRelease(baseInfo.release) + }; + } + } + + /** + * Extract Windows build number from release string + */ + extractWindowsBuildFromRelease(release) { + // Windows release typically contains build number + const match = release.match(/(\d{5,})/); + return match ? parseInt(match[1]) : null; + } + + /** + * Get Windows code name from build number + */ + getWindowsCodeName(buildNumber) { + if (!buildNumber) return 'Unknown'; + + if (buildNumber >= 22000) return 'Windows 11'; + if (buildNumber >= 19041) return 'Windows 10 (2004+)'; + if (buildNumber >= 18363) return 'Windows 10 (1909)'; + if (buildNumber >= 18362) return 'Windows 10 (1903)'; + if (buildNumber >= 17763) return 'Windows 10 (1809)'; + if (buildNumber >= 17134) return 'Windows 10 (1803)'; + if (buildNumber >= 16299) return 'Windows 10 (1709)'; + if (buildNumber >= 15063) return 'Windows 10 (1703)'; + if (buildNumber >= 14393) return 'Windows 10 (1607)'; + if (buildNumber >= 10240) return 'Windows 10 (1507)'; + + return 'Windows (Legacy)'; + } + + /** + * Evaluate OS compatibility based on detected version + */ + evaluateOSCompatibility(osInfo) { + const platform = osInfo.platform; + const requirements = this.osRequirements[platform]; + + if (!requirements) { + return { + osInfo, + isSupported: false, + meetsMinimumRequirements: false, + meetsRecommendedRequirements: false, + issues: [`Unsupported platform: ${platform}`], + warnings: [], + recommendations: ['Please use a supported operating system'], + upgradeInstructions: this.getUnsupportedPlatformInstructions(platform) + }; + } + + const evaluation = { + osInfo, + isSupported: true, + meetsMinimumRequirements: true, + meetsRecommendedRequirements: true, + issues: [], + warnings: [], + recommendations: [], + upgradeInstructions: null + }; + + // Platform-specific compatibility checks + switch (platform) { + case 'linux': + this.evaluateLinuxCompatibility(osInfo, requirements, evaluation); + break; + case 'darwin': + this.evaluateMacOSCompatibility(osInfo, requirements, evaluation); + break; + case 'win32': + this.evaluateWindowsCompatibility(osInfo, requirements, evaluation); + break; + } + + // Add upgrade instructions if not compatible + if (!evaluation.isSupported) { + evaluation.upgradeInstructions = this.getUpgradeInstructions(platform, osInfo); + } + + return evaluation; + } + + /** + * Evaluate Linux kernel version compatibility + */ + evaluateLinuxCompatibility(osInfo, requirements, evaluation) { + const kernelVersion = osInfo.kernelVersion || osInfo.version; + + if (!this.compareVersions(kernelVersion, requirements.minimumKernel)) { + evaluation.isSupported = false; + evaluation.meetsMinimumRequirements = false; + evaluation.issues.push( + `Linux kernel ${kernelVersion} is below minimum required version ${requirements.minimumKernel}` + ); + } else if (!this.compareVersions(kernelVersion, requirements.recommendedKernel)) { + evaluation.meetsRecommendedRequirements = false; + evaluation.warnings.push( + `Linux kernel ${kernelVersion} is below recommended version ${requirements.recommendedKernel}` + ); + evaluation.recommendations.push('Consider upgrading to a newer Linux distribution for optimal performance'); + } + + // Additional checks for specific distributions + if (osInfo.displayName && osInfo.distributionVersion) { + this.checkLinuxDistributionCompatibility(osInfo, evaluation); + } + } + + /** + * Check specific Linux distribution compatibility + */ + checkLinuxDistributionCompatibility(osInfo, evaluation) { + const distName = osInfo.displayName.toLowerCase(); + const distVersion = osInfo.distributionVersion; + + const knownDistributions = { + ubuntu: { minimum: '16.04', recommended: '20.04' }, + debian: { minimum: '9', recommended: '11' }, + centos: { minimum: '7', recommended: '8' }, + rhel: { minimum: '7', recommended: '8' }, + fedora: { minimum: '30', recommended: '35' }, + opensuse: { minimum: '15.0', recommended: '15.3' } + }; + + for (const [dist, versions] of Object.entries(knownDistributions)) { + if (distName.includes(dist)) { + if (!this.compareVersions(distVersion, versions.minimum)) { + evaluation.warnings.push( + `${osInfo.displayName} ${distVersion} may have compatibility issues. Minimum supported: ${versions.minimum}` + ); + } else if (!this.compareVersions(distVersion, versions.recommended)) { + evaluation.recommendations.push( + `Consider upgrading to ${dist} ${versions.recommended}+ for best compatibility` + ); + } + break; + } + } + } + + /** + * Evaluate macOS version compatibility + */ + evaluateMacOSCompatibility(osInfo, requirements, evaluation) { + const version = osInfo.version; + const darwinVersion = osInfo.release; + + // Check Darwin kernel version (more reliable) + if (!this.compareVersions(darwinVersion, requirements.minimumVersion)) { + evaluation.isSupported = false; + evaluation.meetsMinimumRequirements = false; + evaluation.issues.push( + `macOS version ${version} (Darwin ${darwinVersion}) is below minimum required macOS Big Sur 11.0` + ); + } else if (!this.compareVersions(darwinVersion, requirements.recommendedVersion)) { + evaluation.meetsRecommendedRequirements = false; + evaluation.warnings.push( + `macOS ${version} is below recommended version. Consider upgrading to macOS Monterey 12.0+ for optimal performance` + ); + } + + // Additional macOS-specific checks + if (evaluation.isSupported) { + this.checkMacOSSpecificFeatures(osInfo, evaluation); + } + } + + /** + * Check macOS-specific features and compatibility + */ + checkMacOSSpecificFeatures(osInfo, evaluation) { + const majorVersion = parseInt(osInfo.version.split('.')[0]); + + if (majorVersion < 12) { + evaluation.recommendations.push('macOS Monterey 12.0+ recommended for best Metal GPU acceleration support'); + } + + if (majorVersion < 13) { + evaluation.recommendations.push('macOS Ventura 13.0+ recommended for latest security features'); + } + } + + /** + * Evaluate Windows version compatibility + */ + evaluateWindowsCompatibility(osInfo, requirements, evaluation) { + const buildNumber = osInfo.buildNumber; + + if (!buildNumber || buildNumber < requirements.minimumBuild) { + evaluation.isSupported = false; + evaluation.meetsMinimumRequirements = false; + evaluation.issues.push( + `Windows build ${buildNumber || 'unknown'} is below minimum required build ${requirements.minimumBuild} (Windows 10 May 2020 Update)` + ); + } else if (buildNumber < requirements.recommendedBuild) { + evaluation.meetsRecommendedRequirements = false; + evaluation.warnings.push( + `Windows build ${buildNumber} is below recommended build ${requirements.recommendedBuild} (Windows 11)` + ); + evaluation.recommendations.push('Consider upgrading to Windows 11 for optimal performance and features'); + } + + // Additional Windows-specific checks + if (evaluation.isSupported) { + this.checkWindowsSpecificFeatures(osInfo, evaluation); + } + } + + /** + * Check Windows-specific features + */ + checkWindowsSpecificFeatures(osInfo, evaluation) { + const buildNumber = osInfo.buildNumber; + + if (buildNumber && buildNumber < 19041) { + evaluation.warnings.push('WSL2 support requires Windows 10 Build 19041+'); + } + + if (buildNumber && buildNumber < 22000) { + evaluation.recommendations.push('Windows 11 provides better performance for AI workloads'); + } + } + + /** + * Compare version strings (returns true if version1 >= version2) + */ + compareVersions(version1, version2) { + if (!version1 || !version2) return false; + + const v1Parts = version1.split('.').map(x => parseInt(x) || 0); + const v2Parts = version2.split('.').map(x => parseInt(x) || 0); + + const maxLength = Math.max(v1Parts.length, v2Parts.length); + + for (let i = 0; i < maxLength; i++) { + const v1Part = v1Parts[i] || 0; + const v2Part = v2Parts[i] || 0; + + if (v1Part > v2Part) return true; + if (v1Part < v2Part) return false; + } + + return true; // Equal versions + } + + /** + * Get upgrade instructions for unsupported OS versions + */ + getUpgradeInstructions(platform, osInfo) { + switch (platform) { + case 'linux': + return { + title: 'Linux Kernel Update Required', + description: `Your Linux kernel version ${osInfo.kernelVersion || osInfo.version} is not supported.`, + minimumRequired: this.osRequirements.linux.description, + instructions: [ + 'Update your Linux distribution to a newer version:', + '• Ubuntu: Run "sudo apt update && sudo apt upgrade" then "sudo do-release-upgrade"', + '• CentOS/RHEL: Upgrade to CentOS 8+ or RHEL 8+', + '• Debian: Upgrade to Debian 11+ (Bullseye)', + '• Fedora: Upgrade to Fedora 35+', + '• Or install a modern Linux distribution with kernel 4.4+' + ], + downloadLinks: [ + { name: 'Ubuntu LTS', url: 'https://ubuntu.com/download' }, + { name: 'Debian', url: 'https://www.debian.org/distrib/' }, + { name: 'Fedora', url: 'https://getfedora.org/' } + ] + }; + + case 'darwin': + return { + title: 'macOS Update Required', + description: `Your macOS version ${osInfo.version} is not supported.`, + minimumRequired: this.osRequirements.darwin.description, + instructions: [ + 'Update your macOS to Big Sur 11.0 or later:', + '1. Click the Apple menu → About This Mac', + '2. Click "Software Update" to check for updates', + '3. Install macOS Big Sur, Monterey, Ventura, or Sonoma', + '4. Restart your Mac after installation', + '', + 'If Software Update doesn\'t show newer versions:', + '• Your Mac might not be compatible with newer macOS versions', + '• Check Apple\'s compatibility list for your Mac model' + ], + downloadLinks: [ + { name: 'macOS Compatibility', url: 'https://support.apple.com/en-us/HT201260' }, + { name: 'macOS Updates', url: 'https://support.apple.com/en-us/HT201541' } + ] + }; + + case 'win32': + return { + title: 'Windows Update Required', + description: `Your Windows version (build ${osInfo.buildNumber || 'unknown'}) is not supported.`, + minimumRequired: this.osRequirements.win32.description, + instructions: [ + 'Update Windows to build 19041 or later:', + '1. Press Win + I to open Settings', + '2. Go to Update & Security → Windows Update', + '3. Click "Check for updates"', + '4. Install all available updates', + '5. Restart your computer', + '', + 'For Windows 11 (recommended):', + '• Check if your PC meets Windows 11 requirements', + '• Download Windows 11 from Microsoft if compatible' + ], + downloadLinks: [ + { name: 'Windows Update', url: 'https://support.microsoft.com/en-us/windows/update-windows-3c5ae7fc-9fb6-9af1-1984-b5e0412c556a' }, + { name: 'Windows 11', url: 'https://www.microsoft.com/software-download/windows11' }, + { name: 'PC Health Check', url: 'https://aka.ms/GetPCHealthCheckApp' } + ] + }; + + default: + return { + title: 'Unsupported Operating System', + description: `Platform ${platform} is not supported.`, + minimumRequired: 'Supported platforms: Windows 10+, macOS Big Sur 11.0+, Linux Kernel 4.4+', + instructions: [ + 'Please use one of the supported operating systems:', + '• Windows 10 Build 19041+ or Windows 11', + '• macOS Big Sur 11.0 or later', + '• Linux with kernel 4.4+ (Ubuntu 16.04+, CentOS 7+, etc.)' + ], + downloadLinks: [] + }; + } + } + + /** + * Get instructions for completely unsupported platforms + */ + getUnsupportedPlatformInstructions(platform) { + return { + title: `Unsupported Platform: ${platform}`, + description: `The platform ${platform} is not currently supported by ClaraVerse.`, + minimumRequired: 'Supported platforms: Windows, macOS, Linux', + instructions: [ + 'ClaraVerse currently supports:', + '• Windows 10+ (x64)', + '• macOS Big Sur 11.0+ (Intel and Apple Silicon)', + '• Linux x64 with kernel 4.4+ (Ubuntu, Debian, CentOS, RHEL, Fedora, etc.)', + '', + 'Please use one of these supported platforms to run ClaraVerse.' + ], + downloadLinks: [ + { name: 'Windows', url: 'https://www.microsoft.com/windows' }, + { name: 'macOS', url: 'https://www.apple.com/macos' }, + { name: 'Ubuntu Linux', url: 'https://ubuntu.com' } + ] + }; + } + + /** + * Comprehensive system resource validation + * Checks RAM, CPU cores, disk space, and OS compatibility + */ + async validateSystemResources() { + log.info('🔍 Starting comprehensive system resource validation...'); + + try { + // Validate system resources + const systemInfo = await this.getSystemResourceInfo(); + const resourceValidation = this.evaluateSystemResources(systemInfo); + + // Validate OS compatibility + const osCompatibility = await this.validateOSCompatibility(); + + // Combine validations + const combinedValidation = { + ...resourceValidation, + osCompatibility, + overallCompatible: resourceValidation.isCompatible && osCompatibility.isSupported + }; + + // Save system configuration + await this.saveSystemConfiguration(combinedValidation); + + log.info('✅ System resource validation completed'); + log.info(`💾 RAM: ${systemInfo.ramGB}GB (Required: ${this.systemRequirements.minimum.ramGB}GB)`); + log.info(`🖥️ CPU Cores: ${systemInfo.cpuCores} (Required: ${this.systemRequirements.minimum.cpuCores})`); + log.info(`💽 Available Disk Space: ${systemInfo.availableDiskSpaceGB}GB (Required: ${this.systemRequirements.minimum.diskSpaceGB}GB)`); + log.info(`🎯 System Performance Mode: ${resourceValidation.performanceMode}`); + log.info(`📱 OS Compatibility: ${osCompatibility.isSupported ? 'Supported' : 'Not Supported'}`); + + // Log OS compatibility issues + if (!osCompatibility.isSupported) { + log.error('❌ OS compatibility issues detected:'); + osCompatibility.issues.forEach(issue => log.error(` • ${issue}`)); + } + + if (osCompatibility.warnings.length > 0) { + log.warn('⚠️ OS compatibility warnings:'); + osCompatibility.warnings.forEach(warning => log.warn(` • ${warning}`)); + } + + return combinedValidation; + } catch (error) { + log.error('❌ System resource validation failed:', error); + throw error; + } + } + + /** + * Get comprehensive system resource information + */ + async getSystemResourceInfo() { + const cpus = os.cpus(); + const totalMemory = os.totalmem(); + const freeMemory = os.freemem(); + + // Get disk space information + const userDataPath = app.getPath('userData'); + const diskSpace = await this.getDiskSpace(userDataPath); + + const systemInfo = { + // Memory information + ramGB: Math.round(totalMemory / (1024 * 1024 * 1024)), + freeRamGB: Math.round(freeMemory / (1024 * 1024 * 1024)), + usedRamGB: Math.round((totalMemory - freeMemory) / (1024 * 1024 * 1024)), + + // CPU information + cpuCores: cpus.length, + cpuModel: cpus[0]?.model || 'Unknown', + cpuSpeed: cpus[0]?.speed || 0, + + // Disk space information + totalDiskSpaceGB: diskSpace.total, + availableDiskSpaceGB: diskSpace.available, + usedDiskSpaceGB: diskSpace.used, + + // Platform information + platform: this.platformInfo.platform, + arch: this.platformInfo.arch, + osRelease: os.release(), + + // System load (if available) + loadAverage: os.loadavg(), + + // Timestamp + timestamp: new Date().toISOString() + }; + + this.systemResourcesInfo = systemInfo; + return systemInfo; + } + + /** + * Get disk space information for a given path + */ + async getDiskSpace(dirPath) { + try { + const stats = await fs.statvfs ? fs.statvfs(dirPath) : null; + + if (stats) { + // Unix-like systems + const blockSize = stats.f_bsize || stats.f_frsize; + const totalBlocks = stats.f_blocks; + const availableBlocks = stats.f_bavail; + + const total = Math.round((totalBlocks * blockSize) / (1024 * 1024 * 1024)); + const available = Math.round((availableBlocks * blockSize) / (1024 * 1024 * 1024)); + const used = total - available; + + return { total, available, used }; + } else { + // Fallback for systems without statvfs + log.warn('⚠️ statvfs not available, using fallback disk space detection'); + return await this.getDiskSpaceFallback(dirPath); + } + } catch (error) { + log.warn('⚠️ Failed to get disk space info, using fallback:', error.message); + return await this.getDiskSpaceFallback(dirPath); + } + } + + /** + * Fallback disk space detection using df command or Windows dir + */ + async getDiskSpaceFallback(dirPath) { + try { + const { spawn } = require('child_process'); + + return new Promise((resolve) => { + let command, args; + + if (this.platformInfo.isWindows) { + // Windows: Use PowerShell to get disk space + command = 'powershell'; + args = ['-Command', `Get-WmiObject -Class Win32_LogicalDisk | Where-Object {$_.DeviceID -eq "${path.parse(dirPath).root.replace('\\', '')}"} | Select-Object Size,FreeSpace`]; + } else { + // Unix-like: Use df command + command = 'df'; + args = ['-BG', dirPath]; + } + + const proc = spawn(command, args, { stdio: 'pipe' }); + let output = ''; + + proc.stdout.on('data', (data) => { + output += data.toString(); + }); + + proc.on('close', (code) => { + try { + if (this.platformInfo.isWindows) { + // Parse Windows PowerShell output + const lines = output.trim().split('\n'); + const dataLine = lines.find(line => line.includes('Size') || /^\s*\d/.test(line)); + if (dataLine) { + const match = dataLine.match(/(\d+)\s+(\d+)/); + if (match) { + const total = Math.round(parseInt(match[1]) / (1024 * 1024 * 1024)); + const available = Math.round(parseInt(match[2]) / (1024 * 1024 * 1024)); + const used = total - available; + return resolve({ total, available, used }); + } + } + } else { + // Parse Unix df output + const lines = output.trim().split('\n'); + const dataLine = lines[lines.length - 1]; + const parts = dataLine.split(/\s+/); + if (parts.length >= 4) { + const total = Math.round(parseInt(parts[1].replace('G', '')) || 0); + const used = Math.round(parseInt(parts[2].replace('G', '')) || 0); + const available = Math.round(parseInt(parts[3].replace('G', '')) || 0); + return resolve({ total, available, used }); + } + } + } catch (parseError) { + log.warn('⚠️ Failed to parse disk space output:', parseError.message); + } + + // Fallback to conservative estimates + resolve({ total: 100, available: 50, used: 50 }); + }); + + proc.on('error', () => { + // Fallback to conservative estimates + resolve({ total: 100, available: 50, used: 50 }); + }); + }); + } catch (error) { + log.warn('⚠️ Fallback disk space detection failed:', error.message); + return { total: 100, available: 50, used: 50 }; + } + } + + /** + * Evaluate system resources and determine performance mode + */ + evaluateSystemResources(systemInfo) { + const { minimum, recommended } = this.systemRequirements; + + const evaluation = { + systemInfo, + meetsMinimumRequirements: true, + meetsRecommendedRequirements: true, + issues: [], + warnings: [], + recommendations: [], + performanceMode: 'full', // full, lite, core-only + enabledFeatures: { + claraCore: true, + dockerServices: true, + comfyUI: true, + n8nWorkflows: true, + agentStudio: true, + lumaUI: true, + advancedFeatures: true + }, + resourceLimitations: {} + }; + + // Check RAM + if (systemInfo.ramGB < minimum.ramGB) { + evaluation.meetsMinimumRequirements = false; + evaluation.issues.push(`Insufficient RAM: ${systemInfo.ramGB}GB (minimum required: ${minimum.ramGB}GB)`); + evaluation.performanceMode = 'core-only'; + } else if (systemInfo.ramGB < recommended.ramGB) { + evaluation.meetsRecommendedRequirements = false; + evaluation.warnings.push(`RAM below recommended: ${systemInfo.ramGB}GB (recommended: ${recommended.ramGB}GB)`); + evaluation.performanceMode = 'lite'; + } + + // Check CPU cores + if (systemInfo.cpuCores < minimum.cpuCores) { + evaluation.meetsMinimumRequirements = false; + evaluation.issues.push(`Insufficient CPU cores: ${systemInfo.cpuCores} (minimum required: ${minimum.cpuCores})`); + evaluation.performanceMode = 'core-only'; + } else if (systemInfo.cpuCores < recommended.cpuCores) { + evaluation.meetsRecommendedRequirements = false; + evaluation.warnings.push(`CPU cores below recommended: ${systemInfo.cpuCores} (recommended: ${recommended.cpuCores})`); + if (evaluation.performanceMode === 'full') { + evaluation.performanceMode = 'lite'; + } + } + + // Check disk space + if (systemInfo.availableDiskSpaceGB < minimum.diskSpaceGB) { + evaluation.meetsMinimumRequirements = false; + evaluation.issues.push(`Insufficient disk space: ${systemInfo.availableDiskSpaceGB}GB (minimum required: ${minimum.diskSpaceGB}GB)`); + evaluation.performanceMode = 'core-only'; + } else if (systemInfo.availableDiskSpaceGB < recommended.diskSpaceGB) { + evaluation.meetsRecommendedRequirements = false; + evaluation.warnings.push(`Disk space below recommended: ${systemInfo.availableDiskSpaceGB}GB (recommended: ${recommended.diskSpaceGB}GB)`); + } + + // Configure features based on performance mode + this.configureFeaturesByPerformanceMode(evaluation); + + // Add recommendations based on limitations + this.addPerformanceRecommendations(evaluation); + + return evaluation; + } + + /** + * Configure available features based on system performance mode + */ + configureFeaturesByPerformanceMode(evaluation) { + const { performanceMode } = evaluation; + + switch (performanceMode) { + case 'core-only': + evaluation.enabledFeatures = { + claraCore: true, + dockerServices: false, + comfyUI: false, + n8nWorkflows: false, + agentStudio: false, + lumaUI: false, + advancedFeatures: false + }; + evaluation.resourceLimitations = { + maxConcurrentModels: 1, + maxContextSize: 4096, + disableGPUAcceleration: true, + limitedThreads: Math.max(2, Math.floor(evaluation.systemInfo.cpuCores / 2)) + }; + break; + + case 'lite': + evaluation.enabledFeatures = { + claraCore: true, + dockerServices: false, + comfyUI: false, + n8nWorkflows: true, + agentStudio: true, + lumaUI: true, + advancedFeatures: false + }; + evaluation.resourceLimitations = { + maxConcurrentModels: 1, + maxContextSize: 8192, + limitedThreads: Math.max(4, Math.floor(evaluation.systemInfo.cpuCores * 0.75)) + }; + break; + + case 'full': + // All features enabled with no limitations + evaluation.resourceLimitations = { + maxConcurrentModels: 3, + maxContextSize: 32768 + }; + break; + } + } + + /** + * Add performance recommendations based on system analysis + */ + addPerformanceRecommendations(evaluation) { + const { systemInfo, performanceMode } = evaluation; + + if (performanceMode === 'core-only') { + evaluation.recommendations.push('🎯 Running in Core-Only mode for optimal performance on your system'); + evaluation.recommendations.push('💡 Consider upgrading RAM to 16GB+ and CPU to 8+ cores for full features'); + } else if (performanceMode === 'lite') { + evaluation.recommendations.push('⚡ Running in Lite mode - some resource-intensive features are disabled'); + evaluation.recommendations.push('🔧 Docker services disabled to preserve system resources'); + } + + // Memory-specific recommendations + if (systemInfo.ramGB <= 8) { + evaluation.recommendations.push('🧠 Consider closing other applications to free up memory'); + } + + // CPU-specific recommendations + if (systemInfo.cpuCores <= 4) { + evaluation.recommendations.push('⚙️ Model inference may be slower due to limited CPU cores'); + } + + // Disk space recommendations + if (systemInfo.availableDiskSpaceGB < 20) { + evaluation.recommendations.push('💾 Consider freeing up disk space for better performance'); + } + } + + /** + * Save system configuration to YAML file + */ + async saveSystemConfiguration(validation) { + try { + const config = { + version: '1.1.0', // Updated version to include OS compatibility + lastUpdated: new Date().toISOString(), + systemInfo: validation.systemInfo, + performanceMode: validation.performanceMode, + enabledFeatures: validation.enabledFeatures, + resourceLimitations: validation.resourceLimitations, + meetsMinimumRequirements: validation.meetsMinimumRequirements, + meetsRecommendedRequirements: validation.meetsRecommendedRequirements, + issues: validation.issues, + warnings: validation.warnings, + recommendations: validation.recommendations, + + // OS Compatibility Information + osCompatibility: validation.osCompatibility ? { + osInfo: validation.osCompatibility.osInfo, + isSupported: validation.osCompatibility.isSupported, + meetsMinimumRequirements: validation.osCompatibility.meetsMinimumRequirements, + meetsRecommendedRequirements: validation.osCompatibility.meetsRecommendedRequirements, + issues: validation.osCompatibility.issues, + warnings: validation.osCompatibility.warnings, + recommendations: validation.osCompatibility.recommendations, + upgradeInstructions: validation.osCompatibility.upgradeInstructions + } : null, + + // Overall compatibility (resources + OS) + overallCompatible: validation.overallCompatible || false + }; + + const yamlContent = yaml.dump(config, { + indent: 2, + lineWidth: 120, + noRefs: true + }); + + await fs.writeFile(this.systemResourcesConfigPath, yamlContent, 'utf8'); + log.info(`✅ System configuration saved to: ${this.systemResourcesConfigPath}`); + + return config; + } catch (error) { + log.error('❌ Failed to save system configuration:', error); + throw error; + } + } + + /** + * Load system configuration from YAML file + */ + async loadSystemConfiguration() { + try { + if (!fsSync.existsSync(this.systemResourcesConfigPath)) { + log.info('ℹ️ No existing system configuration found, will create new one'); + return null; + } + + const yamlContent = await fs.readFile(this.systemResourcesConfigPath, 'utf8'); + const config = yaml.load(yamlContent); + + log.info('✅ System configuration loaded successfully'); + return config; + } catch (error) { + log.error('❌ Failed to load system configuration:', error); + return null; + } + } + + /** + * Get current system configuration (load from file or validate fresh) + */ + async getSystemConfiguration(forceRefresh = false) { + try { + if (!forceRefresh) { + const existingConfig = await this.loadSystemConfiguration(); + if (existingConfig) { + // Check if config is recent (less than 1 hour old) + const lastUpdated = new Date(existingConfig.lastUpdated); + const hourAgo = new Date(Date.now() - 60 * 60 * 1000); + + if (lastUpdated > hourAgo) { + log.info('📄 Using cached system configuration'); + return existingConfig; + } + } + } + + log.info('🔄 Refreshing system configuration...'); + const validation = await this.validateSystemResources(); + return await this.loadSystemConfiguration(); + } catch (error) { + log.error('❌ Failed to get system configuration:', error); + throw error; + } + } + + /** + * Check if system meets minimum requirements for specific feature + */ + async checkFeatureRequirements(featureName) { + try { + const config = await this.getSystemConfiguration(); + if (!config) { + return { supported: false, reason: 'System configuration not available' }; + } + + const isEnabled = config.enabledFeatures[featureName]; + if (!isEnabled) { + return { + supported: false, + reason: `Feature disabled due to ${config.performanceMode} performance mode`, + performanceMode: config.performanceMode, + recommendations: config.recommendations + }; + } + + return { + supported: true, + performanceMode: config.performanceMode, + limitations: config.resourceLimitations + }; + } catch (error) { + log.error(`❌ Failed to check requirements for feature ${featureName}:`, error); + return { supported: false, reason: 'Failed to check system requirements' }; + } + } + + getPlatformDirectory(platform, arch) { + switch (platform) { + case 'darwin': + return arch === 'arm64' ? 'darwin-arm64' : 'darwin-x64'; + case 'linux': + return arch === 'arm64' ? 'linux-arm64' : 'linux-x64'; + case 'win32': + return 'win32-x64'; + default: + throw new Error(`Unsupported platform: ${platform}-${arch}`); + } + } + + getSupportedPlatforms() { + return { + 'darwin-arm64': { + name: 'macOS (Apple Silicon)', + binaries: ['llama-swap-darwin-arm64', 'llama-server'], + libraries: ['*.dylib'], + headers: ['*.h'], + shaders: ['*.metal'], + supported: true + }, + 'darwin-x64': { + name: 'macOS (Intel)', + binaries: ['llama-swap-darwin-x64', 'llama-server'], + libraries: ['*.dylib'], + headers: ['*.h'], + shaders: ['*.metal'], + supported: false // Will be added in future + }, + 'linux-x64': { + name: 'Linux (x64)', + binaries: ['llama-swap-linux', 'llama-server'], + libraries: ['*.so'], + headers: ['*.h'], + supported: true // Enable Linux support since binaries are working + }, + 'linux-arm64': { + name: 'Linux (ARM64)', + binaries: ['llama-swap-linux-arm64', 'llama-server'], + libraries: ['*.so'], + headers: ['*.h'], + supported: false // Will be added in future + }, + 'win32-x64': { + name: 'Windows (x64)', + binaries: ['llama-swap-win32-x64.exe', 'llama-server.exe'], + libraries: ['*.dll'], + headers: ['*.h'], + supported: false // Will be added in future + } + }; + } + + /** + * Get the binary paths for the current platform + */ + getBinaryPaths() { + const platformBinDir = path.join(this.baseDir, this.platformInfo.platformDir); + const platformConfig = this.supportedPlatforms[this.platformInfo.platformDir]; + + if (!platformConfig) { + throw new Error(`Unsupported platform: ${this.platformInfo.platformDir}`); + } + + const binaryPaths = {}; + + // Map standard binary names to platform-specific names + platformConfig.binaries.forEach(binaryName => { + if (binaryName.includes('llama-swap')) { + binaryPaths.llamaSwap = path.join(platformBinDir, binaryName); + } else if (binaryName.includes('llama-server')) { + binaryPaths.llamaServer = path.join(platformBinDir, binaryName); + } + }); + + return binaryPaths; + } + + /** + * Check if current platform is supported + */ + isCurrentPlatformSupported() { + const platformConfig = this.supportedPlatforms[this.platformInfo.platformDir]; + return platformConfig && platformConfig.supported; + } + + /** + * Validate that all required binaries exist for the current platform + */ + async validatePlatformBinaries() { + if (!this.isCurrentPlatformSupported()) { + throw new Error(`Platform ${this.platformInfo.platformDir} is not yet supported. Supported platforms: ${this.getSupportedPlatformNames().join(', ')}`); + } + + const binaryPaths = this.getBinaryPaths(); + const issues = []; + + for (const [name, binaryPath] of Object.entries(binaryPaths)) { + if (!this.binaryExists(binaryPath)) { + issues.push(`${name} binary not found at: ${binaryPath}`); + } else { + try { + await fs.access(binaryPath, fs.constants.F_OK | fs.constants.X_OK); + } catch (error) { + issues.push(`${name} binary exists but is not executable: ${binaryPath}`); + } + } + } + + if (issues.length > 0) { + const error = new Error(`Platform binary validation failed:\n${issues.join('\n')}`); + error.issues = issues; + throw error; + } + + log.info(`Platform validation successful for ${this.platformInfo.platformDir}`); + return true; + } + + /** + * Check if a binary file exists and is a file + */ + binaryExists(binaryPath) { + try { + return fsSync.existsSync(binaryPath) && fsSync.statSync(binaryPath).isFile(); + } catch (error) { + return false; + } + } + + /** + * Get list of supported platform names + */ + getSupportedPlatformNames() { + return Object.entries(this.supportedPlatforms) + .filter(([_, config]) => config.supported) + .map(([platformDir, config]) => config.name); + } + + /** + * Get platform directory for the current system + */ + getCurrentPlatformDirectory() { + return this.platformInfo.platformDir; + } + + /** + * Get platform-specific library directory + */ + getPlatformLibraryDirectory() { + return path.join(this.baseDir, this.platformInfo.platformDir); + } + + /** + * Get platform-specific environment variables for running binaries + */ + getPlatformEnvironment() { + const platformLibDir = this.getPlatformLibraryDirectory(); + const env = { ...process.env }; + + if (this.platformInfo.isLinux) { + env.LD_LIBRARY_PATH = platformLibDir + ':' + (env.LD_LIBRARY_PATH || ''); + } else if (this.platformInfo.isMac) { + env.DYLD_LIBRARY_PATH = platformLibDir + ':' + (env.DYLD_LIBRARY_PATH || ''); + } + // Windows uses PATH for DLL loading, which should already include the platform directory + + return env; + } + + /** + * Future: Download and install binaries for a specific platform + * This will be implemented when adding JIT compilation support + */ + async downloadPlatformBinaries(platformDir, version = 'latest') { + throw new Error('Binary download functionality not yet implemented. This will support downloading pre-built binaries or compiling from source.'); + } + + /** + * Future: Compile binaries from source (JIT) + * This will be implemented when adding JIT compilation support + */ + async compileBinariesFromSource(options = {}) { + throw new Error('Just-in-time compilation not yet implemented. This will support building llama.cpp from source.'); + } + + /** + * Get platform information for debugging + */ + getPlatformInfo() { + return { + current: this.platformInfo, + supported: this.isCurrentPlatformSupported(), + availablePlatforms: this.getSupportedPlatformNames(), + binaryPaths: this.isCurrentPlatformSupported() ? this.getBinaryPaths() : null + }; + } + + /** + * Check for platform-specific optimizations + */ + getOptimizations() { + const optimizations = { + cpu: [], + gpu: [], + memory: [] + }; + + // Detect CPU features + const cpuFlags = os.cpus()[0]?.flags || []; + if (cpuFlags.includes('avx2')) optimizations.cpu.push('AVX2'); + if (cpuFlags.includes('avx512')) optimizations.cpu.push('AVX-512'); + + // Platform-specific GPU support + if (this.platformInfo.isMac) { + optimizations.gpu.push('Metal'); + } else if (this.platformInfo.isLinux) { + optimizations.gpu.push('CUDA', 'OpenCL'); + } else if (this.platformInfo.isWindows) { + optimizations.gpu.push('CUDA', 'DirectML'); + } + + return optimizations; + } +} + +module.exports = PlatformManager; \ No newline at end of file diff --git a/electron/preload.cjs b/electron/preload.cjs new file mode 100644 index 00000000..86047331 --- /dev/null +++ b/electron/preload.cjs @@ -0,0 +1,541 @@ +const { contextBridge, ipcRenderer, clipboard } = require('electron'); +const path = require('path'); +const fs = require('fs'); +const os = require('os'); +const { app } = require('electron'); + +// Function to safely get app version +function getAppVersion() { + try { + // Read from package.json + const packagePath = path.join(__dirname, '../package.json'); + if (fs.existsSync(packagePath)) { + const packageJson = JSON.parse(fs.readFileSync(packagePath, 'utf8')); + return packageJson.version || 'unknown'; + } + } catch (error) { + console.error('Failed to get app version:', error); + } + return 'unknown'; +} + +// Valid channels for IPC communication +const validChannels = [ + 'app-ready', + 'app-close', + 'update-available', + 'update-downloaded', + 'update-download-progress', + 'update-download-completed', + 'update-download-error', + 'download-progress', + 'llama-progress-update', + 'llama-progress-complete', + 'watchdog-service-restored', + 'watchdog-service-failed', + 'watchdog-service-restarted', + 'docker-update-progress', + 'comfyui-model-download-progress', + 'comfyui-model-download-complete', + 'model-download-progress', + 'trigger-new-chat', + 'hide-to-tray', + 'show-from-tray', + 'app-initialization-state', + 'service-status-update', + 'service-init-progress', + 'comfyui:startup-progress', + 'n8n:startup-progress', + 'python:startup-progress' +]; + +// Add explicit logging for debugging +console.log('Preload script initializing...'); + +contextBridge.exposeInMainWorld('electron', { + // System Info + getAppPath: () => ipcRenderer.invoke('get-app-path'), + getAppVersion: () => app.getVersion(), + getElectronVersion: () => process.versions.electron, + getPlatform: () => process.platform, + isDev: process.env.NODE_ENV === 'development', + + // Permissions + requestMicrophonePermission: () => ipcRenderer.invoke('request-microphone-permission'), + + // Service Info + getServicePorts: () => ipcRenderer.invoke('get-service-ports'), + getPythonPort: () => ipcRenderer.invoke('get-python-port'), + checkPythonBackend: () => ipcRenderer.invoke('check-python-backend'), + checkDockerServices: () => ipcRenderer.invoke('check-docker-services'), + getPythonBackendInfo: () => ipcRenderer.invoke('get-python-backend-info'), + startDockerService: (serviceName) => ipcRenderer.invoke('start-docker-service', serviceName), + stopDockerService: (serviceName) => ipcRenderer.invoke('stop-docker-service', serviceName), + restartDockerService: (serviceName) => ipcRenderer.invoke('restart-docker-service', serviceName), + + // Docker Container Updates + checkDockerUpdates: () => ipcRenderer.invoke('docker-check-updates'), + updateDockerContainers: (containerNames) => ipcRenderer.invoke('docker-update-containers', containerNames), + getSystemInfo: () => ipcRenderer.invoke('docker-get-system-info'), + + // Updates + checkForUpdates: () => ipcRenderer.invoke('check-for-updates'), + getUpdateInfo: () => ipcRenderer.invoke('get-update-info'), + startInAppDownload: (updateInfo) => ipcRenderer.invoke('start-in-app-download', updateInfo), + + // IPC Event Listeners for updates + on: (channel, callback) => { + if (validChannels.includes(channel)) { + ipcRenderer.on(channel, callback); + } + }, + removeAllListeners: (channel) => { + if (validChannels.includes(channel)) { + ipcRenderer.removeAllListeners(channel); + } + }, + + // Llama.cpp Binary Updates + checkLlamacppUpdates: () => ipcRenderer.invoke('check-llamacpp-updates'), + updateLlamacppBinaries: () => ipcRenderer.invoke('update-llamacpp-binaries'), + + // Clipboard + clipboard: { + writeText: (text) => clipboard.writeText(text), + readText: () => clipboard.readText(), + }, + + // IPC Communication + send: (channel, data) => { + if (validChannels.includes(channel)) { + ipcRenderer.send(channel, data); + } + }, + sendReactReady: () => { + ipcRenderer.send('react-app-ready'); + }, + receive: (channel, callback) => { + if (validChannels.includes(channel)) { + ipcRenderer.on(channel, (event, ...args) => callback(...args)); + } + }, + removeListener: (channel, callback) => { + if (validChannels.includes(channel)) { + ipcRenderer.removeListener(channel, callback); + } + }, + removeAllListeners: (channel) => { + if (validChannels.includes(channel)) { + ipcRenderer.removeAllListeners(channel); + } + }, + getWorkflowsPath: () => ipcRenderer.invoke('get-workflows-path'), + dialog: { + showOpenDialog: (options) => ipcRenderer.invoke('show-open-dialog', options) + }, + + // Add tray functionality + hideToTray: () => ipcRenderer.send('hide-to-tray'), + showFromTray: () => ipcRenderer.send('show-from-tray'), + + // DEPRECATED: Legacy startup settings APIs (use startupSettings instead) + setStartupSettings: (settings) => ipcRenderer.invoke('set-startup-settings', settings), + getStartupSettings: () => ipcRenderer.invoke('get-startup-settings'), + + // NEW: Isolated startup settings API with consent management + startupSettings: { + get: () => ipcRenderer.invoke('startup-settings:get'), + update: (settings, userConsent = false) => ipcRenderer.invoke('startup-settings:update', settings, userConsent), + validate: (frontendChecksum) => ipcRenderer.invoke('startup-settings:validate', frontendChecksum), + reset: (confirmed = false) => ipcRenderer.invoke('startup-settings:reset', confirmed), + getFileStatus: () => ipcRenderer.invoke('startup-settings:get-file-status') + }, + + // Fast startup APIs for dashboard + getInitializationState: () => ipcRenderer.invoke('get-initialization-state'), + saveFeatureSelection: (features) => ipcRenderer.invoke('save-feature-selection', features), + initializeService: (serviceName) => ipcRenderer.invoke('initialize-service', serviceName), + + // electron-store API for persistent configuration + store: { + get: (key) => ipcRenderer.invoke('store:get', key), + set: (key, value) => ipcRenderer.invoke('store:set', key, value), + delete: (key) => ipcRenderer.invoke('store:delete', key), + has: (key) => ipcRenderer.invoke('store:has', key), + clear: () => ipcRenderer.invoke('store:clear') + }, + + // Netlify OAuth + netlifyOAuth: { + authenticate: (authUrl) => ipcRenderer.invoke('netlify-oauth:authenticate', authUrl), + cancel: () => ipcRenderer.invoke('netlify-oauth:cancel') + } +}); + +// Add Docker container management API +contextBridge.exposeInMainWorld('electronAPI', { + getContainers: () => ipcRenderer.invoke('get-containers'), + containerAction: (containerId, action) => + ipcRenderer.invoke('container-action', { containerId, action }), + createContainer: (containerConfig) => + ipcRenderer.invoke('create-container', containerConfig), + getContainerStats: (containerId) => + ipcRenderer.invoke('get-container-stats', containerId), + getContainerLogs: (containerId) => + ipcRenderer.invoke('get-container-logs', containerId), + pullImage: (image) => ipcRenderer.invoke('pull-image', image), + createNetwork: (networkConfig) => ipcRenderer.invoke('create-network', networkConfig), + listNetworks: () => ipcRenderer.invoke('list-networks'), + removeNetwork: (networkId) => ipcRenderer.invoke('remove-network', networkId), + getImages: () => ipcRenderer.invoke('get-images'), + removeImage: (imageId) => ipcRenderer.invoke('remove-image', imageId), + pruneContainers: () => ipcRenderer.invoke('prune-containers'), + pruneImages: () => ipcRenderer.invoke('prune-images'), + getDockerInfo: () => ipcRenderer.invoke('get-docker-info'), + getDockerVersion: () => ipcRenderer.invoke('get-docker-version'), + + // ComfyUI specific API + comfyuiStatus: () => ipcRenderer.invoke('comfyui-status'), + comfyuiStart: () => ipcRenderer.invoke('comfyui-start'), + comfyuiStop: () => ipcRenderer.invoke('comfyui-stop'), + comfyuiRestart: () => ipcRenderer.invoke('comfyui-restart'), + comfyuiLogs: () => ipcRenderer.invoke('comfyui-logs'), + comfyuiOptimize: () => ipcRenderer.invoke('comfyui-optimize'), + + // System information methods + getPlatform: () => process.platform, + getSystemInfo: () => ipcRenderer.invoke('get-system-info'), + saveComfyUIConsent: (hasConsented) => ipcRenderer.invoke('save-comfyui-consent', hasConsented), + getComfyUIConsent: () => ipcRenderer.invoke('get-comfyui-consent'), + createUserConsentFile: (consentData) => ipcRenderer.invoke('createUserConsentFile', consentData), + getGPUInfo: () => ipcRenderer.invoke('get-gpu-info'), + + // Services status API + getServicesStatus: () => ipcRenderer.invoke('get-services-status'), + + // Watchdog service API + invoke: (channel, ...args) => ipcRenderer.invoke(channel, ...args), + on: (channel, callback) => { + const subscription = (event, ...args) => callback(event, ...args); + ipcRenderer.on(channel, subscription); + return () => ipcRenderer.removeListener(channel, subscription); + }, + off: (channel, callback) => ipcRenderer.removeListener(channel, callback), + removeAllListeners: (channel) => ipcRenderer.removeAllListeners(channel), + + // Service status update listener + onServiceStatusUpdate: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('service-status-update', subscription); + return () => ipcRenderer.removeListener('service-status-update', subscription); + }, + + // Request initialization + requestInitialization: () => ipcRenderer.invoke('request-initialization'), + + // Get initialization status + getInitializationStatus: () => ipcRenderer.invoke('get-initialization-status'), + + // Get Python Backend URL (for dynamic service URL resolution) + getPythonBackendUrl: () => ipcRenderer.invoke('python-backend:get-url') +}); + +// Add llama-swap service API - REMOVED (service deprecated) +/* +contextBridge.exposeInMainWorld('llamaSwap', { + start: () => ipcRenderer.invoke('start-llama-swap'), + stop: () => ipcRenderer.invoke('stop-llama-swap'), + restart: () => ipcRenderer.invoke('restart-llama-swap'), + getStatus: () => ipcRenderer.invoke('get-llama-swap-status'), + getStatusWithHealth: () => ipcRenderer.invoke('get-llama-swap-status-with-health'), + getModels: () => ipcRenderer.invoke('get-llama-swap-models'), + getApiUrl: () => ipcRenderer.invoke('get-llama-swap-api-url'), + regenerateConfig: () => ipcRenderer.invoke('regenerate-llama-swap-config'), + debugBinaryPaths: () => ipcRenderer.invoke('debug-binary-paths'), + getGPUDiagnostics: () => ipcRenderer.invoke('get-gpu-diagnostics'), + getPerformanceSettings: () => ipcRenderer.invoke('get-performance-settings'), + savePerformanceSettings: (settings) => ipcRenderer.invoke('save-performance-settings', settings), + loadPerformanceSettings: () => ipcRenderer.invoke('load-performance-settings'), + setCustomModelPath: (path) => ipcRenderer.invoke('set-custom-model-path', path), + getCustomModelPaths: () => ipcRenderer.invoke('get-custom-model-paths'), + scanCustomPathModels: (path) => ipcRenderer.invoke('scan-custom-path-models', path), + getModelEmbeddingInfo: (modelPath) => ipcRenderer.invoke('get-model-embedding-info', modelPath), + searchHuggingFaceMmproj: (modelName, embeddingSize) => ipcRenderer.invoke('search-huggingface-mmproj', modelName, embeddingSize), + + // Official llama-swap binary management + downloadOfficialBinary: () => ipcRenderer.invoke('download-official-llama-swap'), + checkForUpdates: () => ipcRenderer.invoke('check-llama-swap-updates'), + updateToLatest: () => ipcRenderer.invoke('update-llama-swap'), + getVersion: () => ipcRenderer.invoke('get-llama-swap-version'), + + // NEW: Configuration override API methods + getAvailableBackends: () => ipcRenderer.invoke('get-available-backends'), + setBackendOverride: (backendId) => ipcRenderer.invoke('set-backend-override', backendId), + getBackendOverride: () => ipcRenderer.invoke('get-backend-override'), + getConfigAsJson: () => ipcRenderer.invoke('get-config-as-json'), + forceReconfigure: () => ipcRenderer.invoke('force-reconfigure'), + getConfigurationInfo: () => ipcRenderer.invoke('get-configuration-info'), + restartWithOverrides: () => ipcRenderer.invoke('restart-llamaswap-with-overrides'), + + // NEW: Configuration saving and management + saveConfigFromJson: (jsonConfig) => ipcRenderer.invoke('save-config-from-json', jsonConfig), + + // NEW: LLaMA Optimizer + runLlamaOptimizer: (preset) => ipcRenderer.invoke('run-llama-optimizer', preset), + saveConfigAndRestart: (jsonConfig) => ipcRenderer.invoke('save-config-and-restart', jsonConfig), + regenerateConfig: () => ipcRenderer.invoke('regenerate-config'), + + // NEW: Model configuration management + getModelConfigurations: () => ipcRenderer.invoke('get-model-configurations'), + saveModelConfiguration: (modelName, modelConfig) => ipcRenderer.invoke('save-model-configuration', modelName, modelConfig), + saveAllModelConfigurations: (modelConfigs) => ipcRenderer.invoke('save-all-model-configurations', modelConfigs) +}); +*/ + +// Add model management API +contextBridge.exposeInMainWorld('modelManager', { + searchHuggingFaceModels: (query, limit, sort) => ipcRenderer.invoke('search-huggingface-models', { query, limit, sort }), + downloadModel: (modelId, fileName, downloadPath) => ipcRenderer.invoke('download-huggingface-model', { modelId, fileName, downloadPath }), + downloadModelWithCustomName: (modelId, fileName, customSaveName, downloadPath) => ipcRenderer.invoke('download-huggingface-model-with-custom-name', { modelId, fileName, customSaveName, downloadPath }), + downloadModelWithDependencies: (modelId, fileName, allFiles, downloadPath) => ipcRenderer.invoke('download-model-with-dependencies', { modelId, fileName, allFiles, downloadPath }), + getLocalModels: () => ipcRenderer.invoke('get-local-models'), + deleteLocalModel: (filePath) => ipcRenderer.invoke('delete-local-model', { filePath }), + stopDownload: (fileName) => ipcRenderer.invoke('stop-download', { fileName }), + + // LlamaSwap service restart for applying mmproj configuration changes + restartLlamaSwap: () => ipcRenderer.invoke('restart-llamaswap'), + + // Mmproj mapping persistence + saveMmprojMappings: (mappings) => ipcRenderer.invoke('save-mmproj-mappings', mappings), + loadMmprojMappings: () => ipcRenderer.invoke('load-mmproj-mappings'), + getAvailableMmprojFiles: () => ipcRenderer.invoke('get-available-mmproj-files'), + + // Listen for download progress updates + onDownloadProgress: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('download-progress', subscription); + return () => ipcRenderer.removeListener('download-progress', subscription); + }, + + // Model Manager APIs + searchCivitAI: (query, types, sort, apiKey, nsfw) => ipcRenderer.invoke('model-manager:search-civitai', { query, types, sort, apiKey, nsfw }), + searchHuggingFace: (query, modelType, author) => ipcRenderer.invoke('model-manager:search-huggingface', { query, modelType, author }), + downloadModelFile: (url, filename, modelType, source) => ipcRenderer.invoke('model-manager:download-model', { url, filename, modelType, source }), + getLocalModelFiles: () => ipcRenderer.invoke('model-manager:get-local-models'), + deleteLocalModelFile: (modelType, filename) => ipcRenderer.invoke('model-manager:delete-local-model', { modelType, filename }), + saveApiKeys: (keys) => ipcRenderer.invoke('model-manager:save-api-keys', keys), + getApiKeys: () => ipcRenderer.invoke('model-manager:get-api-keys'), + + // ComfyUI Model Manager APIs (Host-based) + comfyuiDownloadModel: (url, filename, modelType, source, apiKey) => + ipcRenderer.invoke('comfyui-model-manager:download-model', { url, filename, modelType, source, apiKey }), + comfyuiGetLocalModels: () => ipcRenderer.invoke('comfyui-model-manager:get-local-models'), + comfyuiDeleteModel: (modelType, filename) => ipcRenderer.invoke('comfyui-model-manager:delete-model', { modelType, filename }), + comfyuiGetModelsDir: () => ipcRenderer.invoke('comfyui-model-manager:get-models-dir'), + + // ComfyUI Internal Model Manager APIs (Container-based) + comfyuiInternalListModels: (category) => ipcRenderer.invoke('comfyui-internal:list-models', category), + comfyuiInternalGetStorageInfo: () => ipcRenderer.invoke('comfyui-internal:get-storage-info'), + comfyuiInternalDownloadModel: (url, filename, category) => + ipcRenderer.invoke('comfyui-internal:download-model', { url, filename, category }), + comfyuiInternalRemoveModel: (filename, category) => + ipcRenderer.invoke('comfyui-internal:remove-model', { filename, category }), + comfyuiInternalGetStatus: () => ipcRenderer.invoke('comfyui-internal:get-status'), + comfyuiInternalSearchModels: (query, source, category) => + ipcRenderer.invoke('comfyui-internal:search-models', { query, source, category }), + comfyuiInternalBackupModels: (category, backupPath) => + ipcRenderer.invoke('comfyui-internal:backup-models', { category, backupPath }), + + // ComfyUI Download Progress Events (Host-based) + onComfyUIDownloadProgress: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('comfyui-model-download-progress', subscription); + return () => ipcRenderer.removeListener('comfyui-model-download-progress', subscription); + }, + onComfyUIDownloadComplete: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('comfyui-model-download-complete', subscription); + return () => ipcRenderer.removeListener('comfyui-model-download-complete', subscription); + }, + + // ComfyUI Internal Download Progress Events (Container-based) + onComfyUIInternalDownloadProgress: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('comfyui-internal-download-progress', subscription); + return () => ipcRenderer.removeListener('comfyui-internal-download-progress', subscription); + }, + onComfyUIInternalDownloadStart: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('comfyui-internal-download-start', subscription); + return () => ipcRenderer.removeListener('comfyui-internal-download-start', subscription); + }, + onComfyUIInternalDownloadComplete: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('comfyui-internal-download-complete', subscription); + return () => ipcRenderer.removeListener('comfyui-internal-download-complete', subscription); + }, + onComfyUIInternalDownloadError: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('comfyui-internal-download-error', subscription); + return () => ipcRenderer.removeListener('comfyui-internal-download-error', subscription); + }, + onComfyUIInternalInstallStart: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('comfyui-internal-install-start', subscription); + return () => ipcRenderer.removeListener('comfyui-internal-install-start', subscription); + }, + onComfyUIInternalInstallComplete: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('comfyui-internal-install-complete', subscription); + return () => ipcRenderer.removeListener('comfyui-internal-install-complete', subscription); + }, + onComfyUIInternalInstallError: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('comfyui-internal-install-error', subscription); + return () => ipcRenderer.removeListener('comfyui-internal-install-error', subscription); + }, + + // ============================================== + // Enhanced Local Model Management APIs + // ============================================== + + // Local persistent model management + comfyuiLocalListModels: (category) => ipcRenderer.invoke('comfyui-local:list-models', category), + comfyuiLocalDownloadModel: (url, filename, category, apiKey, source) => + ipcRenderer.invoke('comfyui-local:download-model', { url, filename, category, apiKey, source }), + comfyuiLocalDeleteModel: (filename, category) => + ipcRenderer.invoke('comfyui-local:delete-model', { filename, category }), + comfyuiLocalImportModel: (externalPath, filename, category) => + ipcRenderer.invoke('comfyui-local:import-model', { externalPath, filename, category }), + comfyuiLocalGetStorageInfo: () => ipcRenderer.invoke('comfyui-local:get-storage-info'), + + // Local model management events + onComfyUILocalDownloadProgress: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('comfyui-local-download-progress', subscription); + return () => ipcRenderer.removeListener('comfyui-local-download-progress', subscription); + }, + onComfyUILocalDownloadComplete: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('comfyui-local-download-complete', subscription); + return () => ipcRenderer.removeListener('comfyui-local-download-complete', subscription); + }, + onComfyUILocalDownloadError: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('comfyui-local-download-error', subscription); + return () => ipcRenderer.removeListener('comfyui-local-download-error', subscription); + }, + onModelDownloadProgress: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('model-download-progress', subscription); + return () => ipcRenderer.removeListener('model-download-progress', subscription); + } +}); + +// Add ClaraCore Service API +contextBridge.exposeInMainWorld('claraCore', { + // Local binary mode + start: () => ipcRenderer.invoke('claracore-start'), + stop: () => ipcRenderer.invoke('claracore-stop'), + restart: () => ipcRenderer.invoke('claracore-restart'), + getStatus: () => ipcRenderer.invoke('claracore-status'), + + // Docker mode + startDocker: (options) => ipcRenderer.invoke('claracore-docker-start', options), + stopDocker: () => ipcRenderer.invoke('claracore-docker-stop'), + restartDocker: () => ipcRenderer.invoke('claracore-docker-restart'), + getDockerStatus: () => ipcRenderer.invoke('claracore-docker-status'), + detectGPU: () => ipcRenderer.invoke('claracore-docker-detect-gpu'), + removeContainer: () => ipcRenderer.invoke('claracore-docker-remove'), + getLogs: (options) => ipcRenderer.invoke('claracore-docker-logs', options) +}); + +// Add MCP service API +contextBridge.exposeInMainWorld('mcpService', { + getServers: () => ipcRenderer.invoke('mcp-get-servers'), + addServer: (serverConfig) => ipcRenderer.invoke('mcp-add-server', serverConfig), + removeServer: (name) => ipcRenderer.invoke('mcp-remove-server', name), + updateServer: (name, updates) => ipcRenderer.invoke('mcp-update-server', name, updates), + startServer: (name) => ipcRenderer.invoke('mcp-start-server', name), + stopServer: (name) => ipcRenderer.invoke('mcp-stop-server', name), + restartServer: (name) => ipcRenderer.invoke('mcp-restart-server', name), + getServerStatus: (name) => ipcRenderer.invoke('mcp-get-server-status', name), + testServer: (name) => ipcRenderer.invoke('mcp-test-server', name), + getTemplates: () => ipcRenderer.invoke('mcp-get-templates'), + startAllEnabled: () => ipcRenderer.invoke('mcp-start-all-enabled'), + stopAll: () => ipcRenderer.invoke('mcp-stop-all'), + startPreviouslyRunning: () => ipcRenderer.invoke('mcp-start-previously-running'), + saveRunningState: () => ipcRenderer.invoke('mcp-save-running-state'), + importClaudeConfig: (configPath) => ipcRenderer.invoke('mcp-import-claude-config', configPath), + executeToolCall: (toolCall) => ipcRenderer.invoke('mcp-execute-tool', toolCall), + diagnoseNode: () => ipcRenderer.invoke('mcp-diagnose-node') +}); + +// Add window management API +contextBridge.exposeInMainWorld('windowManager', { + getFullscreenStartupPreference: () => ipcRenderer.invoke('get-fullscreen-startup-preference'), + setFullscreenStartupPreference: (enabled) => ipcRenderer.invoke('set-fullscreen-startup-preference', enabled), + toggleFullscreen: () => ipcRenderer.invoke('toggle-fullscreen'), + getFullscreenStatus: () => ipcRenderer.invoke('get-fullscreen-status') +}); + +// Add feature configuration API +contextBridge.exposeInMainWorld('featureConfig', { + getFeatureConfig: () => ipcRenderer.invoke('get-feature-config'), + updateFeatureConfig: (config) => ipcRenderer.invoke('update-feature-config', config), + resetFeatureConfig: () => ipcRenderer.invoke('reset-feature-config') +}); + +// Add developer logs API +contextBridge.exposeInMainWorld('developerLogs', { + readLogs: (lines = 1000) => ipcRenderer.invoke('developer-logs:read', lines), + getLogFiles: () => ipcRenderer.invoke('developer-logs:get-files'), + clearLogs: () => ipcRenderer.invoke('developer-logs:clear') +}); + +// Add screen sharing API for Electron +contextBridge.exposeInMainWorld('electronScreenShare', { + getDesktopSources: () => ipcRenderer.invoke('get-desktop-sources'), + getScreenAccessStatus: () => ipcRenderer.invoke('get-screen-access-status'), + requestScreenAccess: () => ipcRenderer.invoke('request-screen-access') +}); + +// Add remote server management API +contextBridge.exposeInMainWorld('remoteServer', { + testConnection: (config) => ipcRenderer.invoke('remote-server:test-connection', config), + deploy: (config) => ipcRenderer.invoke('remote-server:deploy', config), + stopService: (config, serviceName) => ipcRenderer.invoke('remote-server:stop-service', { config, serviceName }), + + // Listen for deployment logs + onLog: (callback) => { + const subscription = (event, data) => callback(data); + ipcRenderer.on('remote-server:log', subscription); + return () => ipcRenderer.removeListener('remote-server:log', subscription); + } +}); + +// Add ClaraCore remote deployment API +contextBridge.exposeInMainWorld('claraCoreRemote', { + testSetup: (config) => ipcRenderer.invoke('claracore-remote-test-setup', config), + deploy: (config) => ipcRenderer.invoke('claracore-remote-deploy', config), + monitor: (config) => ipcRenderer.invoke('claracore-remote:monitor', config) +}); + +// Add unified service configuration API +contextBridge.exposeInMainWorld('serviceConfig', { + // Get enhanced service status from central service manager + getEnhancedStatus: () => ipcRenderer.invoke('service-config:get-enhanced-status'), + // Get service configuration (mode, URL) + getServiceConfig: (serviceName) => ipcRenderer.invoke('service-config:get-config', serviceName), + // Update service configuration + updateServiceConfig: (serviceName, mode, url) => ipcRenderer.invoke('service-config:update', serviceName, mode, url), + // Test service connectivity + testService: (serviceName, url) => ipcRenderer.invoke('service-config:test-service', serviceName, url), + // Reset service to defaults + resetService: (serviceName) => ipcRenderer.invoke('service-config:reset', serviceName) +}); + +// Notify main process when preload script has loaded +window.addEventListener('DOMContentLoaded', () => { + ipcRenderer.send('app-ready', 'Preload script has loaded'); +}); \ No newline at end of file diff --git a/electron/remoteServerIPC.cjs b/electron/remoteServerIPC.cjs new file mode 100644 index 00000000..1fdfb2a1 --- /dev/null +++ b/electron/remoteServerIPC.cjs @@ -0,0 +1,85 @@ +const { ipcMain } = require('electron'); +const RemoteServerService = require('./remoteServerService.cjs'); +const log = require('electron-log'); + +const remoteServerService = new RemoteServerService(); + +/** + * Setup IPC handlers for remote server management + * @param {Electron.BrowserWindow} mainWindow - The main window + * @param {Function} stopLocalServicesCallback - Callback to stop local services + */ +function setupRemoteServerIPC(mainWindow, stopLocalServicesCallback = null) { + log.info('[RemoteServerIPC] Setting up IPC handlers'); + + // Test SSH connection + ipcMain.handle('remote-server:test-connection', async (event, config) => { + log.info('[RemoteServerIPC] Testing connection to:', config.host); + try { + const result = await remoteServerService.testConnection(config); + return result; + } catch (error) { + log.error('[RemoteServerIPC] Connection test error:', error); + return { + success: false, + error: error.message + }; + } + }); + + // Deploy backend to remote server + ipcMain.handle('remote-server:deploy', async (event, config) => { + log.info('[RemoteServerIPC] Starting deployment to:', config.host); + + try { + // Stop all local services before deploying to remote + if (stopLocalServicesCallback) { + log.info('[RemoteServerIPC] 🛑 Stopping all local services before remote deployment...'); + + const stopResult = await stopLocalServicesCallback(); + + if (stopResult.stopped.length > 0) { + log.info(`[RemoteServerIPC] ✅ Stopped: ${stopResult.stopped.join(', ')}`); + } + if (stopResult.errors.length > 0) { + log.warn(`[RemoteServerIPC] ⚠️ Some services had errors during stop: ${JSON.stringify(stopResult.errors)}`); + } + } + + const webContents = event.sender; + const result = await remoteServerService.deploy(config, webContents); + + if (result.success) { + log.info('[RemoteServerIPC] ✅ Successfully deployed to remote server'); + } + + return result; + } catch (error) { + log.error('[RemoteServerIPC] Deployment error:', error); + return { + success: false, + error: error.message + }; + } + }); + + // Stop a service + ipcMain.handle('remote-server:stop-service', async (event, { config, serviceName }) => { + log.info('[RemoteServerIPC] Stopping service:', serviceName); + + try { + const result = await remoteServerService.stopService(config, serviceName); + return result; + } catch (error) { + log.error('[RemoteServerIPC] Stop service error:', error); + return { + success: false, + error: error.message + }; + } + }); + + log.info('[RemoteServerIPC] IPC handlers registered'); +} + +module.exports = { setupRemoteServerIPC }; diff --git a/electron/remoteServerService.cjs b/electron/remoteServerService.cjs new file mode 100644 index 00000000..f316d0b0 --- /dev/null +++ b/electron/remoteServerService.cjs @@ -0,0 +1,445 @@ +const { NodeSSH } = require('node-ssh'); +const log = require('electron-log'); + +class RemoteServerService { + constructor() { + this.ssh = null; + } + + /** + * Send log to renderer process + */ + sendLog(webContents, type, message, step = null) { + log.info(`[RemoteServer] ${message}`); + if (webContents && !webContents.isDestroyed()) { + webContents.send('remote-server:log', { + type, + message, + step, + timestamp: new Date().toISOString() + }); + } + } + + /** + * Test SSH connection + */ + async testConnection(config) { + const ssh = new NodeSSH(); + + try { + log.info(`Testing connection to ${config.host}...`); + + await ssh.connect({ + host: config.host, + port: config.port || 22, + username: config.username, + password: config.password, + tryKeyboard: true, + readyTimeout: 30000 + }); + + // Get OS info + const osResult = await ssh.execCommand('uname -a'); + const osInfo = osResult.stdout; + + // Check Docker + const dockerResult = await ssh.execCommand('docker --version'); + const dockerVersion = dockerResult.code === 0 ? dockerResult.stdout : null; + + // Check for running services by checking actual ports (works for Docker, bare metal, PM2, anything!) + const runningServices = {}; + + // Check ComfyUI on port 8188 + const comfyuiCheck = await ssh.execCommand('curl -s -f -o /dev/null -w "%{http_code}" http://localhost:8188 --connect-timeout 2 --max-time 3'); + if (comfyuiCheck.code === 0 && comfyuiCheck.stdout && comfyuiCheck.stdout.trim() !== '000') { + runningServices.comfyui = { + running: true, + url: `http://${config.host}:8188`, + port: 8188, + httpStatus: comfyuiCheck.stdout.trim() + }; + } + + // Check Python Backend on port 5001 + const pythonCheck = await ssh.execCommand('curl -s -f -o /dev/null -w "%{http_code}" http://localhost:5001 --connect-timeout 2 --max-time 3'); + if (pythonCheck.code === 0 && pythonCheck.stdout && pythonCheck.stdout.trim() !== '000') { + runningServices.python = { + running: true, + url: `http://${config.host}:5001`, + port: 5001, + httpStatus: pythonCheck.stdout.trim() + }; + } + + // Check N8N on port 5678 + const n8nCheck = await ssh.execCommand('curl -s -f -o /dev/null -w "%{http_code}" http://localhost:5678 --connect-timeout 2 --max-time 3'); + if (n8nCheck.code === 0 && n8nCheck.stdout && n8nCheck.stdout.trim() !== '000') { + runningServices.n8n = { + running: true, + url: `http://${config.host}:5678`, + port: 5678, + httpStatus: n8nCheck.stdout.trim() + }; + } + + ssh.dispose(); + + return { + success: true, + osInfo, + dockerVersion, + runningServices + }; + } catch (error) { + log.error('Connection test failed:', error); + if (ssh) ssh.dispose(); + return { + success: false, + error: error.message + }; + } + } + + /** + * Deploy services to remote server + */ + async deploy(config, webContents) { + this.ssh = new NodeSSH(); + + try { + // Step 1: Connect + this.sendLog(webContents, 'info', `Connecting to ${config.host}...`, 'connecting'); + + await this.ssh.connect({ + host: config.host, + port: config.port || 22, + username: config.username, + password: config.password, + tryKeyboard: true, + readyTimeout: 30000 + }); + + this.sendLog(webContents, 'success', '✓ Connected successfully', 'checking-docker'); // Move to next step + + // Step 2: Check Docker + this.sendLog(webContents, 'info', 'Checking Docker installation...'); + + const dockerCheck = await this.ssh.execCommand('docker --version'); + if (dockerCheck.code !== 0) { + throw new Error('Docker not found. Please install Docker on the remote server.'); + } + + this.sendLog(webContents, 'success', `✓ Docker found: ${dockerCheck.stdout}`); + + // Check if Docker daemon is running + const dockerPs = await this.ssh.execCommand('docker ps'); + if (dockerPs.code !== 0) { + throw new Error('Docker daemon not running. Please start Docker: sudo systemctl start docker'); + } + + this.sendLog(webContents, 'success', '✓ Docker daemon is running'); + + // Step 2.5: Ensure clara_network exists and get gateway IP + this.sendLog(webContents, 'info', 'Setting up Clara network...'); + + const networkCheck = await this.ssh.execCommand('docker network ls --filter name=clara_network --format "{{.Name}}"'); + if (!networkCheck.stdout || !networkCheck.stdout.includes('clara_network')) { + // Create clara_network with same configuration as docker-compose + const createNetwork = await this.ssh.execCommand('docker network create clara_network --driver bridge --subnet 172.25.0.0/16'); + if (createNetwork.code === 0) { + this.sendLog(webContents, 'success', '✓ Clara network created'); + } else { + this.sendLog(webContents, 'warning', '⚠ Failed to create network, using existing or default'); + } + } else { + this.sendLog(webContents, 'success', '✓ Clara network exists'); + } + + // Get clara_network gateway IP for host.docker.internal mapping + const gatewayResult = await this.ssh.execCommand('docker network inspect clara_network --format "{{range .IPAM.Config}}{{.Gateway}}{{end}}"'); + const claraNetworkGateway = gatewayResult.stdout ? gatewayResult.stdout.trim() : '172.25.0.1'; + this.sendLog(webContents, 'info', `Clara network gateway: ${claraNetworkGateway}`); + + // Store gateway for use in container deployment + this.claraNetworkGateway = claraNetworkGateway; + + // Step 3: Check for existing containers + const servicesToDeploy = Object.entries(config.services) + .filter(([_, enabled]) => enabled) + .map(([name, _]) => name); + + const existingServices = {}; + + this.sendLog(webContents, 'info', 'Checking for existing containers...'); + + for (const service of servicesToDeploy) { + const containerName = `clara_${service}`; + const checkResult = await this.ssh.execCommand(`docker ps -a --filter name=${containerName} --format "{{.Status}}"`); + + if (checkResult.stdout) { + existingServices[service] = checkResult.stdout; + if (checkResult.stdout.includes('Up')) { + this.sendLog(webContents, 'warning', ` ⚠ ${service} is already running, stopping it first...`); + } else { + this.sendLog(webContents, 'info', ` → ${service} container exists but stopped, removing it...`); + } + + // Stop and remove existing container + await this.ssh.execCommand(`docker stop ${containerName} 2>/dev/null || true`); + await this.ssh.execCommand(`docker rm ${containerName} 2>/dev/null || true`); + } + } + + this.sendLog(webContents, 'success', '✓ Ready to deploy'); + + // Step 4: Pull and deploy containers + this.sendLog(webContents, 'info', 'Pulling container images...', 'pulling-images'); + + const deployedServices = {}; + let isFirstDeploy = true; + + for (const service of servicesToDeploy) { + // Mark pulling-images as complete on first service deploy + if (isFirstDeploy) { + this.sendLog(webContents, 'info', `Deploying ${service}...`, 'deploying'); + isFirstDeploy = false; + } else { + this.sendLog(webContents, 'info', `Deploying ${service}...`); + } + + const deployment = await this.deployService(service, webContents, config); + if (deployment.success) { + deployedServices[service] = { + url: `http://${config.host}:${deployment.port}`, + port: deployment.port, + containerId: deployment.containerId + }; + this.sendLog(webContents, 'success', `✓ ${service} deployed on port ${deployment.port}`); + } else { + this.sendLog(webContents, 'warning', `⚠ ${service} failed: ${deployment.error}`); + } + } + + // Step 5: Verify deployment + this.sendLog(webContents, 'info', 'Verifying deployment...', 'verifying'); + + await new Promise(resolve => setTimeout(resolve, 3000)); // Wait for containers to start + + for (const [service, info] of Object.entries(deployedServices)) { + const status = await this.ssh.execCommand(`docker ps --filter name=clara_${service} --format "{{.Status}}"`); + if (status.stdout.includes('Up')) { + this.sendLog(webContents, 'success', `✓ ${service} is running`); + } else { + this.sendLog(webContents, 'warning', `⚠ ${service} may not be running properly`); + } + } + + this.ssh.dispose(); + this.ssh = null; + + return { + success: true, + services: deployedServices + }; + + } catch (error) { + log.error('Deployment failed:', error); + this.sendLog(webContents, 'error', `✗ Error: ${error.message}`, 'error'); + + if (this.ssh) { + this.ssh.dispose(); + this.ssh = null; + } + + return { + success: false, + error: error.message + }; + } + } + + /** + * Deploy individual service + */ + async deployService(serviceName, webContents, deployConfig = {}) { + const serviceConfigs = { + comfyui: { + image: 'clara17verse/clara-comfyui:with-custom-nodes', + port: 8188, + environment: [ + 'NVIDIA_VISIBLE_DEVICES=all', + 'CUDA_VISIBLE_DEVICES=0' + ], + runtime: '--gpus all', // Will fail gracefully on non-GPU systems + volumes: [ + 'clara_comfyui_models:/app/ComfyUI/models', + 'clara_comfyui_output:/app/ComfyUI/output', + 'clara_comfyui_input:/app/ComfyUI/input', + 'clara_comfyui_custom_nodes:/app/ComfyUI/custom_nodes', + 'clara_comfyui_temp:/app/ComfyUI/temp', + 'clara_comfyui_user:/app/ComfyUI/user' + ] + }, + python: { + image: 'clara17verse/clara-backend:latest', + port: 5001, // External port for reference (not used in host network mode) + environment: [ + 'PYTHONUNBUFFERED=1', + 'PORT=5000', // Host network mode - container binds to port 5000 + 'HOST=0.0.0.0' + ], + runtime: '', + volumes: [ + 'clara_python_data:/home/clara', + 'clara_python_models:/app/models' + ] + }, + n8n: { + image: 'n8nio/n8n:latest', + port: 5678, + environment: [ + 'N8N_BASIC_AUTH_ACTIVE=true', + 'N8N_BASIC_AUTH_USER=admin', + 'N8N_BASIC_AUTH_PASSWORD=clara123', + 'N8N_HOST=0.0.0.0', + 'N8N_PORT=5678', + 'N8N_PROTOCOL=http' + // N8N_SECURE_COOKIE will be added conditionally based on user choice + ], + runtime: '', + volumes: [ + 'clara_n8n_data:/home/node/.n8n' + ] + } + }; + + const config = serviceConfigs[serviceName]; + if (!config) { + return { success: false, error: 'Unknown service' }; + } + + // Handle N8N secure cookie configuration + if (serviceName === 'n8n') { + log.info(`[RemoteServer] N8N deployment - n8nSecureCookie setting: ${deployConfig.n8nSecureCookie}`); + + // If n8nSecureCookie is explicitly false, disable secure cookies for HTTP + // If n8nSecureCookie is true or undefined, use default (secure cookies enabled) + if (deployConfig.n8nSecureCookie === false) { + config.environment.push('N8N_SECURE_COOKIE=false'); + this.sendLog(webContents, 'info', ` → Configuring N8N for HTTP (secure cookies disabled)`); + } else { + config.environment.push('N8N_SECURE_COOKIE=true'); + this.sendLog(webContents, 'info', ` → Configuring N8N with secure cookies (HTTPS required)`); + } + } + + try { + this.sendLog(webContents, 'info', ` → Pulling ${config.image}...`); + + // Pull image (this might take time) + const pullResult = await this.ssh.execCommand(`docker pull ${config.image}`, { + onStdout: (chunk) => { + // Stream pull progress + const message = chunk.toString('utf8').trim(); + if (message) { + this.sendLog(webContents, 'info', ` ${message}`); + } + } + }); + + if (pullResult.code !== 0) { + return { success: false, error: pullResult.stderr }; + } + + this.sendLog(webContents, 'success', ` ✓ Image pulled`); + this.sendLog(webContents, 'info', ` → Starting container...`); + + // Build docker run command + const envVars = config.environment.map(e => `-e ${e}`).join(' '); + const runtime = config.runtime || ''; + const containerName = `clara_${serviceName}`; + + // Determine internal port (container port) - Python backend runs on 5000 internally + const internalPort = config.port === 5001 ? 5000 : config.port; + + // Python backend uses host network to access ClaraCore via host IP + // Other services use clara_network for container-to-container communication + const networkMode = serviceName === 'python' ? 'host' : 'clara_network'; + const portMapping = serviceName === 'python' ? '' : `-p ${config.port}:${internalPort}`; + + // Build volume mounts for data persistence + const volumeMounts = config.volumes ? config.volumes.map(v => `-v ${v}`).join(' ') : ''; + if (volumeMounts) { + this.sendLog(webContents, 'info', ` → Configuring persistent storage (${config.volumes.length} volumes)`); + } + + const runCommand = `docker run -d --name ${containerName} --network ${networkMode} ${portMapping} ${volumeMounts} ${envVars} ${runtime} --restart unless-stopped ${config.image}`; + + const runResult = await this.ssh.execCommand(runCommand); + + if (runResult.code !== 0) { + // Try without GPU if it fails + if (runtime.includes('gpus')) { + this.sendLog(webContents, 'warning', ` ⚠ GPU not available, trying CPU mode...`); + const cpuCommand = runCommand.replace('--gpus all', ''); + const cpuResult = await this.ssh.execCommand(cpuCommand); + + if (cpuResult.code !== 0) { + return { success: false, error: cpuResult.stderr }; + } + + // For host network mode, report the actual port the service runs on + const actualPort = serviceName === 'python' ? internalPort : config.port; + return { + success: true, + port: actualPort, + containerId: cpuResult.stdout.trim() + }; + } + + return { success: false, error: runResult.stderr }; + } + + // For host network mode, report the actual port the service runs on + const actualPort = serviceName === 'python' ? internalPort : config.port; + return { + success: true, + port: actualPort, + containerId: runResult.stdout.trim() + }; + + } catch (error) { + return { success: false, error: error.message }; + } + } + + /** + * Stop a service + */ + async stopService(config, serviceName) { + const ssh = new NodeSSH(); + + try { + await ssh.connect({ + host: config.host, + port: config.port || 22, + username: config.username, + password: config.password + }); + + const containerName = `clara_${serviceName}`; + await ssh.execCommand(`docker stop ${containerName}`); + await ssh.execCommand(`docker rm ${containerName}`); + + ssh.dispose(); + + return { success: true }; + } catch (error) { + if (ssh) ssh.dispose(); + return { success: false, error: error.message }; + } + } +} + +module.exports = RemoteServerService; diff --git a/electron/schedulerBridge.cjs b/electron/schedulerBridge.cjs new file mode 100644 index 00000000..8d0ec1e5 --- /dev/null +++ b/electron/schedulerBridge.cjs @@ -0,0 +1,259 @@ +// ClaraVerse Scheduler Bridge for Electron Main Process +// This bridge handles the TypeScript/ES module compatibility issues + +const path = require('path'); + +class SchedulerBridge { + constructor() { + this.isRunning = false; + this.activeExecutions = 0; + this.schedulerInstance = null; + this.checkInterval = null; + this.mainWindow = null; // Reference to main window for IPC communication + this.lastSchedulerAvailable = true; // Track scheduler availability state + } + + setMainWindow(window) { + this.mainWindow = window; + } + + async start() { + if (this.isRunning) { + console.log('⚠️ Scheduler already running'); + return; + } + + try { + this.isRunning = true; + this.activeExecutions = 0; + + // Start more frequent checking for short intervals + // Check every 10 seconds to handle 30-second and minute intervals properly + this.checkInterval = setInterval(() => { + this.checkAndExecuteTasks(); + }, 10000); // 10 seconds + + // Run initial check for overdue tasks + setTimeout(() => { + this.checkAndExecuteTasks(); + }, 2000); // 2 seconds after start + + console.log('✅ Scheduler bridge started successfully (checking every 10 seconds)'); + } catch (error) { + console.error('❌ Failed to start scheduler bridge:', error); + this.isRunning = false; + throw error; + } + } + + async stop() { + if (!this.isRunning) { + console.log('⚠️ Scheduler already stopped'); + return; + } + + try { + this.isRunning = false; + + if (this.checkInterval) { + clearInterval(this.checkInterval); + this.checkInterval = null; + } + + console.log('🛑 Scheduler bridge stopped'); + } catch (error) { + console.error('❌ Failed to stop scheduler bridge:', error); + throw error; + } + } + + getStatus() { + return { + isRunning: this.isRunning, + activeExecutions: this.activeExecutions, + nextCheck: this.checkInterval ? new Date(Date.now() + 10000) : undefined // Next check in 10 seconds + }; + } + + async checkAndExecuteTasks() { + if (!this.isRunning) return; + + try { + // Request the renderer process to check and execute tasks + // since IndexedDB and ClaraFlowRunner are only available there + if (this.mainWindow && this.mainWindow.webContents) { + try { + const result = await this.mainWindow.webContents.executeJavaScript(` + (async () => { + try { + // Check if schedulerStorage is available + if (typeof window !== 'undefined' && window.schedulerStorage) { + + // Get tasks that are due + const tasksDue = await window.schedulerStorage.getTasksDueForExecution(); + + if (tasksDue.length > 0) { + // Notify about overdue tasks + const now = new Date(); + tasksDue.forEach(task => { + if (task.schedule.nextRun) { + const nextRun = new Date(task.schedule.nextRun); + const overdue = Math.floor((now - nextRun) / 60000); + if (overdue > 0) { + console.log('⏰ Task "' + task.agentName + '" is ' + overdue + ' minutes overdue'); + } else { + console.log('🎯 Task "' + task.agentName + '" is due now'); + } + } + }); + + // Execute each task + for (const task of tasksDue) { + try { + + // Try to execute manually using available globals + if (window.ClaraFlowRunner && window.agentWorkflowStorage) { + const execution = { + id: 'exec_' + Date.now() + '_' + Math.random().toString(36).substr(2, 9), + taskId: task.id, + startTime: new Date().toISOString(), + status: 'running', + outputs: {}, + logs: [] + }; + + // Load agent flow + const agentFlow = await window.agentWorkflowStorage.getWorkflow(task.agentFlowId); + if (!agentFlow) { + throw new Error('Agent flow not found: ' + task.agentFlowId); + } + + // Create runner + const runner = new window.ClaraFlowRunner(); + + // Register custom nodes if any + if (agentFlow.customNodes?.length > 0) { + for (const customNode of agentFlow.customNodes) { + runner.registerCustomNode(customNode); + } + } + + // Convert scheduled inputs to SDK format + const sdkInputs = {}; + task.inputs.forEach(input => { + sdkInputs[input.nodeName] = input.value; + }); + + console.log('🔧 Executing with inputs:', Object.keys(sdkInputs)); + + // Execute + const result = await runner.executeFlow(agentFlow, sdkInputs); + + execution.endTime = new Date().toISOString(); + execution.status = 'completed'; + execution.outputs = result.outputs || {}; + execution.logs = result.logs || []; + execution.duration = new Date(execution.endTime).getTime() - new Date(execution.startTime).getTime(); + + // Save execution + await window.schedulerStorage.saveTaskExecution(execution); + + // Update task + task.schedule.lastRun = execution.endTime; + task.schedule.nextRun = window.schedulerStorage.calculateNextRun(task.schedule); + task.schedule.status = 'idle'; + await window.schedulerStorage.saveScheduledTask(task); + + console.log('✅ Execution completed for:', task.agentName); + } else { + console.log('❌ Required execution dependencies not available (ClaraFlowRunner, agentWorkflowStorage)'); + } + } catch (execError) { + console.error('❌ Failed to execute task "' + task.agentName + '":', execError); + + // Save failed execution + const failedExecution = { + id: 'exec_' + Date.now() + '_' + Math.random().toString(36).substr(2, 9), + taskId: task.id, + startTime: new Date().toISOString(), + endTime: new Date().toISOString(), + status: 'error', + error: execError.message, + outputs: {}, + logs: [] + }; + await window.schedulerStorage.saveTaskExecution(failedExecution); + + // Update task status + task.schedule.status = 'error'; + task.metadata = task.metadata || {}; + task.metadata.lastError = execError.message; + await window.schedulerStorage.saveScheduledTask(task); + } + } + } + + return { success: true, tasksExecuted: tasksDue.length }; + } else { + return { success: false, error: 'schedulerStorage not available' }; + } + } catch (error) { + console.error('❌ Error in task execution script:', error); + return { success: false, error: error.message }; + } + })(); + `); + + if (result.success) { + // Only log when tasks are actually executed + if (result.tasksExecuted > 0) { + console.log('📊 Executed', result.tasksExecuted, 'scheduled task(s)'); + } + this.activeExecutions = result.tasksExecuted || 0; + + // Update scheduler availability state (now available) + if (!this.lastSchedulerAvailable) { + console.log('✅ Scheduler storage is now available'); + this.lastSchedulerAvailable = true; + } + } else { + // Only log scheduler unavailability on state change + if (this.lastSchedulerAvailable && result.error === 'schedulerStorage not available') { + console.log('⚠️ Scheduler storage not yet available, will retry...'); + this.lastSchedulerAvailable = false; + } else if (result.error !== 'schedulerStorage not available') { + // Log other errors always + console.log('⚠️ Task check failed:', result.error); + } + } + } catch (jsError) { + console.error('❌ JavaScript execution failed:', jsError); + } + } else { + console.log('⚠️ Main window not available for task execution'); + } + + } catch (error) { + console.error('❌ Error during task check:', error); + } + } + + // Placeholder methods for IPC compatibility + async getActiveTasks() { + return []; + } + + async getTaskExecutions(taskId) { + return []; + } + + async cancelTask(taskId) { + console.log(`🚫 Cancel task requested: ${taskId}`); + return true; + } +} + +// Create singleton instance +const schedulerBridge = new SchedulerBridge(); + +module.exports = { schedulerBridge }; diff --git a/electron/schedulerElectronService.cjs b/electron/schedulerElectronService.cjs new file mode 100644 index 00000000..aeb42684 --- /dev/null +++ b/electron/schedulerElectronService.cjs @@ -0,0 +1,197 @@ +// ClaraVerse Scheduler Service for Electron Main Process +const { ipcMain } = require('electron'); + +class SchedulerElectronService { + constructor(mainWindow = null) { + this.scheduler = null; + this.isRunning = false; + this.activeExecutions = 0; + this.mainWindow = mainWindow; + this.initializeIPC(); + } + + setMainWindow(window) { + this.mainWindow = window; + if (this.scheduler && this.scheduler.setMainWindow) { + this.scheduler.setMainWindow(window); + } + } + + initializeIPC() { + // Scheduler control commands + ipcMain.handle('scheduler:start', async () => { + try { + console.log('📞 IPC received scheduler:start'); + + if (!this.scheduler) { + console.log('⚙️ Initializing scheduler...'); + await this.initializeScheduler(); + } + + if (this.scheduler && typeof this.scheduler.start === 'function') { + await this.scheduler.start(); + this.isRunning = true; + console.log('✅ Scheduler started from main process'); + return { success: true }; + } else { + throw new Error('Scheduler not properly initialized'); + } + } catch (error) { + console.error('❌ Failed to start scheduler:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('scheduler:stop', async () => { + try { + console.log('📞 IPC received scheduler:stop'); + if (this.scheduler && typeof this.scheduler.stop === 'function') { + await this.scheduler.stop(); + this.isRunning = false; + console.log('🛑 Scheduler stopped from main process'); + } + return { success: true }; + } catch (error) { + console.error('❌ Failed to stop scheduler:', error); + return { success: false, error: error.message }; + } + }); + + ipcMain.handle('scheduler:status', async () => { + try { + console.log('📞 IPC received scheduler:status'); + + let status = { + isRunning: this.isRunning, + activeExecutions: this.activeExecutions + }; + + // If scheduler exists, get status from it + if (this.scheduler && typeof this.scheduler.getStatus === 'function') { + try { + const schedulerStatus = this.scheduler.getStatus(); + status = { ...status, ...schedulerStatus }; + } catch (statusError) { + console.warn('⚠️ Failed to get scheduler status:', statusError); + } + } + + console.log('📊 Scheduler status:', status); + return status; + } catch (error) { + console.error('❌ Failed to get scheduler status:', error); + return { isRunning: false, activeExecutions: 0, error: error.message }; + } + }); + + ipcMain.handle('scheduler:getActiveTasks', async () => { + try { + console.log('📞 IPC received scheduler:getActiveTasks'); + if (!this.scheduler || typeof this.scheduler.getActiveTasks !== 'function') { + return []; + } + + const tasks = await this.scheduler.getActiveTasks(); + return tasks; + } catch (error) { + console.error('❌ Failed to get active tasks:', error); + return []; + } + }); + + ipcMain.handle('scheduler:getTaskExecutions', async (event, taskId) => { + try { + console.log('📞 IPC received scheduler:getTaskExecutions'); + if (!this.scheduler || typeof this.scheduler.getTaskExecutions !== 'function') { + return []; + } + + const executions = await this.scheduler.getTaskExecutions(taskId); + return executions; + } catch (error) { + console.error('❌ Failed to get task executions:', error); + return []; + } + }); + + ipcMain.handle('scheduler:cancelTask', async (event, taskId) => { + try { + console.log('📞 IPC received scheduler:cancelTask'); + if (!this.scheduler || typeof this.scheduler.cancelTask !== 'function') { + return { success: false, error: 'Scheduler not initialized' }; + } + + await this.scheduler.cancelTask(taskId); + return { success: true }; + } catch (error) { + console.error('❌ Failed to cancel task:', error); + return { success: false, error: error.message }; + } + }); + + // Auto-start scheduler when app starts + this.autoStartScheduler(); + } + + async initializeScheduler() { + try { + console.log('🔧 Initializing scheduler bridge...'); + + // Use the CommonJS bridge instead of trying to import TypeScript + const { schedulerBridge } = require('./schedulerBridge.cjs'); + this.scheduler = schedulerBridge; + + // Set the main window reference + if (this.mainWindow && this.scheduler.setMainWindow) { + this.scheduler.setMainWindow(this.mainWindow); + } + + console.log('✅ Scheduler bridge loaded successfully'); + + if (this.scheduler) { + console.log('🔍 Scheduler methods available:', Object.getOwnPropertyNames(Object.getPrototypeOf(this.scheduler))); + } + + } catch (error) { + console.error('❌ Failed to initialize scheduler bridge:', error); + throw error; + } + } + + async autoStartScheduler() { + try { + // Wait a bit for the app to fully initialize + setTimeout(async () => { + console.log('🚀 Auto-starting ClaraVerse Scheduler...'); + + try { + if (!this.scheduler) { + await this.initializeScheduler(); + } + + if (this.scheduler && typeof this.scheduler.start === 'function') { + await this.scheduler.start(); + this.isRunning = true; + console.log('✅ Scheduler auto-started successfully'); + } else { + console.error('❌ Scheduler not properly initialized for auto-start'); + } + } catch (initError) { + console.error('❌ Failed to auto-start scheduler:', initError); + } + }, 5000); // 5 second delay + } catch (error) { + console.error('❌ Failed to setup auto-start scheduler:', error); + } + } + + // Call this when app is closing + async cleanup() { + if (this.scheduler) { + await this.scheduler.stop(); + console.log('🧹 Scheduler cleaned up'); + } + } +} + +module.exports = { SchedulerElectronService }; diff --git a/electron/serviceConfiguration.cjs b/electron/serviceConfiguration.cjs new file mode 100644 index 00000000..5ecf96f8 --- /dev/null +++ b/electron/serviceConfiguration.cjs @@ -0,0 +1,337 @@ +const { app } = require('electron'); +const fs = require('fs'); +const path = require('path'); +const log = require('electron-log'); +const { isServiceModeSupported, getSupportedDeploymentModes, createManualHealthCheck } = require('./serviceDefinitions.cjs'); + +/** + * Service Configuration Manager + * Handles user preferences for deployment modes and manual service URLs + * Provides persistent storage and validation + */ +class ServiceConfigurationManager { + constructor() { + this.configPath = path.join(app.getPath('userData'), 'service-config.json'); + this.config = this.loadConfig(); + this.platform = require('os').platform(); + + log.info('🔧 Service Configuration Manager initialized'); + } + + /** + * Load configuration from disk + */ + loadConfig() { + try { + if (fs.existsSync(this.configPath)) { + const configData = fs.readFileSync(this.configPath, 'utf8'); + const config = JSON.parse(configData); + + // Validate loaded config + return this.validateConfig(config); + } + } catch (error) { + log.warn('Failed to load service configuration, using defaults:', error.message); + } + + // Return default configuration + return { + version: '1.0.0', + platform: this.platform, + services: {}, + lastModified: Date.now() + }; + } + + /** + * Save configuration to disk + */ + saveConfig() { + try { + this.config.lastModified = Date.now(); + this.config.platform = this.platform; + + fs.writeFileSync(this.configPath, JSON.stringify(this.config, null, 2), 'utf8'); + log.info('✅ Service configuration saved'); + } catch (error) { + log.error('❌ Failed to save service configuration:', error); + throw error; + } + } + + /** + * Validate configuration structure and remove invalid entries + */ + validateConfig(config) { + const validatedConfig = { + version: config.version || '1.0.0', + platform: this.platform, + services: {}, + lastModified: config.lastModified || Date.now() + }; + + if (config.services && typeof config.services === 'object') { + Object.keys(config.services).forEach(serviceName => { + const serviceConfig = config.services[serviceName]; + + // Validate service configuration + if (this.isValidServiceConfig(serviceName, serviceConfig)) { + validatedConfig.services[serviceName] = serviceConfig; + } else { + log.warn(`Invalid configuration for service ${serviceName}, removing`); + } + }); + } + + return validatedConfig; + } + + /** + * Check if service configuration is valid + */ + isValidServiceConfig(serviceName, serviceConfig) { + if (!serviceConfig || typeof serviceConfig !== 'object') { + return false; + } + + const { mode, url } = serviceConfig; + + // Check if mode is supported on current platform + if (!isServiceModeSupported(serviceName, mode, this.platform)) { + return false; + } + + // Check URL requirement for manual and remote modes + if ((mode === 'manual' || mode === 'remote') && (!url || typeof url !== 'string')) { + return false; + } + + return true; + } + + /** + * Get deployment mode for a service + */ + getServiceMode(serviceName) { + const serviceConfig = this.config.services[serviceName]; + if (serviceConfig && serviceConfig.mode) { + // Verify mode is still supported on current platform + if (isServiceModeSupported(serviceName, serviceConfig.mode, this.platform)) { + return serviceConfig.mode; + } + } + + // Special case for ClaraCore - always default to 'local' mode + if (serviceName === 'claracore') { + return 'local'; + } + + // Return default mode (prefer docker if available) + const supportedModes = getSupportedDeploymentModes(serviceName, this.platform); + return supportedModes.includes('docker') ? 'docker' : supportedModes[0] || 'docker'; + } + + /** + * Get service URL for manual deployment + */ + getServiceUrl(serviceName) { + const serviceConfig = this.config.services[serviceName]; + return serviceConfig?.url || null; + } + + /** + * Set service configuration (mode and URL) + */ + setServiceConfig(serviceName, mode, url = null) { + // Validate mode is supported + if (!isServiceModeSupported(serviceName, mode, this.platform)) { + throw new Error(`Deployment mode '${mode}' is not supported for service '${serviceName}' on platform '${this.platform}'`); + } + + // Validate URL for manual and remote modes + if (mode === 'manual' || mode === 'remote') { + if (!url || typeof url !== 'string') { + throw new Error(`URL is required for ${mode} deployment mode of service '${serviceName}'`); + } + + // Basic URL validation + try { + new URL(url); + } catch (error) { + throw new Error(`Invalid URL format for service '${serviceName}': ${url}`); + } + } + + // Initialize services object if needed + if (!this.config.services) { + this.config.services = {}; + } + + // Set configuration + this.config.services[serviceName] = { mode, url }; + + // Save immediately + this.saveConfig(); + + log.info(`📝 Service ${serviceName} configured: mode=${mode}${url ? `, url=${url}` : ''}`); + } + + /** + * Remove service configuration (revert to defaults) + */ + removeServiceConfig(serviceName) { + if (this.config.services && this.config.services[serviceName]) { + delete this.config.services[serviceName]; + this.saveConfig(); + log.info(`🗑️ Service ${serviceName} configuration removed (reverted to defaults)`); + } + } + + /** + * Get all service configurations + */ + getAllServiceConfigs() { + return { ...this.config.services }; + } + + /** + * Test connectivity to a manual service + */ + async testManualService(serviceName, url, healthEndpoint = '/') { + try { + const healthCheck = createManualHealthCheck(url, healthEndpoint); + const isHealthy = await healthCheck(); + + return { + success: isHealthy, + url: url, + endpoint: healthEndpoint, + timestamp: Date.now() + }; + } catch (error) { + return { + success: false, + url: url, + endpoint: healthEndpoint, + error: error.message, + timestamp: Date.now() + }; + } + } + + /** + * Get supported deployment modes for a service + */ + getSupportedModes(serviceName) { + return getSupportedDeploymentModes(serviceName, this.platform); + } + + /** + * Check if service can use Docker deployment on current platform + */ + canUseDocker(serviceName) { + return isServiceModeSupported(serviceName, 'docker', this.platform); + } + + /** + * Check if service can use manual deployment on current platform + */ + canUseManual(serviceName) { + return isServiceModeSupported(serviceName, 'manual', this.platform); + } + + /** + * Get platform-specific service information + */ + getPlatformInfo(serviceName = null) { + const { getPlatformCompatibility } = require('./serviceDefinitions.cjs'); + const compatibility = getPlatformCompatibility(this.platform); + + if (serviceName) { + return compatibility[serviceName] || null; + } + + return compatibility; + } + + /** + * Reset all configurations to defaults + */ + resetToDefaults() { + this.config = { + version: '1.0.0', + platform: this.platform, + services: {}, + lastModified: Date.now() + }; + + this.saveConfig(); + log.info('🔄 Service configuration reset to defaults'); + } + + /** + * Export configuration for backup + */ + exportConfig() { + return { + ...this.config, + exportedAt: Date.now(), + exportedPlatform: this.platform + }; + } + + /** + * Import configuration from backup + */ + importConfig(configData) { + try { + // Validate imported configuration + const validatedConfig = this.validateConfig(configData); + + // Warn if platform mismatch + if (configData.exportedPlatform && configData.exportedPlatform !== this.platform) { + log.warn(`Platform mismatch: configuration exported from ${configData.exportedPlatform}, importing to ${this.platform}`); + } + + this.config = validatedConfig; + this.saveConfig(); + + log.info('📥 Service configuration imported successfully'); + return true; + } catch (error) { + log.error('❌ Failed to import service configuration:', error); + throw error; + } + } + + /** + * Get configuration summary for UI display + */ + getConfigSummary() { + const { SERVICE_DEFINITIONS } = require('./serviceDefinitions.cjs'); + const summary = {}; + + Object.keys(SERVICE_DEFINITIONS).forEach(serviceName => { + const service = SERVICE_DEFINITIONS[serviceName]; + const platformInfo = this.getPlatformInfo(serviceName); + const currentMode = this.getServiceMode(serviceName); + const currentUrl = this.getServiceUrl(serviceName); + + summary[serviceName] = { + name: service.name, + critical: service.critical, + currentMode: currentMode, + currentUrl: currentUrl, + supportedModes: platformInfo?.supportedModes || [], + dockerSupported: platformInfo?.dockerSupported || false, + manualSupported: platformInfo?.manualSupported || false, + manualConfig: platformInfo?.manualConfig || null, + configured: !!this.config.services[serviceName] + }; + }); + + return summary; + } +} + +module.exports = ServiceConfigurationManager; \ No newline at end of file diff --git a/electron/serviceDefinitions.cjs b/electron/serviceDefinitions.cjs new file mode 100644 index 00000000..0d505cf8 --- /dev/null +++ b/electron/serviceDefinitions.cjs @@ -0,0 +1,720 @@ +/** + * ClaraVerse Service Definitions + * Centralized configuration for all services + * Replaces scattered service configurations across multiple files + */ + +const path = require('path'); +const os = require('os'); +const log = require('electron-log'); + +// Get platform info +const platform = os.platform(); +const isWindows = platform === 'win32'; +const isMac = platform === 'darwin'; +const isLinux = platform === 'linux'; + +// Base paths +const appDataPath = path.join(os.homedir(), '.clara'); +const pythonBackendDataPath = path.join(appDataPath, 'python_backend_data'); + +/** + * Service definitions with all configuration + */ +const SERVICE_DEFINITIONS = { + + // Docker Engine (foundational service) + docker: { + name: 'Docker Engine', + type: 'docker-daemon', + critical: true, + autoRestart: true, + priority: 1, + dependencies: [], + + healthCheck: async () => { + const Docker = require('dockerode'); + try { + const docker = new Docker(); + await docker.ping(); + return true; + } catch { + return false; + } + }, + + customStart: async () => { + // Docker daemon startup logic + const DockerSetup = require('./dockerSetup.cjs'); + const dockerSetup = new DockerSetup(); + await dockerSetup.ensureDockerRunning(); + return dockerSetup; + } + }, + + // Python Backend Service + 'python-backend': { + name: 'Python Backend Service', + type: 'docker-container', + critical: true, + autoRestart: true, + priority: 2, + dependencies: ['docker'], + + // NEW: Deployment mode support + deploymentModes: ['docker', 'manual', 'remote'], + platformSupport: { + docker: ['win32', 'darwin', 'linux'], // Docker supported on all platforms + manual: ['win32', 'darwin', 'linux'], // Manual/BYOS supported on all platforms + remote: ['win32', 'darwin', 'linux'] // Remote server supported on all platforms + }, + + // NEW: Manual service configuration + manual: { + urlRequired: true, + // On Linux (host network mode), use port 5000. On Windows/Mac (bridge mode), use port 5001 + defaultUrl: `http://localhost:${isLinux ? 5000 : 5001}`, + healthEndpoint: '/health', + configKey: 'python_backend_url', + description: 'Bring Your Own Python Backend - Connect to external Python Backend instance' + }, + + dockerContainer: { + name: 'clara_python', + image: 'clara17verse/clara-backend:latest', + // On Linux (host network mode), container runs on 5000 directly. On Windows/Mac, map 5001->5000 + ports: isLinux ? { '5000': '5000' } : { '5001': '5000' }, + volumes: [ + `${pythonBackendDataPath}:/home/clara`, + 'clara_python_models:/app/models' + ], + environment: [ + 'PYTHONUNBUFFERED=1', + 'CLARA_ENV=production' + ] + }, + + healthCheck: async (serviceUrl = null) => { + const http = require('http'); + // On Linux (host network mode), use port 5000. On Windows/Mac (bridge mode), use port 5001 + const defaultPort = isLinux ? 5000 : 5001; + const url = serviceUrl || `http://localhost:${defaultPort}`; + const endpoint = serviceUrl ? `${url}/health` : `http://localhost:${defaultPort}/health`; + return new Promise((resolve) => { + const req = http.get(endpoint, (res) => { + resolve(res.statusCode === 200); + }); + req.on('error', () => resolve(false)); + req.setTimeout(5000, () => { + req.destroy(); + resolve(false); + }); + }); + } + }, + + // ClaraCore Service (Core AI Engine) + claracore: { + name: 'Clara Core AI Engine', + type: 'binary', + critical: true, + autoRestart: true, + priority: 3, + dependencies: ['docker'], // Docker dependency for docker mode + + // NEW: Deployment mode support + deploymentModes: ['local', 'remote', 'docker'], + platformSupport: { + local: ['win32', 'darwin', 'linux'], // Native binary supported on all platforms + remote: ['win32', 'darwin', 'linux'], // Remote server supported on all platforms + docker: ['win32', 'linux'] // Docker supported on Windows and Linux (GPU support) + }, + + // Binary paths for each platform + binaryPath: platform === 'win32' + ? './claracore/claracore-windows-amd64.exe' + : platform === 'darwin' + ? os.arch() === 'arm64' + ? './claracore/claracore-darwin-arm64' + : './claracore/claracore-darwin-amd64' + : platform === 'linux' + ? os.arch() === 'arm64' + ? './claracore/claracore-linux-arm64' + : './claracore/claracore-linux-amd64' + : './claracore/claracore-linux-amd64', + + // Service arguments + args: ['-listen', ':8091'], + + ports: { main: 8091 }, + + // Docker container configuration + dockerContainer: { + name: 'clara_core', + imageBase: 'clara17verse/claracore', // Base image name, variant added based on GPU + ports: { '8091': '5890' }, // Host:Container (container runs on 5890, mapped to host 8091) + volumes: [ + 'claracore:/app/downloads' // Named volume for downloads persistence + ], + environment: [ + 'NODE_ENV=production', + 'CLARA_PORT=5890' // Container internal port + ], + // GPU-specific configurations + gpuConfigs: { + cuda: { + image: 'clara17verse/claracore:cuda', + runtime: 'nvidia', + environment: [ + 'NVIDIA_VISIBLE_DEVICES=all', + 'NVIDIA_DRIVER_CAPABILITIES=compute,utility' + ] + }, + rocm: { + image: 'clara17verse/claracore:rocm', + devices: ['/dev/kfd', '/dev/dri'], + environment: [ + 'HSA_OVERRIDE_GFX_VERSION=10.3.0' + ] + }, + vulkan: { + image: 'clara17verse/claracore:vulkan', + devices: ['/dev/dri'], + environment: [ + 'VK_ICD_FILENAMES=/usr/share/vulkan/icd.d/nvidia_icd.json' + ] + }, + cpu: { + image: 'clara17verse/claracore:cpu', + environment: [] + } + } + }, + + // Health check + healthCheck: async (serviceUrl = null) => { + const http = require('http'); + const url = serviceUrl || 'http://localhost:8091'; + const endpoint = serviceUrl ? `${url}/health` : 'http://localhost:8091/health'; + return new Promise((resolve) => { + const req = http.get(endpoint, (res) => { + resolve(res.statusCode === 200); + }); + req.on('error', () => resolve(false)); + req.setTimeout(3000, () => { + req.destroy(); + resolve(false); + }); + }); + }, + + // Custom start method (for local mode) + customStart: async () => { + const ClaraCoreService = require('./claraCoreService.cjs'); + const service = new ClaraCoreService(); + await service.start(); + return service; + }, + + // Custom stop method (for local mode) + customStop: async (service) => { + if (service.instance && service.instance.stop) { + await service.instance.stop(); + } + }, + + // Manual/Remote service configuration + manual: { + urlRequired: true, + defaultUrl: 'http://localhost:8091', + healthEndpoint: '/health', + configKey: 'claracore_url', + description: 'Connect to external ClaraCore instance (local, remote, or docker)' + } + }, + + // ComfyUI Image Generation Service + comfyui: { + name: 'ComfyUI Image Generation', + type: 'docker-container', + critical: false, // Non-critical - user can disable + autoRestart: true, + priority: 4, + dependencies: ['docker', 'python-backend'], + + // NEW: Deployment mode support + deploymentModes: ['docker', 'manual', 'remote'], + platformSupport: { + docker: ['win32'], // Docker only supported on Windows + manual: ['win32', 'darwin', 'linux'], // Manual/BYOS supported on all platforms + remote: ['win32', 'darwin', 'linux'] // Remote server supported on all platforms + }, + + // NEW: Manual service configuration + manual: { + urlRequired: true, + defaultUrl: 'http://localhost:8188', + healthEndpoint: '/', + configKey: 'comfyui_url', + description: 'Bring Your Own ComfyUI - Connect to external ComfyUI instance' + }, + + dockerContainer: { + name: 'clara_comfyui', + image: 'clara17verse/clara-comfyui:with-custom-nodes', + ports: { '8188': '8188' }, + volumes: [ + `${path.join(appDataPath, 'comfyui_models')}:/app/ComfyUI/models`, + `${path.join(appDataPath, 'comfyui_output')}:/app/ComfyUI/output`, + `${path.join(appDataPath, 'comfyui_input')}:/app/ComfyUI/input`, + `${path.join(appDataPath, 'comfyui_custom_nodes')}:/app/ComfyUI/custom_nodes`, + `${path.join(appDataPath, 'comfyui_temp')}:/tmp` + ], + environment: [ + 'NVIDIA_VISIBLE_DEVICES=all', + 'CUDA_VISIBLE_DEVICES=0', + 'PYTORCH_CUDA_ALLOC_CONF=max_split_size_mb:2048,expandable_segments:True', + 'COMFYUI_FORCE_FP16=1', + 'COMFYUI_HIGHVRAM=1' + ], + runtime: 'nvidia' // GPU support + }, + + healthCheck: async (serviceUrl = null) => { + const http = require('http'); + const url = serviceUrl || 'http://localhost:8188'; + return new Promise((resolve) => { + const req = http.get(url, (res) => { + resolve(res.statusCode === 200); + }); + req.on('error', () => resolve(false)); + req.setTimeout(10000, () => { + req.destroy(); + resolve(false); + }); + }); + } + }, + + // N8N Workflow Automation + n8n: { + name: 'N8N Workflow Engine', + type: 'docker-container', + critical: false, + autoRestart: true, + priority: 5, + dependencies: ['docker'], + + // NEW: Deployment mode support + deploymentModes: ['docker', 'manual', 'remote'], + platformSupport: { + docker: ['win32', 'darwin', 'linux'], // Docker supported on all platforms + manual: ['win32', 'darwin', 'linux'], // Manual/BYOS supported on all platforms + remote: ['win32', 'darwin', 'linux'] // Remote server supported on all platforms + }, + + // NEW: Manual service configuration + manual: { + urlRequired: true, + defaultUrl: 'http://localhost:5678', + healthEndpoint: '/healthz', + configKey: 'n8n_url', + description: 'Bring Your Own N8N - Connect to external N8N instance' + }, + + dockerContainer: { + name: 'clara_n8n', + image: 'n8nio/n8n:latest', + ports: { '5678': '5678' }, + volumes: [ + `${path.join(appDataPath, 'n8n')}:/home/node/.n8n` + ], + environment: [ + 'N8N_BASIC_AUTH_ACTIVE=false', + 'N8N_METRICS=true', + 'WEBHOOK_URL=http://localhost:5678/', + 'GENERIC_TIMEZONE=UTC' + ] + }, + + healthCheck: async (serviceUrl = null) => { + const http = require('http'); + const url = serviceUrl || 'http://localhost:5678'; + const endpoint = serviceUrl ? `${url}/healthz` : 'http://localhost:5678/healthz'; + return new Promise((resolve) => { + const req = http.get(endpoint, (res) => { + resolve(res.statusCode === 200); + }); + req.on('error', () => resolve(false)); + req.setTimeout(5000, () => { + req.destroy(); + resolve(false); + }); + }); + } + }, + + // Model Context Protocol Service + mcp: { + name: 'Model Context Protocol', + type: 'service', + critical: false, + autoRestart: true, + priority: 6, + dependencies: ['python-backend'], + + customStart: async () => { + const MCPService = require('./mcpService.cjs'); + const service = new MCPService(); + await service.start(); + return service; + }, + + customStop: async (service) => { + if (service.instance && service.instance.stop) { + await service.instance.stop(); + } + }, + + healthCheck: async () => { + // MCP health check logic + return true; // Placeholder + } + }, + + // MCP HTTP Proxy Service (for browser support) + 'mcp-proxy': { + name: 'MCP HTTP Proxy', + type: 'http-service', + critical: false, + autoRestart: true, + priority: 7, + dependencies: ['mcp'], + + ports: { main: 8092 }, + + customStart: async (mcpServiceInstance) => { + const MCPProxyService = require('./mcpProxyService.cjs'); + + // Get the MCP service instance from dependencies + // If not provided, create one (fallback) + let mcpService = mcpServiceInstance; + if (!mcpService) { + const MCPService = require('./mcpService.cjs'); + mcpService = new MCPService(); + } + + const proxyService = new MCPProxyService(mcpService); + const result = await proxyService.start(8092); + + return { + instance: proxyService, + url: result.url, + port: result.port, + healthCheck: result.healthCheck + }; + }, + + customStop: async (service) => { + if (service.instance && service.instance.stop) { + await service.instance.stop(); + } + }, + + healthCheck: async (serviceUrl = null) => { + const http = require('http'); + const url = serviceUrl || 'http://localhost:8092'; + const endpoint = serviceUrl ? `${url}/health` : 'http://localhost:8092/health'; + return new Promise((resolve) => { + const req = http.get(endpoint, (res) => { + resolve(res.statusCode === 200); + }); + req.on('error', () => resolve(false)); + req.setTimeout(3000, () => { + req.destroy(); + resolve(false); + }); + }); + } + } +}; + +/** + * Platform-specific service configurations + */ +const PLATFORM_OVERRIDES = { + darwin: {}, + linux: {}, + win32: {} +}; + +/** + * Feature-based service selection + * Based on user's feature selection during setup + */ +function getEnabledServices(selectedFeatures = {}) { + const enabledServices = {}; + + // Core services (always enabled) + const coreServices = ['docker', 'python-backend', 'claracore']; + + // Optional services based on user selection + if (selectedFeatures.comfyUI) { + enabledServices.comfyui = SERVICE_DEFINITIONS.comfyui; + } + + if (selectedFeatures.n8n) { + enabledServices.n8n = SERVICE_DEFINITIONS.n8n; + } + + // Add core services + coreServices.forEach(serviceName => { + if (SERVICE_DEFINITIONS[serviceName]) { + enabledServices[serviceName] = SERVICE_DEFINITIONS[serviceName]; + } + }); + + return enabledServices; +} + +/** + * Apply platform-specific overrides + */ +function applyPlatformOverrides(services) { + const platformOverrides = PLATFORM_OVERRIDES[platform] || {}; + + const result = { ...services }; + + Object.keys(platformOverrides).forEach(serviceName => { + if (result[serviceName]) { + result[serviceName] = { + ...result[serviceName], + ...platformOverrides[serviceName] + }; + } + }); + + return result; +} + +/** + * Get complete service configuration for current platform and features + */ +function getServiceConfiguration(selectedFeatures = {}) { + let services = getEnabledServices(selectedFeatures); + services = applyPlatformOverrides(services); + + return services; +} + +/** + * Validate service configuration + */ +function validateServiceConfiguration(services) { + const errors = []; + + // Check for circular dependencies + const dependencies = {}; + Object.keys(services).forEach(name => { + dependencies[name] = services[name].dependencies || []; + }); + + try { + // Simple cycle detection + const visited = new Set(); + const temp = new Set(); + + const visit = (node) => { + if (temp.has(node)) { + errors.push(`Circular dependency detected involving ${node}`); + return; + } + if (!visited.has(node)) { + temp.add(node); + dependencies[node].forEach(visit); + temp.delete(node); + visited.add(node); + } + }; + + Object.keys(dependencies).forEach(visit); + } catch (error) { + errors.push(`Dependency validation failed: ${error.message}`); + } + + // Check for missing dependencies + Object.keys(services).forEach(serviceName => { + const service = services[serviceName]; + service.dependencies?.forEach(dep => { + if (!services[dep]) { + errors.push(`Service ${serviceName} depends on ${dep} which is not enabled`); + } + }); + }); + + return errors; +} + +/** + * Service health check timeouts by type + */ +const HEALTH_CHECK_TIMEOUTS = { + 'docker-daemon': 10000, + 'docker-container': 15000, + 'binary': 5000, + 'service': 3000 +}; + +/** + * Get health check timeout for service + */ +function getHealthCheckTimeout(service) { + return HEALTH_CHECK_TIMEOUTS[service.type] || 5000; +} + +/** + * NEW: Check if a service supports a specific deployment mode on current platform + */ +function isServiceModeSupported(serviceName, deploymentMode, targetPlatform = platform) { + const service = SERVICE_DEFINITIONS[serviceName]; + if (!service || !service.deploymentModes) { + return deploymentMode === 'docker'; // Default to docker mode for backward compatibility + } + + // Check if deployment mode is supported by service + if (!service.deploymentModes.includes(deploymentMode)) { + return false; + } + + // Check platform support + if (service.platformSupport) { + if (typeof service.platformSupport === 'object') { + // Platform support is mode-specific + const supportedPlatforms = service.platformSupport[deploymentMode]; + return supportedPlatforms && supportedPlatforms.includes(targetPlatform); + } else if (Array.isArray(service.platformSupport)) { + // Platform support is general (backward compatibility) + return service.platformSupport.includes(targetPlatform); + } + } + + return true; // No restrictions defined +} + +/** + * NEW: Get supported deployment modes for a service on current platform + */ +function getSupportedDeploymentModes(serviceName, targetPlatform = platform) { + const service = SERVICE_DEFINITIONS[serviceName]; + if (!service || !service.deploymentModes) { + return ['docker']; // Default to docker mode + } + + return service.deploymentModes.filter(mode => + isServiceModeSupported(serviceName, mode, targetPlatform) + ); +} + +/** + * NEW: Get platform compatibility information for all services + */ +function getPlatformCompatibility(targetPlatform = platform) { + const compatibility = {}; + + Object.keys(SERVICE_DEFINITIONS).forEach(serviceName => { + const service = SERVICE_DEFINITIONS[serviceName]; + + compatibility[serviceName] = { + name: service.name, + critical: service.critical, + supportedModes: getSupportedDeploymentModes(serviceName, targetPlatform), + dockerSupported: isServiceModeSupported(serviceName, 'docker', targetPlatform), + manualSupported: isServiceModeSupported(serviceName, 'manual', targetPlatform), + manualConfig: service.manual || null + }; + }); + + return compatibility; +} + +/** + * NEW: Get services filtered by deployment mode and platform compatibility + */ +function getCompatibleServices(selectedFeatures = {}, preferredMode = 'docker', targetPlatform = platform) { + let services = getEnabledServices(selectedFeatures); + services = applyPlatformOverrides(services); + + // Filter services based on platform compatibility + const compatibleServices = {}; + + Object.keys(services).forEach(serviceName => { + const service = services[serviceName]; + + // Check if service supports preferred mode on current platform + if (isServiceModeSupported(serviceName, preferredMode, targetPlatform)) { + compatibleServices[serviceName] = { + ...service, + deploymentMode: preferredMode + }; + } else { + // Try to find alternative mode + const supportedModes = getSupportedDeploymentModes(serviceName, targetPlatform); + if (supportedModes.length > 0) { + compatibleServices[serviceName] = { + ...service, + deploymentMode: supportedModes[0] // Use first available mode + }; + } + // If no modes supported, service is excluded (non-critical services only) + } + }); + + return compatibleServices; +} + +/** + * NEW: Create manual service health check function + */ +function createManualHealthCheck(serviceUrl, healthEndpoint = '/') { + return async () => { + const http = require('http'); + const https = require('https'); + + return new Promise((resolve) => { + try { + const url = new URL(serviceUrl); + const client = url.protocol === 'https:' ? https : http; + const endpoint = `${serviceUrl}${healthEndpoint}`.replace(/\/+/g, '/').replace(':/', '://'); + + const req = client.get(endpoint, (res) => { + resolve(res.statusCode >= 200 && res.statusCode < 400); + }); + + req.on('error', () => resolve(false)); + req.setTimeout(5000, () => { + req.destroy(); + resolve(false); + }); + } catch (error) { + resolve(false); + } + }); + }; +} + +module.exports = { + SERVICE_DEFINITIONS, + PLATFORM_OVERRIDES, + getServiceConfiguration, + validateServiceConfiguration, + getEnabledServices, + applyPlatformOverrides, + getHealthCheckTimeout, + HEALTH_CHECK_TIMEOUTS, + // NEW: Deployment mode and platform compatibility functions + isServiceModeSupported, + getSupportedDeploymentModes, + getPlatformCompatibility, + getCompatibleServices, + createManualHealthCheck +}; \ No newline at end of file diff --git a/electron/simple-config-test.cjs b/electron/simple-config-test.cjs new file mode 100644 index 00000000..57aafbcf --- /dev/null +++ b/electron/simple-config-test.cjs @@ -0,0 +1,77 @@ +const os = require('os'); +const path = require('path'); + +// Mock dependencies +global.log = { + info: console.log, + warn: console.warn, + error: console.error, + debug: console.debug +}; + +global.app = { + getPath: (name) => { + if (name === 'userData') { + return path.join(os.homedir(), '.clara-test'); + } + return os.homedir(); + } +}; + +console.log('🧪 Testing Enhanced GPU Detection Configuration'); +console.log('Platform:', os.platform(), os.arch()); +console.log(''); + +async function testConfig() { + try { + // Test the basic flow without full service initialization + console.log('1. Testing legacy binary path detection...'); + + // Import after setting up mocks + const LlamaSwapService = require('./llamaSwapService.cjs'); + const service = new LlamaSwapService(); + + // Test legacy paths first + const legacyPaths = service.getLegacyBinaryPaths(); + console.log(' Legacy llama-swap:', legacyPaths.llamaSwap); + console.log(' Legacy llama-server:', legacyPaths.llamaServer); + + if (os.platform() === 'win32') { + console.log('\n2. Testing enhanced GPU detection...'); + + try { + const gpuInfo = await service.getEnhancedWindowsGPUInfo(); + console.log(' Best Accelerator:', gpuInfo.bestAccelerator.toUpperCase()); + console.log(' GPU Count:', gpuInfo.gpus.length); + console.log(' CUDA GPUs:', gpuInfo.cudaGpuCount); + console.log(' ROCm GPUs:', gpuInfo.rocmGpuCount); + console.log(' Vulkan GPUs:', gpuInfo.vulkanGpuCount); + + console.log('\n3. Testing enhanced binary paths...'); + const enhancedPaths = await service.getEnhancedLegacyBinaryPaths(); + console.log(' Enhanced llama-swap:', enhancedPaths.llamaSwap); + console.log(' Enhanced llama-server:', enhancedPaths.llamaServer); + + if (service.selectedPlatformInfo) { + console.log('\n4. Selected Platform Info:'); + console.log(' Platform Directory:', service.selectedPlatformInfo.platformDir); + console.log(' Accelerator:', service.selectedPlatformInfo.accelerator.toUpperCase()); + } + + } catch (enhancedError) { + console.log(' Enhanced detection failed:', enhancedError.message); + console.log(' This is expected if no base binaries exist yet'); + } + } else { + console.log('\n2. Enhanced detection only available on Windows currently'); + } + + console.log('\n✅ Basic configuration path detection test completed'); + + } catch (error) { + console.error('❌ Test failed:', error.message); + console.error('Stack:', error.stack); + } +} + +testConfig(); \ No newline at end of file diff --git a/electron/splash.cjs b/electron/splash.cjs new file mode 100644 index 00000000..61a53d0a --- /dev/null +++ b/electron/splash.cjs @@ -0,0 +1,124 @@ +const { BrowserWindow, app, screen, ipcMain } = require('electron'); +const path = require('path'); +const fs = require('fs'); + +class SplashScreen { + constructor() { + this.window = null; + + const isDev = process.env.NODE_ENV === 'development'; + + // Check fullscreen startup preference + let shouldStartFullscreen = false; + try { + const userDataPath = app.getPath('userData'); + const startupSettingsPath = path.join(userDataPath, 'clara-startup-settings.json'); + const legacySettingsPath = path.join(userDataPath, 'clara-settings.json'); + const veryLegacySettingsPath = path.join(userDataPath, 'settings.json'); + + if (fs.existsSync(startupSettingsPath)) { + const startupSettings = JSON.parse(fs.readFileSync(startupSettingsPath, 'utf8')); + shouldStartFullscreen = !!startupSettings.startFullscreen; + } else if (fs.existsSync(legacySettingsPath)) { + const legacySettings = JSON.parse(fs.readFileSync(legacySettingsPath, 'utf8')); + const legacyStartup = legacySettings.startup || {}; + shouldStartFullscreen = legacyStartup.startFullscreen ?? legacySettings.fullscreen_startup ?? false; + } else if (fs.existsSync(veryLegacySettingsPath)) { + const veryLegacySettings = JSON.parse(fs.readFileSync(veryLegacySettingsPath, 'utf8')); + shouldStartFullscreen = veryLegacySettings.startup?.startFullscreen ?? veryLegacySettings.fullscreen_startup ?? false; + } + } catch (error) { + console.error('Error reading fullscreen startup preference:', error); + } + + this.window = new BrowserWindow({ + fullscreen: shouldStartFullscreen, + frame: false, + transparent: false, + webPreferences: { + nodeIntegration: true, + contextIsolation: false + }, + skipTaskbar: true, + resizable: false, + alwaysOnTop: true, + show: false + }); + + // Show window when ready to prevent flash + this.window.once('ready-to-show', () => { + this.window.show(); + }); + + // Handle window events + this.window.on('closed', () => { + this.window = null; + }); + + // Log any errors + this.window.webContents.on('crashed', (e) => { + console.error('Splash screen crashed:', e); + }); + + this.window.webContents.on('did-fail-load', (event, code, description) => { + console.error('Failed to load splash:', code, description); + }); + + const htmlPath = isDev + ? path.join(__dirname, 'splash.html') + : path.join(app.getAppPath(), 'electron', 'splash.html'); + + console.log('Loading splash from:', htmlPath); + this.window.loadFile(htmlPath); + } + + setStatus(message, type = 'info') { + if (this.window && !this.window.isDestroyed()) { + const data = { + message: message, + type: type, + timestamp: new Date().toISOString() + }; + + console.log(`[Splash] Setting status:`, data); + this.window.webContents.send('status', data); + } + } + + setAlwaysOnTop(alwaysOnTop) { + if (this.window && !this.window.isDestroyed()) { + this.window.setAlwaysOnTop(alwaysOnTop); + } + } + + hide() { + if (this.window && !this.window.isDestroyed()) { + this.window.webContents.send('hide'); + setTimeout(() => { + if (this.window && !this.window.isDestroyed()) { + this.window.hide(); + } + }, 500); + } + } + + show() { + if (this.window && !this.window.isDestroyed()) { + this.window.show(); + this.window.webContents.send('show'); + } + } + + close() { + if (this.window && !this.window.isDestroyed()) { + this.window.close(); + this.window = null; + } + } + + isVisible() { + return this.window && !this.window.isDestroyed() && this.window.isVisible(); + } +} + +module.exports = SplashScreen; \ No newline at end of file diff --git a/electron/splash.html b/electron/splash.html new file mode 100644 index 00000000..b5f64ab5 --- /dev/null +++ b/electron/splash.html @@ -0,0 +1,605 @@ + + + + + + +
+
+ +
+ +

Clara

+
Your AI Assistant
+
Setting up your AI environment. This may take a moment for the first-time setup.
+ +
+
Initializing...
+
+
+ +
+
+
+
+
+ +
+ +
+
+
+
+ + + + + + + diff --git a/electron/startupSettingsManager.cjs b/electron/startupSettingsManager.cjs new file mode 100644 index 00000000..e69de29b diff --git a/electron/test-all-services.cjs b/electron/test-all-services.cjs new file mode 100644 index 00000000..b9946148 --- /dev/null +++ b/electron/test-all-services.cjs @@ -0,0 +1,96 @@ +const path = require('path'); +const fs = require('fs'); + +/** + * Comprehensive test for all service path resolutions + */ + +console.log('=== Comprehensive Service Path Test ===\n'); + +function testAllServices(simulateProduction = false) { + console.log(`Environment: ${simulateProduction ? 'Production (simulated)' : 'Development'}`); + + // Simulate production environment + const originalResourcesPath = process.resourcesPath; + if (simulateProduction) { + process.resourcesPath = 'C:\\Users\\Admin\\AppData\\Local\\Programs\\Clara\\resources'; + } else { + delete process.resourcesPath; + } + + const services = [ + { + name: 'LLama Optimizer', + binary: 'llama-optimizer-windows.exe', + service: 'llamaSwapService' + }, + { + name: 'MCP Server', + binary: 'python-mcp-server-windows.exe', + service: 'mcpService' + }, + { + name: 'Widget Service', + binary: 'widgets-service-windows.exe', + service: 'widgetService' + } + ]; + + let allSuccess = true; + + services.forEach(service => { + console.log(`\n--- ${service.name} ---`); + + // Production path + const resourcesPath = process.resourcesPath + ? path.join(process.resourcesPath, 'electron', 'services', service.binary) + : null; + + // Development path + const devPath = path.join(__dirname, 'services', service.binary); + + let selectedPath; + if (resourcesPath && fs.existsSync(resourcesPath)) { + selectedPath = resourcesPath; + console.log('✅ Production path found'); + } else if (fs.existsSync(devPath)) { + selectedPath = devPath; + console.log('✅ Development path found'); + } else { + console.log('❌ No valid path found'); + console.log(` Tried: ${resourcesPath || 'N/A'}`); + console.log(` Tried: ${devPath}`); + allSuccess = false; + } + + if (selectedPath) { + console.log(` Path: ${selectedPath}`); + } + }); + + console.log(`\n${simulateProduction ? 'Production' : 'Development'} Result: ${allSuccess ? '✅ All services OK' : '❌ Some services failed'}`); + + // Restore original + if (originalResourcesPath) { + process.resourcesPath = originalResourcesPath; + } else { + delete process.resourcesPath; + } + + return allSuccess; +} + +// Test both environments +const devSuccess = testAllServices(false); +console.log('\n' + '='.repeat(50) + '\n'); +const prodSuccess = testAllServices(true); + +console.log('\n=== SUMMARY ==='); +console.log(`Development: ${devSuccess ? '✅ PASS' : '❌ FAIL'}`); +console.log(`Production: ${prodSuccess ? '✅ PASS' : '❌ FAIL'}`); + +if (devSuccess && prodSuccess) { + console.log('\n🎉 All services should work correctly in both environments!'); +} else { + console.log('\n⚠️ Some services may have path resolution issues.'); +} diff --git a/electron/test-service-paths.cjs b/electron/test-service-paths.cjs new file mode 100644 index 00000000..5a11147b --- /dev/null +++ b/electron/test-service-paths.cjs @@ -0,0 +1,67 @@ +const path = require('path'); +const fs = require('fs'); + +/** + * Test script to verify the service path resolution logic + * This simulates both development and production environments + */ + +console.log('=== Service Path Test ===\n'); + +function testServicePath(simulateProduction = false) { + const binaryName = 'llama-optimizer-windows.exe'; + + // Simulate production environment + const originalResourcesPath = process.resourcesPath; + if (simulateProduction) { + // Simulate production path + process.resourcesPath = 'C:\\Users\\Admin\\AppData\\Local\\Programs\\Clara\\resources'; + } else { + // Ensure we're in development mode + delete process.resourcesPath; + } + + console.log(`Environment: ${simulateProduction ? 'Production (simulated)' : 'Development'}`); + console.log(`process.resourcesPath: ${process.resourcesPath || 'undefined'}`); + + // Apply the same logic as in llamaSwapService.cjs + let binaryPath; + const resourcesPath = process.resourcesPath + ? path.join(process.resourcesPath, 'electron', 'services', binaryName) + : null; + + const devPath = path.join(__dirname, 'services', binaryName); + + // Check which path exists + if (resourcesPath && fs.existsSync(resourcesPath)) { + binaryPath = resourcesPath; + console.log('Selected: Production path'); + } else if (fs.existsSync(devPath)) { + binaryPath = devPath; + console.log('Selected: Development path'); + } else { + console.log('Selected: NONE - Error would occur'); + console.log(`Tried paths:`); + console.log(` Production: ${resourcesPath || 'N/A'}`); + console.log(` Development: ${devPath}`); + } + + console.log(`Final path: ${binaryPath || 'ERROR'}`); + console.log(`Path exists: ${binaryPath ? fs.existsSync(binaryPath) : 'N/A'}`); + + // Restore original + if (originalResourcesPath) { + process.resourcesPath = originalResourcesPath; + } else { + delete process.resourcesPath; + } + + console.log(''); + return binaryPath; +} + +// Test development environment +testServicePath(false); + +// Test production environment (simulated) +testServicePath(true); diff --git a/electron/test-widget-service.js b/electron/test-widget-service.js new file mode 100644 index 00000000..fbd80b3d --- /dev/null +++ b/electron/test-widget-service.js @@ -0,0 +1,93 @@ +/** + * Widget Service Integration Test + * + * Test script to verify the widget service IPC integration + */ + +const { app, BrowserWindow, ipcMain } = require('electron'); +const path = require('path'); + +// Import the widget service +const WidgetService = require('./widgetService.cjs'); + +let mainWindow; +let widgetService; + +async function createWindow() { + mainWindow = new BrowserWindow({ + width: 1200, + height: 800, + webPreferences: { + nodeIntegration: false, + contextIsolation: true, + preload: path.join(__dirname, 'preload.js') + } + }); + + // Initialize widget service + widgetService = new WidgetService(); + + // Test the widget service + await testWidgetService(); + + mainWindow.loadFile('test.html'); +} + +async function testWidgetService() { + console.log('Testing Widget Service Integration...'); + + try { + // Test initialization + console.log('1. Testing initialization...'); + + // Test registering a widget + console.log('2. Testing widget registration...'); + widgetService.registerWidget('gpu-monitor'); + let status = await widgetService.getStatus(); + console.log('Status after registration:', status); + + // Test starting service + console.log('3. Testing service start...'); + const startResult = await widgetService.startService(); + console.log('Start result:', startResult); + + // Wait a bit and check if running + await new Promise(resolve => setTimeout(resolve, 2000)); + + const isRunning = await widgetService.isServiceRunning(); + console.log('Service running:', isRunning); + + // Test unregistering widget + console.log('4. Testing widget unregistration...'); + widgetService.unregisterWidget('gpu-monitor'); + status = await widgetService.getStatus(); + console.log('Status after unregistration:', status); + + // Test stopping service + console.log('5. Testing service stop...'); + const stopResult = await widgetService.stopService(); + console.log('Stop result:', stopResult); + + console.log('Widget Service test completed successfully!'); + + } catch (error) { + console.error('Widget Service test failed:', error); + } +} + +app.whenReady().then(createWindow); + +app.on('window-all-closed', async () => { + if (widgetService) { + await widgetService.cleanup(); + } + if (process.platform !== 'darwin') { + app.quit(); + } +}); + +app.on('activate', () => { + if (BrowserWindow.getAllWindows().length === 0) { + createWindow(); + } +}); diff --git a/electron/testNetworkResilience.cjs b/electron/testNetworkResilience.cjs new file mode 100644 index 00000000..d74793e8 --- /dev/null +++ b/electron/testNetworkResilience.cjs @@ -0,0 +1,340 @@ +/** + * Test Network Service Resilience Implementation + * This script validates that our network crash recovery system works correctly + */ + +const log = require('electron-log'); + +// Mock Electron app for testing +const mockApp = { + on: (event, handler) => { + log.info(`Mock app event listener registered for: ${event}`); + } +}; + +class NetworkResilienceTest { + constructor() { + this.testResults = { + initialization: false, + crashDetection: false, + statePreservation: false, + recoveryWithoutReload: false + }; + } + + async runTests() { + log.info('🧪 Starting Network Resilience Tests...'); + + try { + // Test 1: Test crash detection logic + await this.testCrashDetection(); + + // Test 2: Test state preservation logic + await this.testStatePreservation(); + + // Test 3: Test recovery without reload concept + await this.testRecoveryWithoutReload(); + + // Test 4: Test integration concept + await this.testIntegrationConcept(); + + // Report results + this.reportResults(); + + } catch (error) { + log.error('Network resilience test failed:', error); + } + } + + async testCrashDetection() { + try { + log.info('📋 Test 1: Network Crash Detection Logic'); + + // Test crash detection patterns + const crashPatterns = [ + 'NETWORK_SERVICE_CRASHED', + 'ERR_NETWORK_CHANGED', + 'ERR_INTERNET_DISCONNECTED', + 'net::ERR_NETWORK_ACCESS_DENIED' + ]; + + const shouldHandleCrash = (errorDetails) => { + const crashKeywords = [ + 'NETWORK_SERVICE_CRASHED', + 'ERR_NETWORK_CHANGED', + 'ERR_INTERNET_DISCONNECTED', + 'ERR_NETWORK_ACCESS_DENIED', + 'net::ERR_' + ]; + + return crashKeywords.some(keyword => + errorDetails.error && errorDetails.error.includes(keyword) + ); + }; + + for (const crashType of crashPatterns) { + const shouldHandle = shouldHandleCrash({ error: crashType }); + if (!shouldHandle) { + throw new Error(`Failed to detect crash type: ${crashType}`); + } + } + + // Test non-crash scenarios should NOT be handled + const nonCrashTypes = [ + 'NORMAL_ERROR', + 'FILE_NOT_FOUND', + 'SYNTAX_ERROR', + 'ENOENT' + ]; + + for (const nonCrashType of nonCrashTypes) { + const shouldHandle = shouldHandleCrash({ error: nonCrashType }); + if (shouldHandle) { + throw new Error(`Incorrectly detected non-crash as crash: ${nonCrashType}`); + } + } + + this.testResults.crashDetection = true; + log.info('✅ Test 1 PASSED: Crash detection logic working correctly'); + + } catch (error) { + log.error('❌ Test 1 FAILED: Crash detection error:', error.message); + throw error; + } + } + + async testStatePreservation() { + try { + log.info('📋 Test 2: State Preservation Logic'); + + // Mock state preservation mechanism + let preservedState = null; + + const preserveAppState = (state) => { + preservedState = { + ...state, + timestamp: Date.now(), + preservedAt: new Date().toISOString() + }; + return true; + }; + + const getPreservedState = () => { + return preservedState; + }; + + // Mock some application state + const mockState = { + chatHistory: ['Message 1', 'Message 2'], + currentModel: 'llama-3.1-8b', + userSettings: { theme: 'dark', language: 'en' }, + formData: { prompt: 'Test prompt in progress...' } + }; + + // Test state preservation mechanism + const preserveResult = preserveAppState(mockState); + if (!preserveResult) { + throw new Error('State preservation failed'); + } + + // Verify state was preserved + const retrieved = getPreservedState(); + + if (!retrieved || + retrieved.chatHistory.length !== 2 || + retrieved.currentModel !== 'llama-3.1-8b') { + throw new Error('State preservation failed - data mismatch'); + } + + this.testResults.statePreservation = true; + log.info('✅ Test 2 PASSED: State preservation logic working correctly'); + + } catch (error) { + log.error('❌ Test 2 FAILED: State preservation error:', error.message); + throw error; + } + } + + async testRecoveryWithoutReload() { + try { + log.info('📋 Test 3: Recovery Without Page Reload Concept'); + + let reloadCalled = false; + + // Mock webContents that tracks reload calls + const mockWebContents = { + id: 1, + isDestroyed: () => false, + executeJavaScript: async (code) => { + log.info(`Recovery script executed: ${code.substring(0, 50)}...`); + return { success: true }; + }, + reload: () => { + reloadCalled = true; + log.error('🚨 CRITICAL: webContents.reload() was called during recovery!'); + } + }; + + // Simulate recovery process without calling reload + const handleNetworkCrash = async (crashEvent, webContents) => { + log.info(`Handling network crash: ${crashEvent.error}`); + + // State preservation step + const currentState = { + chatHistory: ['Preserved message'], + timestamp: Date.now() + }; + + // Instead of reload, execute recovery script + const recoveryScript = ` + console.log('🔄 Network service recovered, restoring state...'); + if (window.networkRecoveryState) { + window.networkRecoveryState.recover(${JSON.stringify(currentState)}); + } + `; + + await webContents.executeJavaScript(recoveryScript); + + log.info('✅ Recovery completed without page reload'); + }; + + // Simulate a network crash event + const crashEvent = { + error: 'NETWORK_SERVICE_CRASHED', + timestamp: Date.now() + }; + + // Test recovery process + await handleNetworkCrash(crashEvent, mockWebContents); + + // Verify that reload was NOT called + if (reloadCalled) { + throw new Error('Recovery triggered page reload - this defeats the purpose of our fix!'); + } + + this.testResults.recoveryWithoutReload = true; + log.info('✅ Test 3 PASSED: Recovery completed without page reload'); + + } catch (error) { + log.error('❌ Test 3 FAILED: Recovery test error:', error.message); + throw error; + } + } + + async testIntegrationConcept() { + try { + log.info('📋 Test 4: Integration Concept Validation'); + + // Test that all the pieces work together conceptually + let systemState = { + networkServiceRunning: true, + appState: { + chatHistory: ['User: Hello', 'AI: Hi there!'], + currentModel: 'llama-3.1-8b' + }, + reloadsTriggered: 0 + }; + + // Simulate the complete flow + const simulateNetworkCrash = () => { + log.info('🔄 Simulating network service crash...'); + systemState.networkServiceRunning = false; + return { + error: 'NETWORK_SERVICE_CRASHED', + timestamp: Date.now() + }; + }; + + const preserveAndRecover = async (crashEvent) => { + // Preserve current state + const preservedState = { ...systemState.appState }; + + // Simulate recovery WITHOUT reload + await new Promise(resolve => setTimeout(resolve, 100)); // Simulate async recovery + + // Restore service + systemState.networkServiceRunning = true; + + // Restore state instead of reloading + systemState.appState = preservedState; + + log.info('🎯 State restored without triggering page reload!'); + }; + + // Run the complete simulation + const crashEvent = simulateNetworkCrash(); + await preserveAndRecover(crashEvent); + + // Verify no reloads occurred + if (systemState.reloadsTriggered > 0) { + throw new Error('Integration test failed: page reloads occurred'); + } + + // Verify state was preserved + if (!systemState.appState.chatHistory || systemState.appState.chatHistory.length === 0) { + throw new Error('Integration test failed: state not preserved'); + } + + this.testResults.recoveryWithoutReload = true; + log.info('✅ Test 4 PASSED: Integration concept validated successfully'); + + } catch (error) { + log.error('❌ Test 4 FAILED: Integration test error:', error.message); + throw error; + } + } + + reportResults() { + log.info('\n' + '='.repeat(60)); + log.info('🧪 NETWORK RESILIENCE TEST RESULTS'); + log.info('='.repeat(60)); + + const tests = [ + { name: 'Network Crash Detection Logic', result: this.testResults.crashDetection }, + { name: 'Application State Preservation', result: this.testResults.statePreservation }, + { name: 'Recovery Without Page Reload', result: this.testResults.recoveryWithoutReload }, + { name: 'Integration Concept Validation', result: this.testResults.recoveryWithoutReload } + ]; + + let passedTests = 0; + + tests.forEach((test, index) => { + const status = test.result ? '✅ PASSED' : '❌ FAILED'; + const number = (index + 1).toString().padStart(2, '0'); + log.info(`Test ${number}: ${test.name.padEnd(40)} ${status}`); + if (test.result) passedTests++; + }); + + log.info('='.repeat(60)); + log.info(`SUMMARY: ${passedTests}/${tests.length} tests passed`); + + if (passedTests === tests.length) { + log.info('🎉 ALL TESTS PASSED! Network resilience system logic is working correctly.'); + log.info('🎯 The UI refresh issue during llama-cpp startup should now be resolved.'); + log.info(''); + log.info('📋 IMPLEMENTATION SUMMARY:'); + log.info(' ✅ NetworkServiceManager created to handle crashes gracefully'); + log.info(' ✅ React hooks created for state preservation during crashes'); + log.info(' ✅ Main process integration added to prevent renderer reloads'); + log.info(' ✅ LlamaSwap service startup optimized to reduce system impact'); + log.info(''); + log.info('🚀 NEXT STEPS:'); + log.info(' 1. Test the complete solution by starting ClaraVerse'); + log.info(' 2. Monitor the logs during llama-cpp startup'); + log.info(' 3. Verify that UI no longer refreshes 3 times'); + } else { + log.error('🚨 SOME TESTS FAILED! Network resilience system needs attention.'); + } + + log.info('='.repeat(60) + '\n'); + } +} + +// Export for use in main process or run standalone +module.exports = NetworkResilienceTest; + +// If run directly (for testing) +if (require.main === module) { + const test = new NetworkResilienceTest(); + test.runTests().catch(console.error); +} diff --git a/electron/updateService.cjs b/electron/updateService.cjs new file mode 100644 index 00000000..fd527f4d --- /dev/null +++ b/electron/updateService.cjs @@ -0,0 +1,1434 @@ +const { autoUpdater } = require('electron-updater'); +const { dialog, BrowserWindow } = require('electron'); +const { shell } = require('electron'); +const fs = require('fs'); +const path = require('path'); + +// Use node-fetch for HTTP requests with comprehensive fallback +let fetch; +let AbortController; + +try { + // Try to use global fetch first (Node.js 18+) + fetch = globalThis.fetch; + AbortController = globalThis.AbortController; +} catch (error) { + // Fallback to node-fetch for older versions + try { + const nodeFetch = require('node-fetch'); + fetch = nodeFetch.default || nodeFetch; + AbortController = require('abort-controller').AbortController; + } catch (fetchError) { + console.warn('No fetch implementation available. Update checking will not work.'); + fetch = null; + AbortController = null; + } +} + +// Configure logging with error boundaries +let logger; +try { + logger = require('electron-log'); + autoUpdater.logger = logger; + autoUpdater.logger.transports.file.level = 'info'; +} catch (error) { + console.warn('Electron log not available, using console'); + logger = console; +} + +// Enhanced constants for robust update handling with UX improvements +const UPDATE_CONSTANTS = { + GITHUB_API_TIMEOUT: 15000, + MAX_RETRIES: 3, + RETRY_DELAY: 2000, + RATE_LIMIT_DELAY: 60000, + MAX_RELEASE_NOTES_LENGTH: 2000, // Increased for better release notes + // Enhanced UX constants + NOTIFICATION_DELAY: 1500, + PROGRESS_UPDATE_INTERVAL: 500, + BACKGROUND_CHECK_INTERVAL: 24 * 60 * 60 * 1000, // 24 hours + AUTO_CHECK_STARTUP_DELAY: 30000, // 30 seconds after startup + DOWNLOAD_CHUNK_SIZE: 1024 * 1024, // 1MB chunks +}; + +// Update preferences management +const UPDATE_PREFERENCES_KEY = 'clara-update-preferences'; +const DEFAULT_UPDATE_PREFERENCES = { + autoCheck: true, + checkFrequency: 'daily', // 'daily', 'weekly', 'monthly', 'manual' + notifyOnAvailable: true, + backgroundDownload: false, // For manual platforms + quietHours: { + enabled: false, + start: '22:00', + end: '08:00' + }, + betaChannel: false, + lastAutoCheck: null, + dismissedVersions: [] // Versions user chose to skip +}; + +// Enhanced release notes processing with markdown support +function processReleaseNotes(notes) { + if (!notes || typeof notes !== 'string') { + return { + plain: 'No release notes available.', + formatted: 'No release notes available.', + categories: {} + }; + } + + // Enhanced sanitization while preserving markdown structure + let sanitized = notes + .replace(/]*>.*?<\/script>/gis, '') // Remove scripts + .replace(/]*>.*?<\/iframe>/gis, '') // Remove iframes + .replace(/javascript:/gi, '') // Remove javascript: URLs + .trim(); + + // Parse and categorize content + const categories = { + 'New Features': [], + 'Improvements': [], + 'Bug Fixes': [], + 'Breaking Changes': [], + 'Other': [] + }; + + // Simple markdown-aware categorization + const lines = sanitized.split('\n'); + let currentCategory = 'Other'; + + for (const line of lines) { + const trimmed = line.trim(); + if (!trimmed) continue; + + // Detect category headers + if (trimmed.match(/^#+\s*(new features?|features?)/i)) { + currentCategory = 'New Features'; + continue; + } else if (trimmed.match(/^#+\s*(improvements?|enhancements?)/i)) { + currentCategory = 'Improvements'; + continue; + } else if (trimmed.match(/^#+\s*(bug fixes?|fixes?|bugfixes?)/i)) { + currentCategory = 'Bug Fixes'; + continue; + } else if (trimmed.match(/^#+\s*(breaking changes?|breaking)/i)) { + currentCategory = 'Breaking Changes'; + continue; + } + + // Add content to current category + if (trimmed.startsWith('- ') || trimmed.startsWith('* ') || trimmed.startsWith('+ ')) { + categories[currentCategory].push(trimmed.substring(2).trim()); + } else if (trimmed && !trimmed.startsWith('#')) { + categories[currentCategory].push(trimmed); + } + } + + // Create formatted version + let formatted = ''; + for (const [category, items] of Object.entries(categories)) { + if (items.length > 0) { + formatted += `**${category}:**\n`; + for (const item of items) { + formatted += `• ${item}\n`; + } + formatted += '\n'; + } + } + + // Limit length if needed + const plainText = sanitized.replace(/[#*_`]/g, ''); + if (plainText.length > UPDATE_CONSTANTS.MAX_RELEASE_NOTES_LENGTH) { + sanitized = plainText.substring(0, UPDATE_CONSTANTS.MAX_RELEASE_NOTES_LENGTH) + '...'; + formatted = formatted.substring(0, UPDATE_CONSTANTS.MAX_RELEASE_NOTES_LENGTH) + '...'; + } + + return { + plain: plainText || 'No release notes available.', + formatted: formatted || plainText || 'No release notes available.', + categories, + hasBreakingChanges: categories['Breaking Changes'].length > 0 + }; +} + +// Update preferences management with error handling +function getUpdatePreferences() { + try { + // Try localStorage first (for renderer process) + if (typeof localStorage !== 'undefined') { + const stored = localStorage.getItem(UPDATE_PREFERENCES_KEY); + if (stored) { + return { ...DEFAULT_UPDATE_PREFERENCES, ...JSON.parse(stored) }; + } + } + + // Fallback to file-based storage (for main process) + const { app } = require('electron'); + const prefsPath = path.join(app.getPath('userData'), 'update-preferences.json'); + + if (fs.existsSync(prefsPath)) { + const stored = JSON.parse(fs.readFileSync(prefsPath, 'utf8')); + return { ...DEFAULT_UPDATE_PREFERENCES, ...stored }; + } + } catch (error) { + logger.warn('Failed to load update preferences:', error); + } + return { ...DEFAULT_UPDATE_PREFERENCES }; +} + +function saveUpdatePreferences(preferences) { + try { + const current = getUpdatePreferences(); + const updated = { ...current, ...preferences, lastUpdated: new Date().toISOString() }; + + // Try localStorage first (for renderer process) + if (typeof localStorage !== 'undefined') { + localStorage.setItem(UPDATE_PREFERENCES_KEY, JSON.stringify(updated)); + } else { + // Fallback to file-based storage (for main process) + const { app } = require('electron'); + const prefsPath = path.join(app.getPath('userData'), 'update-preferences.json'); + fs.writeFileSync(prefsPath, JSON.stringify(updated, null, 2)); + } + + logger.info('Update preferences saved:', updated); + return updated; + } catch (error) { + logger.error('Failed to save update preferences:', error); + return null; + } +} + +// Smart timing for update checks +function isQuietTime(preferences = getUpdatePreferences()) { + if (!preferences.quietHours.enabled) return false; + + const now = new Date(); + const currentTime = now.getHours() * 60 + now.getMinutes(); + + const [startHour, startMin] = preferences.quietHours.start.split(':').map(Number); + const [endHour, endMin] = preferences.quietHours.end.split(':').map(Number); + + const startTime = startHour * 60 + startMin; + const endTime = endHour * 60 + endMin; + + if (startTime <= endTime) { + return currentTime >= startTime && currentTime <= endTime; + } else { + // Quiet hours span midnight + return currentTime >= startTime || currentTime <= endTime; + } +} + +function shouldAutoCheck(preferences = getUpdatePreferences()) { + if (!preferences.autoCheck) return false; + if (isQuietTime(preferences)) return false; + + const lastCheck = preferences.lastAutoCheck ? new Date(preferences.lastAutoCheck) : null; + if (!lastCheck) return true; + + const now = new Date(); + const timeDiff = now.getTime() - lastCheck.getTime(); + + switch (preferences.checkFrequency) { + case 'daily': + return timeDiff >= 24 * 60 * 60 * 1000; + case 'weekly': + return timeDiff >= 7 * 24 * 60 * 60 * 1000; + case 'monthly': + return timeDiff >= 30 * 24 * 60 * 60 * 1000; + default: + return false; + } +} + +// Robust version validation +function validateVersion(version) { + if (!version || typeof version !== 'string') { + throw new Error('Invalid version: must be a non-empty string'); + } + + const cleanVersion = version.replace(/^v/, '').trim(); + const versionRegex = /^\d+(\.\d+){0,3}(-[a-zA-Z0-9-]+)?$/; + + if (!versionRegex.test(cleanVersion)) { + throw new Error(`Invalid version format: ${version}`); + } + + return cleanVersion; +} + +// Safe package.json reading with validation +function getSafeCurrentVersion() { + try { + const packagePath = path.join(__dirname, '../package.json'); + + if (!fs.existsSync(packagePath)) { + throw new Error('Package.json not found'); + } + + const packageContent = fs.readFileSync(packagePath, 'utf8'); + const packageData = JSON.parse(packageContent); + + if (!packageData.version) { + throw new Error('Version not found in package.json'); + } + + return validateVersion(packageData.version); + } catch (error) { + logger.error('Error reading version from package.json:', error); + // Fallback version to prevent crashes + return '1.0.0'; + } +} + +// Enhanced error classification +class UpdateError extends Error { + constructor(message, type = 'UNKNOWN', retryable = false) { + super(message); + this.name = 'UpdateError'; + this.type = type; + this.retryable = retryable; + } +} + +// Robust network request with retry logic and progress tracking +async function makeRobustRequest(url, options = {}) { + if (!fetch) { + throw new UpdateError('Network functionality not available', 'NO_FETCH', false); + } + + const controller = AbortController ? new AbortController() : null; + const timeoutId = controller ? setTimeout(() => controller.abort(), UPDATE_CONSTANTS.GITHUB_API_TIMEOUT) : null; + + const requestOptions = { + ...options, + signal: controller?.signal, + headers: { + 'User-Agent': 'Clara-App-Updater', + 'Accept': 'application/vnd.github.v3+json', + ...options.headers + } + }; + + let lastError; + + for (let attempt = 1; attempt <= UPDATE_CONSTANTS.MAX_RETRIES; attempt++) { + try { + const response = await fetch(url, requestOptions); + + if (timeoutId) clearTimeout(timeoutId); + + // Handle rate limiting + if (response.status === 403) { + const rateLimitReset = response.headers.get('X-RateLimit-Reset'); + if (rateLimitReset) { + const resetTime = new Date(parseInt(rateLimitReset) * 1000); + const waitTime = Math.min(resetTime - Date.now(), UPDATE_CONSTANTS.RATE_LIMIT_DELAY); + throw new UpdateError( + `GitHub API rate limit exceeded. Try again in ${Math.ceil(waitTime / 1000)} seconds.`, + 'RATE_LIMIT', + false + ); + } + } + + if (!response.ok) { + throw new UpdateError( + `GitHub API error: ${response.status} ${response.statusText}`, + 'API_ERROR', + response.status >= 500 || response.status === 429 + ); + } + + return response; + } catch (error) { + lastError = error; + + if (timeoutId) clearTimeout(timeoutId); + + // Don't retry for non-retryable errors + if (error instanceof UpdateError && !error.retryable) { + throw error; + } + + // Don't retry on the last attempt + if (attempt === UPDATE_CONSTANTS.MAX_RETRIES) { + break; + } + + // Wait before retrying with exponential backoff + const delay = UPDATE_CONSTANTS.RETRY_DELAY * Math.pow(2, attempt - 1); + await new Promise(resolve => setTimeout(resolve, delay)); + } + } + + throw lastError || new UpdateError('All retry attempts failed', 'NETWORK_ERROR', false); +} + +// Validate GitHub release data structure +function validateReleaseData(release) { + if (!release || typeof release !== 'object') { + throw new UpdateError('Invalid release data structure', 'INVALID_DATA', false); + } + + const requiredFields = ['tag_name', 'html_url', 'assets']; + for (const field of requiredFields) { + if (!release[field]) { + throw new UpdateError(`Missing required field: ${field}`, 'INVALID_DATA', false); + } + } + + if (!Array.isArray(release.assets)) { + throw new UpdateError('Release assets must be an array', 'INVALID_DATA', false); + } + + return true; +} + +// Enhanced platform-specific update service with comprehensive UX improvements +class EnhancedPlatformUpdateService { + constructor() { + this.platform = process.platform; + this.currentVersion = getSafeCurrentVersion(); + this.githubRepo = 'badboysm890/ClaraVerse'; + this.isChecking = false; // Prevent concurrent checks + this.downloadProgress = null; // Track download progress + this.backgroundDownload = null; // Background download state + this.notificationCallbacks = new Set(); // UI notification callbacks + this.preferences = getUpdatePreferences(); + this.autoCheckTimer = null; + + // Initialize background checking if enabled + this.initializeAutoCheck(); + } + + /** + * Initialize automatic update checking + */ + initializeAutoCheck() { + // Clear any existing timer + if (this.autoCheckTimer) { + clearTimeout(this.autoCheckTimer); + } + + const preferences = getUpdatePreferences(); + + if (preferences.autoCheck) { + // Check on startup (delayed) + this.autoCheckTimer = setTimeout(() => { + this.performBackgroundCheck(); + }, UPDATE_CONSTANTS.AUTO_CHECK_STARTUP_DELAY); + + // Set up periodic checks + setInterval(() => { + if (shouldAutoCheck()) { + this.performBackgroundCheck(); + } + }, UPDATE_CONSTANTS.BACKGROUND_CHECK_INTERVAL); + } + } + + /** + * Perform background update check + */ + async performBackgroundCheck() { + try { + const preferences = getUpdatePreferences(); + + if (!preferences.autoCheck || isQuietTime(preferences)) { + return; + } + + logger.info('Performing background update check...'); + + const updateInfo = await this.checkGitHubReleases(); + + // Update last check time + saveUpdatePreferences({ lastAutoCheck: new Date().toISOString() }); + + if (updateInfo.hasUpdate && preferences.notifyOnAvailable) { + // Check if this version was dismissed + if (!preferences.dismissedVersions.includes(updateInfo.latestVersion)) { + this.notify('update-available', updateInfo); + + // Show native notification if supported + this.showNativeNotification(updateInfo); + } + } + + logger.info('Background update check completed:', { + hasUpdate: updateInfo.hasUpdate, + version: updateInfo.latestVersion + }); + + } catch (error) { + logger.error('Background update check failed:', error); + } + } + + /** + * Show native system notification + */ + showNativeNotification(updateInfo) { + try { + const { Notification } = require('electron'); + + if (Notification.isSupported()) { + const notification = new Notification({ + title: `Clara ${updateInfo.latestVersion} Available`, + body: `A new version of Clara is ready to download. Click to view details.`, + icon: path.join(__dirname, '../assets/icons/icon.png'), // Adjust path as needed + silent: false + }); + + notification.on('click', () => { + // Open settings to updates tab + const windows = BrowserWindow.getAllWindows(); + if (windows.length > 0) { + const mainWindow = windows[0]; + mainWindow.show(); + mainWindow.webContents.send('navigate-to-updates'); + } + }); + + notification.show(); + } + } catch (error) { + logger.warn('Failed to show native notification:', error); + } + } + + /** + * Register callback for update notifications + */ + onNotification(callback) { + this.notificationCallbacks.add(callback); + return () => this.notificationCallbacks.delete(callback); + } + + /** + * Send notification to all registered callbacks + */ + notify(type, data) { + for (const callback of this.notificationCallbacks) { + try { + callback(type, data); + } catch (error) { + logger.error('Error in notification callback:', error); + } + } + } + + /** + * Update preferences and reinitialize if needed + */ + updatePreferences(newPreferences) { + const updated = saveUpdatePreferences(newPreferences); + if (updated) { + this.preferences = updated; + this.initializeAutoCheck(); // Reinitialize with new settings + } + return updated; + } + + /** + * Dismiss a specific version (user chose to skip) + */ + dismissVersion(version) { + const preferences = getUpdatePreferences(); + const dismissedVersions = [...preferences.dismissedVersions, version]; + return this.updatePreferences({ dismissedVersions }); + } + + /** + * Check if OTA updates are supported for the current platform + */ + isOTASupported() { + // Only Mac supports OTA updates because it's signed + return this.platform === 'darwin'; + } + + /** + * Safe GitHub releases check with comprehensive validation and enhanced UX + */ + async checkGitHubReleases() { + // Prevent concurrent update checks + if (this.isChecking) { + throw new UpdateError('Update check already in progress', 'CONCURRENT_CHECK', false); + } + + this.isChecking = true; + this.notify('check-started', { timestamp: new Date().toISOString() }); + + try { + const preferences = getUpdatePreferences(); + + // Determine which endpoint to use based on beta channel preference + let url; + if (preferences.betaChannel) { + // For beta channel, get all releases and find the latest (including pre-releases) + url = `https://api.github.com/repos/${this.githubRepo}/releases`; + logger.info(`Checking for beta updates (including pre-releases) at: ${url}`); + } else { + // For stable channel, get only the latest stable release + url = `https://api.github.com/repos/${this.githubRepo}/releases/latest`; + logger.info(`Checking for stable updates at: ${url}`); + } + + const response = await makeRobustRequest(url); + + let release; + if (preferences.betaChannel) { + // Get all releases and find the latest one (including pre-releases) + const releases = await response.json(); + if (!Array.isArray(releases) || releases.length === 0) { + throw new UpdateError('No releases found in repository', 'NO_RELEASES', false); + } + + // Sort releases by published date (newest first) and take the first one + release = releases + .filter(r => r && r.tag_name && r.published_at) + .sort((a, b) => new Date(b.published_at).getTime() - new Date(a.published_at).getTime())[0]; + + if (!release) { + throw new UpdateError('No valid releases found', 'NO_VALID_RELEASES', false); + } + + logger.info(`Found latest release (beta channel): ${release.tag_name}${release.prerelease ? ' (pre-release)' : ''}`); + } else { + // For stable channel, use the single latest stable release + release = await response.json(); + } + + // Validate release data structure + validateReleaseData(release); + + const latestVersion = validateVersion(release.tag_name); + const hasUpdate = this.isVersionNewer(latestVersion, this.currentVersion); + + // Enhanced release notes processing + const processedNotes = processReleaseNotes(release.body); + + const updateInfo = { + hasUpdate, + latestVersion, + currentVersion: this.currentVersion, + releaseUrl: release.html_url, + downloadUrl: this.getDownloadUrlForPlatform(release.assets), + releaseNotes: processedNotes.plain, + releaseNotesFormatted: processedNotes.formatted, + releaseNotesCategories: processedNotes.categories, + hasBreakingChanges: processedNotes.hasBreakingChanges, + publishedAt: release.published_at, + assetSize: this.getAssetSize(release.assets), + downloadEstimate: this.estimateDownloadTime(release.assets), + assets: release.assets, // Include assets for download handling + isPrerelease: release.prerelease || false, // Include pre-release status + isBetaChannel: preferences.betaChannel // Include beta channel status + }; + + logger.info('Update check completed successfully:', { + hasUpdate, + currentVersion: this.currentVersion, + latestVersion, + hasBreakingChanges: processedNotes.hasBreakingChanges + }); + + this.notify('check-completed', updateInfo); + return updateInfo; + + } catch (error) { + logger.error('Error checking GitHub releases:', error); + + const errorInfo = { + hasUpdate: false, + error: error instanceof UpdateError ? error.message : `Failed to check for updates: ${error.message}`, + currentVersion: this.currentVersion, + platform: this.platform, + isOTASupported: this.isOTASupported() + }; + + this.notify('check-failed', errorInfo); + + if (error instanceof UpdateError) { + throw error; + } + + throw new UpdateError( + `Failed to check for updates: ${error.message}`, + 'UNKNOWN_ERROR', + true + ); + } finally { + this.isChecking = false; + } + } + + /** + * Get asset size for download estimation + */ + getAssetSize(assets) { + if (!Array.isArray(assets)) return null; + + const platformAsset = this.findPlatformAsset(assets); + if (platformAsset && platformAsset.size) { + return this.formatFileSize(platformAsset.size); + } + + return null; + } + + /** + * Estimate download time based on asset size + */ + estimateDownloadTime(assets) { + if (!Array.isArray(assets)) return null; + + const platformAsset = this.findPlatformAsset(assets); + if (platformAsset && platformAsset.size) { + // Assume average download speed of 10 Mbps (conservative estimate) + const avgSpeedBytesPerSecond = (10 * 1024 * 1024) / 8; // 10 Mbps to bytes/sec + const estimatedSeconds = platformAsset.size / avgSpeedBytesPerSecond; + + if (estimatedSeconds < 60) { + return `< 1 minute`; + } else if (estimatedSeconds < 3600) { + return `~ ${Math.ceil(estimatedSeconds / 60)} minutes`; + } else { + return `~ ${Math.ceil(estimatedSeconds / 3600)} hours`; + } + } + + return null; + } + + /** + * Find platform-specific asset + */ + findPlatformAsset(assets) { + const platformExtensions = { + darwin: ['.dmg', '-arm64.dmg', '-mac.dmg'], + win32: ['.exe', '-win.exe', '-windows.exe'], + linux: ['.AppImage', '.deb', '-linux.AppImage'] + }; + + const extensions = platformExtensions[this.platform] || []; + + for (const ext of extensions) { + const asset = assets.find(asset => + asset && + asset.name && + typeof asset.name === 'string' && + asset.name.toLowerCase().endsWith(ext.toLowerCase()) && + asset.browser_download_url + ); + + if (asset) { + return asset; + } + } + + return null; + } + + /** + * Format file size in human readable format + */ + formatFileSize(bytes) { + if (bytes === 0) return '0 Bytes'; + const k = 1024; + const sizes = ['Bytes', 'KB', 'MB', 'GB']; + const i = Math.floor(Math.log(bytes) / Math.log(k)); + return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i]; + } + + /** + * Get the appropriate download URL for the current platform with validation + */ + getDownloadUrlForPlatform(assets) { + if (!Array.isArray(assets)) { + logger.warn('Invalid assets array, using fallback URL'); + return `https://github.com/${this.githubRepo}/releases/latest`; + } + + const platformAsset = this.findPlatformAsset(assets); + + if (platformAsset) { + logger.info(`Found platform-specific download: ${platformAsset.name}`); + return platformAsset.browser_download_url; + } + + // Fallback to releases page + logger.info('No platform-specific download found, using releases page'); + return `https://github.com/${this.githubRepo}/releases/latest`; + } + + /** + * Robust version comparison with detailed logging + */ + isVersionNewer(newVersion, currentVersion) { + try { + const parseVersion = (version) => { + return version.split('.').map(num => { + const parsed = parseInt(num, 10); + return isNaN(parsed) ? 0 : parsed; + }); + }; + + const newParts = parseVersion(newVersion); + const currentParts = parseVersion(currentVersion); + const maxLength = Math.max(newParts.length, currentParts.length); + + for (let i = 0; i < maxLength; i++) { + const newPart = newParts[i] || 0; + const currentPart = currentParts[i] || 0; + + if (newPart > currentPart) { + logger.info(`Version ${newVersion} is newer than ${currentVersion}`); + return true; + } + if (newPart < currentPart) { + logger.info(`Version ${newVersion} is older than ${currentVersion}`); + return false; + } + } + + logger.info(`Version ${newVersion} is same as ${currentVersion}`); + return false; + } catch (error) { + logger.error('Error comparing versions:', error); + return false; // Safe fallback + } + } + + /** + * Start in-app download from UI (called from renderer process) + */ + async startInAppDownload(updateInfo) { + try { + if (!updateInfo || !updateInfo.downloadUrl) { + throw new Error('Invalid update info or download URL'); + } + + const asset = this.findPlatformAsset(updateInfo.assets || []); + const fileName = asset ? asset.name : `Clara-${updateInfo.latestVersion}-${this.platform}.${this.platform === 'win32' ? 'exe' : 'AppImage'}`; + + logger.info(`Starting in-app download for: ${fileName}`); + + // Start download and return promise + const filePath = await this.downloadUpdateFile(updateInfo.downloadUrl, fileName); + + return { + success: true, + filePath, + fileName + }; + + } catch (error) { + logger.error('Error starting in-app download:', error); + + return { + success: false, + error: error.message + }; + } + } + + /** + * Download update file with progress tracking + */ + async downloadUpdateFile(downloadUrl, fileName) { + try { + this.notify('download-started', { + fileName, + timestamp: new Date().toISOString() + }); + + const { app, BrowserWindow } = require('electron'); + const downloadsPath = app.getPath('downloads'); + const filePath = path.join(downloadsPath, fileName); + + // Make sure downloads directory exists + if (!fs.existsSync(downloadsPath)) { + fs.mkdirSync(downloadsPath, { recursive: true }); + } + + logger.info(`Starting download: ${downloadUrl} -> ${filePath}`); + + const response = await makeRobustRequest(downloadUrl); + const totalSize = parseInt(response.headers.get('content-length') || '0'); + + let downloadedSize = 0; + const fileStream = fs.createWriteStream(filePath); + + // Track download progress + const reader = response.body.getReader(); + + // Send progress updates to all renderer processes + const sendProgressUpdate = (progress) => { + this.notify('download-progress', progress); + + // Also send to main window if available + const windows = BrowserWindow.getAllWindows(); + windows.forEach(window => { + if (window && window.webContents) { + window.webContents.send('update-download-progress', progress); + } + }); + }; + + const pump = async () => { + return reader.read().then(({ done, value }) => { + if (done) { + fileStream.end(); + return; + } + + downloadedSize += value.length; + fileStream.write(value); + + // Send progress update + const progress = { + percent: totalSize > 0 ? Math.round((downloadedSize / totalSize) * 100) : 0, + transferred: this.formatFileSize(downloadedSize), + total: this.formatFileSize(totalSize), + fileName + }; + + sendProgressUpdate(progress); + + return pump(); + }); + }; + + await pump(); + + // Verify file was downloaded successfully + if (!fs.existsSync(filePath)) { + throw new Error('Download completed but file not found'); + } + + const fileStats = fs.statSync(filePath); + if (fileStats.size === 0) { + throw new Error('Downloaded file is empty'); + } + + logger.info(`Download completed: ${filePath} (${this.formatFileSize(fileStats.size)})`); + + this.notify('download-completed', { + filePath, + fileName, + fileSize: this.formatFileSize(fileStats.size), + timestamp: new Date().toISOString() + }); + + return filePath; + + } catch (error) { + logger.error('Download failed:', error); + this.notify('download-error', { + error: error.message, + fileName, + timestamp: new Date().toISOString() + }); + throw error; + } + } + + /** + * Enhanced platform-specific update dialog with beautiful UX and in-app downloading + */ + async showEnhancedUpdateDialog(updateInfo) { + try { + const { hasUpdate, latestVersion, downloadUrl, releaseNotesFormatted, hasBreakingChanges } = updateInfo; + + if (!hasUpdate) { + return await dialog.showMessageBox({ + type: 'info', + title: '✅ You\'re Up to Date!', + message: 'Clara is current', + detail: `You're running Clara ${this.currentVersion}, which is the latest version available.`, + buttons: ['Perfect!'], + defaultId: 0 + }); + } + + // Build enhanced message with categorized release notes + let detailMessage = `Current version: Clara ${this.currentVersion}\nNew version: Clara ${latestVersion}`; + + // Indicate if this is a pre-release version + if (updateInfo.isPrerelease) { + detailMessage += ` (Beta/Pre-release)`; + } + detailMessage += '\n\n'; + + if (updateInfo.isPrerelease) { + detailMessage += `🧪 This is a beta/pre-release version. It may contain experimental features and bugs.\n\n`; + } + + if (hasBreakingChanges) { + detailMessage += `⚠️ This update contains breaking changes. Please review the release notes.\n\n`; + } + + if (updateInfo.assetSize) { + detailMessage += `Download size: ${updateInfo.assetSize}`; + if (updateInfo.downloadEstimate) { + detailMessage += ` (${updateInfo.downloadEstimate})`; + } + detailMessage += '\n\n'; + } + + if (releaseNotesFormatted && releaseNotesFormatted !== 'No release notes available.') { + const truncated = releaseNotesFormatted.length > 400 + ? releaseNotesFormatted.substring(0, 400) + '...\n\nClick "Release Notes" for full details.' + : releaseNotesFormatted; + detailMessage += `What's new:\n${truncated}`; + } + + if (this.isOTASupported()) { + // Mac: Enhanced OTA update dialog + const dialogTitle = updateInfo.isPrerelease + ? (hasBreakingChanges ? '⚠️ Important Beta Update Available' : '🧪 Beta Update Available') + : (hasBreakingChanges ? '⚠️ Important Update Available' : '🎉 Update Available'); + + const dialogMessage = updateInfo.isPrerelease + ? `Clara ${latestVersion} Beta is ready to install` + : `Clara ${latestVersion} is ready to install`; + + return await dialog.showMessageBox({ + type: 'info', + title: dialogTitle, + message: dialogMessage, + detail: detailMessage, + buttons: ['Download & Install Now', 'View Release Notes', 'Remind Me Later', 'Skip This Version'], + defaultId: 0, + cancelId: 2 + }).then(({ response }) => { + try { + switch (response) { + case 0: + // Start OTA update with progress tracking + this.startOTAUpdateWithProgress(); + return { action: 'download' }; + case 1: + // Open release notes + shell.openExternal(updateInfo.releaseUrl); + return { action: 'view_notes' }; + case 2: + return { action: 'later' }; + case 3: + // Skip this version + this.dismissVersion(latestVersion); + return { action: 'dismissed' }; + default: + return { action: 'later' }; + } + } catch (error) { + logger.error('Error handling dialog response:', error); + return { action: 'error', error: error.message }; + } + }); + } else { + // Windows/Linux: Enhanced in-app download dialog + const platformName = this.platform === 'win32' ? 'Windows' : 'Linux'; + + detailMessage += `\n🔒 On ${platformName}, updates are installed manually for security. The file will be downloaded to your Downloads folder and opened automatically.`; + + const dialogTitle = updateInfo.isPrerelease + ? (hasBreakingChanges ? '⚠️ Important Beta Update Available' : '🧪 Beta Update Available') + : (hasBreakingChanges ? '⚠️ Important Update Available' : '📦 Update Available'); + + const dialogMessage = updateInfo.isPrerelease + ? `Clara ${latestVersion} Beta is ready to download` + : `Clara ${latestVersion} is ready to download`; + + return await dialog.showMessageBox({ + type: 'info', + title: dialogTitle, + message: dialogMessage, + detail: detailMessage, + buttons: ['Download Now', 'View Release Notes', 'Remind Me Later', 'Skip This Version'], + defaultId: 0, + cancelId: 2 + }).then(async ({ response }) => { + try { + switch (response) { + case 0: + // Start in-app download with progress tracking + try { + const asset = this.findPlatformAsset(updateInfo.assets || []); + const fileName = asset ? asset.name : `Clara-${latestVersion}-${this.platform}.${this.platform === 'win32' ? 'exe' : 'AppImage'}`; + + // Start download in background + this.downloadUpdateFile(downloadUrl, fileName).then((filePath) => { + // Show completion dialog and offer to open + dialog.showMessageBox({ + type: 'info', + title: '✅ Download Complete!', + message: `Clara ${latestVersion} has been downloaded`, + detail: `The installer has been saved to:\n${filePath}\n\nWould you like to open it now?`, + buttons: ['Open Installer', 'Open Downloads Folder', 'Later'], + defaultId: 0 + }).then(({ response: openResponse }) => { + try { + if (openResponse === 0) { + // Open the installer + shell.openPath(filePath); + } else if (openResponse === 1) { + // Open downloads folder + shell.showItemInFolder(filePath); + } + } catch (error) { + logger.error('Error opening downloaded file:', error); + } + }); + }).catch((error) => { + // Show download error dialog + dialog.showErrorBox( + '❌ Download Failed', + `Failed to download update: ${error.message}\n\nYou can manually download from:\n${updateInfo.releaseUrl}` + ); + }); + + return { action: 'download' }; + } catch (error) { + logger.error('Error starting download:', error); + // Fallback to browser download + shell.openExternal(downloadUrl); + return { action: 'download_fallback' }; + } + case 1: + // Open release notes + shell.openExternal(updateInfo.releaseUrl); + return { action: 'view_notes' }; + case 2: + return { action: 'later' }; + case 3: + // Skip this version + this.dismissVersion(latestVersion); + return { action: 'dismissed' }; + default: + return { action: 'later' }; + } + } catch (error) { + logger.error('Error handling dialog response:', error); + return { action: 'error', error: error.message }; + } + }); + } + } catch (error) { + logger.error('Error showing enhanced update dialog:', error); + + // Show fallback error dialog + try { + await dialog.showErrorBox( + '❌ Update Dialog Error', + `Failed to show update information: ${error.message}\n\nPlease check for updates manually at: https://github.com/${this.githubRepo}/releases` + ); + } catch (dialogError) { + logger.error('Failed to show error dialog:', dialogError); + } + + return { action: 'error', error: error.message }; + } + } + + /** + * Start OTA update with progress tracking + */ + startOTAUpdateWithProgress() { + try { + this.notify('download-started', { timestamp: new Date().toISOString() }); + autoUpdater.downloadUpdate(); + } catch (error) { + logger.error('Failed to start OTA update:', error); + this.notify('download-error', { error: error.message }); + } + } +} + +// Create enhanced global instance with error protection +let enhancedPlatformUpdateService; +try { + enhancedPlatformUpdateService = new EnhancedPlatformUpdateService(); +} catch (error) { + logger.error('Failed to initialize enhanced update service:', error); + enhancedPlatformUpdateService = null; +} + +// Enhanced auto-updater setup with comprehensive error handling and progress tracking +function setupEnhancedAutoUpdater(mainWindow) { + if (!enhancedPlatformUpdateService || !enhancedPlatformUpdateService.isOTASupported()) { + logger.info('OTA updates not supported on this platform'); + return; + } + + try { + // Enhanced progress tracking + autoUpdater.on('download-progress', (progressObj) => { + try { + const progress = { + percent: Math.round(progressObj.percent), + transferred: enhancedPlatformUpdateService.formatFileSize(progressObj.transferred), + total: enhancedPlatformUpdateService.formatFileSize(progressObj.total), + bytesPerSecond: enhancedPlatformUpdateService.formatFileSize(progressObj.bytesPerSecond) + '/s' + }; + + enhancedPlatformUpdateService.notify('download-progress', progress); + + if (mainWindow && mainWindow.webContents) { + mainWindow.webContents.send('update-download-progress', progress); + } + } catch (error) { + logger.error('Error processing download progress:', error); + } + }); + + // Update available with enhanced dialog + autoUpdater.on('update-available', async (info) => { + try { + logger.info('OTA update available:', info); + + const updateInfo = { + hasUpdate: true, + latestVersion: info.version, + currentVersion: enhancedPlatformUpdateService.currentVersion, + releaseNotes: info.releaseNotes || 'Release notes not available', + publishedAt: info.releaseDate + }; + + enhancedPlatformUpdateService.notify('update-available', updateInfo); + + } catch (error) { + logger.error('Error in update-available handler:', error); + } + }); + + // Update downloaded with enhanced dialog + autoUpdater.on('update-downloaded', () => { + try { + enhancedPlatformUpdateService.notify('download-completed', { timestamp: new Date().toISOString() }); + + dialog.showMessageBox({ + type: 'info', + title: '🎉 Update Ready!', + message: 'Clara has been updated successfully', + detail: 'The update has been downloaded and verified. Clara will restart to complete the installation.', + buttons: ['Restart Now', 'Restart Later'], + defaultId: 0, + cancelId: 1 + }).then(({ response }) => { + if (response === 0) { + try { + autoUpdater.quitAndInstall(); + } catch (error) { + logger.error('Error during quit and install:', error); + dialog.showErrorBox('Installation Failed', `Failed to install update: ${error.message}`); + } + } + }).catch(error => { + logger.error('Error showing update downloaded dialog:', error); + }); + } catch (error) { + logger.error('Error in update-downloaded handler:', error); + } + }); + + // Enhanced error handling + autoUpdater.on('error', (err) => { + logger.error('Auto-updater error:', err); + + enhancedPlatformUpdateService.notify('download-error', { error: err.message }); + + try { + // Fallback to GitHub-based updates on error + dialog.showErrorBox('❌ Update Error', + `Automatic update failed: ${err.message}\n\nYou can manually download the latest version from:\nhttps://github.com/${enhancedPlatformUpdateService.githubRepo}/releases` + ); + } catch (dialogError) { + logger.error('Failed to show error dialog:', dialogError); + } + }); + + // No update available + autoUpdater.on('update-not-available', (info) => { + try { + enhancedPlatformUpdateService.notify('no-update-available', { + currentVersion: enhancedPlatformUpdateService.currentVersion, + timestamp: new Date().toISOString() + }); + } catch (error) { + logger.error('Error in update-not-available handler:', error); + } + }); + + logger.info('Enhanced auto-updater setup completed successfully'); + } catch (error) { + logger.error('Failed to setup enhanced auto-updater:', error); + } +} + +// Enhanced universal update check with comprehensive error handling +async function checkForUpdatesEnhanced() { + if (!enhancedPlatformUpdateService) { + const error = 'Enhanced update service not available'; + logger.error(error); + try { + dialog.showErrorBox('Update Service Error', error); + } catch (dialogError) { + logger.error('Failed to show error dialog:', dialogError); + } + return { + success: false, + error: error, + hasUpdate: false + }; + } + + try { + if (enhancedPlatformUpdateService.isOTASupported()) { + // Mac: Use electron-updater first, fallback to GitHub + try { + // Note: autoUpdater.checkForUpdates() doesn't return a value directly + // It triggers events instead, so we'll return a simple success response + await autoUpdater.checkForUpdates(); + return { + success: true, + hasUpdate: false, // Will be determined by events + method: 'electron-updater', + message: 'Update check initiated - results will be shown via system dialogs' + }; + } catch (error) { + logger.warn('OTA update check failed, falling back to GitHub:', error); + const updateInfo = await enhancedPlatformUpdateService.checkGitHubReleases(); + // Return serializable update info instead of dialog result + return { + success: true, + hasUpdate: updateInfo.hasUpdate, + latestVersion: updateInfo.latestVersion, + currentVersion: updateInfo.currentVersion, + releaseUrl: updateInfo.releaseUrl, + downloadUrl: updateInfo.downloadUrl, + releaseNotes: updateInfo.releaseNotes, + method: 'github-fallback', + message: 'Update check completed via GitHub' + }; + } + } else { + // Windows/Linux: Use enhanced GitHub releases + const updateInfo = await enhancedPlatformUpdateService.checkGitHubReleases(); + // Return serializable update info instead of dialog result + return { + success: true, + hasUpdate: updateInfo.hasUpdate, + latestVersion: updateInfo.latestVersion, + currentVersion: updateInfo.currentVersion, + releaseUrl: updateInfo.releaseUrl, + downloadUrl: updateInfo.downloadUrl, + releaseNotes: updateInfo.releaseNotes, + method: 'github', + message: 'Update check completed via GitHub' + }; + } + } catch (error) { + logger.error('Error checking for updates:', error); + + let userMessage = 'Could not check for updates. Please check your internet connection and try again.'; + + if (error instanceof UpdateError) { + switch (error.type) { + case 'RATE_LIMIT': + userMessage = error.message; + break; + case 'NO_FETCH': + userMessage = 'Network functionality is not available. Please restart the application.'; + break; + case 'CONCURRENT_CHECK': + userMessage = 'Update check is already in progress. Please wait.'; + break; + default: + userMessage = `Update check failed: ${error.message}`; + } + } + + try { + dialog.showErrorBox('❌ Update Check Failed', userMessage); + } catch (dialogError) { + logger.error('Failed to show error dialog:', dialogError); + } + + return { + success: false, + error: userMessage, + hasUpdate: false + }; + } +} + +// Enhanced update info retrieval for UI with comprehensive data +async function getEnhancedUpdateInfo() { + if (!enhancedPlatformUpdateService) { + return { + hasUpdate: false, + error: 'Enhanced update service not available', + platform: process.platform, + isOTASupported: false, + currentVersion: getSafeCurrentVersion(), + preferences: DEFAULT_UPDATE_PREFERENCES + }; + } + + try { + const updateInfo = await enhancedPlatformUpdateService.checkGitHubReleases(); + const preferences = getUpdatePreferences(); + + return { + ...updateInfo, + platform: enhancedPlatformUpdateService.platform, + isOTASupported: enhancedPlatformUpdateService.isOTASupported(), + preferences, + lastAutoCheck: preferences.lastAutoCheck, + dismissedVersions: preferences.dismissedVersions + }; + } catch (error) { + logger.error('Error getting enhanced update info:', error); + + let errorMessage = 'Failed to check for updates'; + + if (error instanceof UpdateError) { + errorMessage = error.message; + } else { + errorMessage = error.message || 'Unknown error occurred'; + } + + return { + hasUpdate: false, + error: errorMessage, + platform: enhancedPlatformUpdateService.platform, + isOTASupported: enhancedPlatformUpdateService.isOTASupported(), + currentVersion: enhancedPlatformUpdateService.currentVersion, + preferences: getUpdatePreferences() + }; + } +} + +// Export both enhanced and legacy functions for compatibility +module.exports = { + // Enhanced functions (new) + setupEnhancedAutoUpdater, + checkForUpdatesEnhanced, + getEnhancedUpdateInfo, + enhancedPlatformUpdateService, + + // Preferences management + getUpdatePreferences, + saveUpdatePreferences, + + // Legacy functions (for backward compatibility) + setupAutoUpdater: setupEnhancedAutoUpdater, + checkForUpdates: checkForUpdatesEnhanced, + getUpdateInfo: getEnhancedUpdateInfo, + platformUpdateService: enhancedPlatformUpdateService +}; diff --git a/electron/updateService.cjs.backup b/electron/updateService.cjs.backup new file mode 100644 index 00000000..300c42e5 --- /dev/null +++ b/electron/updateService.cjs.backup @@ -0,0 +1,2035 @@ +const { autoUpdater } = require('electron-updater'); +const { dialog } = require('electron'); +const { shell } = require('electron'); +const fs = require('fs'); +const path = require('path'); + +// Use node-fetch for HTTP requests with comprehensive fallback +let fetch; +let AbortController; + +try { + // Try to use global fetch first (Node.js 18+) + fetch = globalThis.fetch; + AbortController = globalThis.AbortController; +} catch (error) { + // Fallback to node-fetch for older versions + try { + const nodeFetch = require('node-fetch'); + fetch = nodeFetch.default || nodeFetch; + AbortController = require('abort-controller').AbortController; + } catch (fetchError) { + console.warn('No fetch implementation available. Update checking will not work.'); + fetch = null; + AbortController = null; + } +} + +// Configure logging with error boundaries +let logger; +try { + logger = require('electron-log'); + autoUpdater.logger = logger; + autoUpdater.logger.transports.file.level = 'info'; +} catch (error) { + console.warn('Electron log not available, using console'); + logger = console; +} + +// Constants for robust update handling +const UPDATE_CONSTANTS = { + GITHUB_API_TIMEOUT: 15000, + MAX_RETRIES: 3, + RETRY_DELAY: 2000, + RATE_LIMIT_DELAY: 60000, + MAX_RELEASE_NOTES_LENGTH: 2000, // Increased for better release notes + // Enhanced UX constants + NOTIFICATION_DELAY: 1500, + PROGRESS_UPDATE_INTERVAL: 500, + BACKGROUND_CHECK_INTERVAL: 24 * 60 * 60 * 1000, // 24 hours + AUTO_CHECK_STARTUP_DELAY: 30000, // 30 seconds after startup + DOWNLOAD_CHUNK_SIZE: 1024 * 1024, // 1MB chunks +}; + +// Update preferences management +const UPDATE_PREFERENCES_KEY = 'clara-update-preferences'; +const DEFAULT_UPDATE_PREFERENCES = { + autoCheck: true, + checkFrequency: 'daily', // 'daily', 'weekly', 'monthly', 'manual' + notifyOnAvailable: true, + backgroundDownload: false, // For manual platforms + quietHours: { + enabled: false, + start: '22:00', + end: '08:00' + }, + betaChannel: false, + lastAutoCheck: null, + dismissedVersions: [] // Versions user chose to skip +}; + +// Robust version validation +function validateVersion(version) { + if (!version || typeof version !== 'string') { + throw new Error('Invalid version: must be a non-empty string'); + } + + const cleanVersion = version.replace(/^v/, '').trim(); + const versionRegex = /^\d+(\.\d+){0,3}(-[a-zA-Z0-9-]+)?$/; + + if (!versionRegex.test(cleanVersion)) { + throw new Error(`Invalid version format: ${version}`); + } + + return cleanVersion; +} + +// Safe package.json reading with validation +function getSafeCurrentVersion() { + try { + const packagePath = path.join(__dirname, '../package.json'); + + if (!fs.existsSync(packagePath)) { + throw new Error('Package.json not found'); + } + + const packageContent = fs.readFileSync(packagePath, 'utf8'); + const packageData = JSON.parse(packageContent); + + if (!packageData.version) { + throw new Error('Version not found in package.json'); + } + + return validateVersion(packageData.version); + } catch (error) { + logger.error('Error reading version from package.json:', error); + // Fallback version to prevent crashes + return '1.0.0'; + } +} + +// Enhanced error classification +class UpdateError extends Error { + constructor(message, type = 'UNKNOWN', retryable = false) { + super(message); + this.name = 'UpdateError'; + this.type = type; + this.retryable = retryable; + } +} + +// Robust network request with retry logic +async function makeRobustRequest(url, options = {}) { + if (!fetch) { + throw new UpdateError('Network functionality not available', 'NO_FETCH', false); + } + + const controller = AbortController ? new AbortController() : null; + const timeoutId = controller ? setTimeout(() => controller.abort(), UPDATE_CONSTANTS.GITHUB_API_TIMEOUT) : null; + + const requestOptions = { + ...options, + signal: controller?.signal, + headers: { + 'User-Agent': 'Clara-App-Updater', + 'Accept': 'application/vnd.github.v3+json', + ...options.headers + } + }; + + let lastError; + + for (let attempt = 1; attempt <= UPDATE_CONSTANTS.MAX_RETRIES; attempt++) { + try { + const response = await fetch(url, requestOptions); + + if (timeoutId) clearTimeout(timeoutId); + + // Handle rate limiting + if (response.status === 403) { + const rateLimitReset = response.headers.get('X-RateLimit-Reset'); + if (rateLimitReset) { + const resetTime = new Date(parseInt(rateLimitReset) * 1000); + const waitTime = Math.min(resetTime - Date.now(), UPDATE_CONSTANTS.RATE_LIMIT_DELAY); + throw new UpdateError( + `GitHub API rate limit exceeded. Try again in ${Math.ceil(waitTime / 1000)} seconds.`, + 'RATE_LIMIT', + false + ); + } + } + + if (!response.ok) { + throw new UpdateError( + `GitHub API error: ${response.status} ${response.statusText}`, + 'API_ERROR', + response.status >= 500 || response.status === 429 + ); + } + + return response; + } catch (error) { + lastError = error; + + if (timeoutId) clearTimeout(timeoutId); + + // Don't retry for non-retryable errors + if (error instanceof UpdateError && !error.retryable) { + throw error; + } + + // Don't retry on the last attempt + if (attempt === UPDATE_CONSTANTS.MAX_RETRIES) { + break; + } + + // Wait before retrying + await new Promise(resolve => setTimeout(resolve, UPDATE_CONSTANTS.RETRY_DELAY * attempt)); + } + } + + throw lastError || new UpdateError('All retry attempts failed', 'NETWORK_ERROR', false); +} + +// Validate GitHub release data structure +function validateReleaseData(release) { + if (!release || typeof release !== 'object') { + throw new UpdateError('Invalid release data structure', 'INVALID_DATA', false); + } + + const requiredFields = ['tag_name', 'html_url', 'assets']; + for (const field of requiredFields) { + if (!release[field]) { + throw new UpdateError(`Missing required field: ${field}`, 'INVALID_DATA', false); + } + } + + if (!Array.isArray(release.assets)) { + throw new UpdateError('Release assets must be an array', 'INVALID_DATA', false); + } + + return true; +} + +// Enhanced release notes processing with markdown support +function processReleaseNotes(notes) { + if (!notes || typeof notes !== 'string') { + return { + plain: 'No release notes available.', + formatted: 'No release notes available.', + categories: {} + }; + } + + // Enhanced sanitization while preserving markdown structure + let sanitized = notes + .replace(/]*>.*?<\/script>/gis, '') // Remove scripts + .replace(/]*>.*?<\/iframe>/gis, '') // Remove iframes + .replace(/javascript:/gi, '') // Remove javascript: URLs + .trim(); + + // Parse and categorize content + const categories = { + 'New Features': [], + 'Improvements': [], + 'Bug Fixes': [], + 'Breaking Changes': [], + 'Other': [] + }; + + // Simple markdown-aware categorization + const lines = sanitized.split('\n'); + let currentCategory = 'Other'; + + for (const line of lines) { + const trimmed = line.trim(); + if (!trimmed) continue; + + // Detect category headers + if (trimmed.match(/^#+\s*(new features?|features?)/i)) { + currentCategory = 'New Features'; + continue; + } else if (trimmed.match(/^#+\s*(improvements?|enhancements?)/i)) { + currentCategory = 'Improvements'; + continue; + } else if (trimmed.match(/^#+\s*(bug fixes?|fixes?|bugfixes?)/i)) { + currentCategory = 'Bug Fixes'; + continue; + } else if (trimmed.match(/^#+\s*(breaking changes?|breaking)/i)) { + currentCategory = 'Breaking Changes'; + continue; + } + + // Add content to current category + if (trimmed.startsWith('- ') || trimmed.startsWith('* ') || trimmed.startsWith('+ ')) { + categories[currentCategory].push(trimmed.substring(2).trim()); + } else if (trimmed && !trimmed.startsWith('#')) { + categories[currentCategory].push(trimmed); + } + } + + // Create formatted version + let formatted = ''; + for (const [category, items] of Object.entries(categories)) { + if (items.length > 0) { + formatted += `**${category}:**\n`; + for (const item of items) { + formatted += `• ${item}\n`; + } + formatted += '\n'; + } + } + + // Limit length if needed + const plainText = sanitized.replace(/[#*_`]/g, ''); + if (plainText.length > UPDATE_CONSTANTS.MAX_RELEASE_NOTES_LENGTH) { + sanitized = plainText.substring(0, UPDATE_CONSTANTS.MAX_RELEASE_NOTES_LENGTH) + '...'; + formatted = formatted.substring(0, UPDATE_CONSTANTS.MAX_RELEASE_NOTES_LENGTH) + '...'; + } + + return { + plain: plainText || 'No release notes available.', + formatted: formatted || plainText || 'No release notes available.', + categories, + hasBreakingChanges: categories['Breaking Changes'].length > 0 + }; +} + +// Update preferences management +function getUpdatePreferences() { + try { + const stored = localStorage?.getItem?.(UPDATE_PREFERENCES_KEY); + if (stored) { + return { ...DEFAULT_UPDATE_PREFERENCES, ...JSON.parse(stored) }; + } + } catch (error) { + logger.warn('Failed to load update preferences:', error); + } + return { ...DEFAULT_UPDATE_PREFERENCES }; +} + +function saveUpdatePreferences(preferences) { + try { + const current = getUpdatePreferences(); + const updated = { ...current, ...preferences }; + localStorage?.setItem?.(UPDATE_PREFERENCES_KEY, JSON.stringify(updated)); + logger.info('Update preferences saved:', updated); + return updated; + } catch (error) { + logger.error('Failed to save update preferences:', error); + return null; + } +} + +// Smart timing for update checks +function isQuietTime(preferences = getUpdatePreferences()) { + if (!preferences.quietHours.enabled) return false; + + const now = new Date(); + const currentTime = now.getHours() * 60 + now.getMinutes(); + + const [startHour, startMin] = preferences.quietHours.start.split(':').map(Number); + const [endHour, endMin] = preferences.quietHours.end.split(':').map(Number); + + const startTime = startHour * 60 + startMin; + const endTime = endHour * 60 + endMin; + + if (startTime <= endTime) { + return currentTime >= startTime && currentTime <= endTime; + } else { + // Quiet hours span midnight + return currentTime >= startTime || currentTime <= endTime; + } +} + +function shouldAutoCheck(preferences = getUpdatePreferences()) { + if (!preferences.autoCheck) return false; + if (isQuietTime(preferences)) return false; + + const lastCheck = preferences.lastAutoCheck ? new Date(preferences.lastAutoCheck) : null; + if (!lastCheck) return true; + + const now = new Date(); + const timeDiff = now.getTime() - lastCheck.getTime(); + + switch (preferences.checkFrequency) { + case 'daily': + return timeDiff >= 24 * 60 * 60 * 1000; + case 'weekly': + return timeDiff >= 7 * 24 * 60 * 60 * 1000; + case 'monthly': + return timeDiff >= 30 * 24 * 60 * 60 * 1000; + default: + return false; + } +} + +// Platform-specific update service with comprehensive error handling +class PlatformUpdateService { + constructor() { + this.platform = process.platform; + this.currentVersion = getSafeCurrentVersion(); + this.githubRepo = 'badboysm890/ClaraVerse'; + this.isChecking = false; // Prevent concurrent checks + this.downloadProgress = null; // Track download progress + this.backgroundDownload = null; // Background download state + this.notificationCallbacks = new Set(); // UI notification callbacks + } + + /** + * Register callback for update notifications + */ + onNotification(callback) { + this.notificationCallbacks.add(callback); + return () => this.notificationCallbacks.delete(callback); + } + + /** + * Send notification to all registered callbacks + */ + notify(type, data) { + for (const callback of this.notificationCallbacks) { + try { + callback(type, data); + } catch (error) { + logger.error('Error in notification callback:', error); + } + } + } + + /** + * Check if OTA updates are supported for the current platform + */ + isOTASupported() { + // Only Mac supports OTA updates because it's signed + return this.platform === 'darwin'; + } + + /** + * Enhanced release notes processing + */ + sanitizeReleaseNotes(notes) { + return processReleaseNotes(notes); + } + + /** + * Safe GitHub releases check with comprehensive validation + */ + async checkGitHubReleases() { + // Prevent concurrent update checks + if (this.isChecking) { + throw new UpdateError('Update check already in progress', 'CONCURRENT_CHECK', false); + } + + this.isChecking = true; + + try { + const url = `https://api.github.com/repos/${this.githubRepo}/releases/latest`; + logger.info(`Checking for updates at: ${url}`); + + const response = await makeRobustRequest(url); + const release = await response.json(); + + // Validate release data structure + validateReleaseData(release); + + const latestVersion = validateVersion(release.tag_name); + const hasUpdate = this.isVersionNewer(latestVersion, this.currentVersion); + + const updateInfo = { + hasUpdate, + latestVersion, + currentVersion: this.currentVersion, + releaseUrl: release.html_url, + downloadUrl: this.getDownloadUrlForPlatform(release.assets), + releaseNotes: this.sanitizeReleaseNotes(release.body), + publishedAt: release.published_at + }; + + logger.info('Update check completed successfully:', { + hasUpdate, + currentVersion: this.currentVersion, + latestVersion + }); + + return updateInfo; + } catch (error) { + logger.error('Error checking GitHub releases:', error); + + if (error instanceof UpdateError) { + throw error; + } + + throw new UpdateError( + `Failed to check for updates: ${error.message}`, + 'UNKNOWN_ERROR', + true + ); + } finally { + this.isChecking = false; + } + } + +// Enhanced release notes processing with markdown support +function processReleaseNotes(notes) { + if (!notes || typeof notes !== 'string') { + return { + plain: 'No release notes available.', + formatted: 'No release notes available.', + categories: {} + }; + } + + // Enhanced sanitization while preserving markdown structure + let sanitized = notes + .replace(/]*>.*?<\/script>/gis, '') // Remove scripts + .replace(/]*>.*?<\/iframe>/gis, '') // Remove iframes + .replace(/javascript:/gi, '') // Remove javascript: URLs + .trim(); + + // Parse and categorize content + const categories = { + 'New Features': [], + 'Improvements': [], + 'Bug Fixes': [], + 'Breaking Changes': [], + 'Other': [] + }; + + // Simple markdown-aware categorization + const lines = sanitized.split('\n'); + let currentCategory = 'Other'; + + for (const line of lines) { + const trimmed = line.trim(); + if (!trimmed) continue; + + // Detect category headers + if (trimmed.match(/^#+\s*(new features?|features?)/i)) { + currentCategory = 'New Features'; + continue; + } else if (trimmed.match(/^#+\s*(improvements?|enhancements?)/i)) { + currentCategory = 'Improvements'; + continue; + } else if (trimmed.match(/^#+\s*(bug fixes?|fixes?|bugfixes?)/i)) { + currentCategory = 'Bug Fixes'; + continue; + } else if (trimmed.match(/^#+\s*(breaking changes?|breaking)/i)) { + currentCategory = 'Breaking Changes'; + continue; + } + + // Add content to current category + if (trimmed.startsWith('- ') || trimmed.startsWith('* ') || trimmed.startsWith('+ ')) { + categories[currentCategory].push(trimmed.substring(2).trim()); + } else if (trimmed && !trimmed.startsWith('#')) { + categories[currentCategory].push(trimmed); + } + } + + // Create formatted version + let formatted = ''; + for (const [category, items] of Object.entries(categories)) { + if (items.length > 0) { + formatted += `**${category}:**\n`; + for (const item of items) { + formatted += `• ${item}\n`; + } + formatted += '\n'; + } + } + + // Limit length if needed + const plainText = sanitized.replace(/[#*_`]/g, ''); + if (plainText.length > UPDATE_CONSTANTS.MAX_RELEASE_NOTES_LENGTH) { + sanitized = plainText.substring(0, UPDATE_CONSTANTS.MAX_RELEASE_NOTES_LENGTH) + '...'; + formatted = formatted.substring(0, UPDATE_CONSTANTS.MAX_RELEASE_NOTES_LENGTH) + '...'; + } + + return { + plain: plainText || 'No release notes available.', + formatted: formatted || plainText || 'No release notes available.', + categories, + hasBreakingChanges: categories['Breaking Changes'].length > 0 + }; +} + +// Update preferences management +function getUpdatePreferences() { + try { + const stored = localStorage?.getItem?.(UPDATE_PREFERENCES_KEY); + if (stored) { + return { ...DEFAULT_UPDATE_PREFERENCES, ...JSON.parse(stored) }; + } + } catch (error) { + logger.warn('Failed to load update preferences:', error); + } + return { ...DEFAULT_UPDATE_PREFERENCES }; +} + +function saveUpdatePreferences(preferences) { + try { + const current = getUpdatePreferences(); + const updated = { ...current, ...preferences }; + localStorage?.setItem?.(UPDATE_PREFERENCES_KEY, JSON.stringify(updated)); + logger.info('Update preferences saved:', updated); + return updated; + } catch (error) { + logger.error('Failed to save update preferences:', error); + return null; + } +} + +// Smart timing for update checks +function isQuietTime(preferences = getUpdatePreferences()) { + if (!preferences.quietHours.enabled) return false; + + const now = new Date(); + const currentTime = now.getHours() * 60 + now.getMinutes(); + + const [startHour, startMin] = preferences.quietHours.start.split(':').map(Number); + const [endHour, endMin] = preferences.quietHours.end.split(':').map(Number); + + const startTime = startHour * 60 + startMin; + const endTime = endHour * 60 + endMin; + + if (startTime <= endTime) { + return currentTime >= startTime && currentTime <= endTime; + } else { + // Quiet hours span midnight + return currentTime >= startTime || currentTime <= endTime; + } +} + +function shouldAutoCheck(preferences = getUpdatePreferences()) { + if (!preferences.autoCheck) return false; + if (isQuietTime(preferences)) return false; + + const lastCheck = preferences.lastAutoCheck ? new Date(preferences.lastAutoCheck) : null; + if (!lastCheck) return true; + + const now = new Date(); + const timeDiff = now.getTime() - lastCheck.getTime(); + + switch (preferences.checkFrequency) { + case 'daily': + return timeDiff >= 24 * 60 * 60 * 1000; + case 'weekly': + return timeDiff >= 7 * 24 * 60 * 60 * 1000; + case 'monthly': + return timeDiff >= 30 * 24 * 60 * 60 * 1000; + default: + return false; + } +} + + /** + * Get the appropriate download URL for the current platform with validation + */ + getDownloadUrlForPlatform(assets) { + if (!Array.isArray(assets)) { + logger.warn('Invalid assets array, using fallback URL'); + return `https://github.com/${this.githubRepo}/releases/latest`; + } + + const platformExtensions = { + darwin: ['.dmg', '-arm64.dmg', '-mac.dmg'], + win32: ['.exe', '-win.exe', '-windows.exe'], + linux: ['.AppImage', '.deb', '-linux.AppImage'] + }; + + const extensions = platformExtensions[this.platform] || []; + + // Try to find platform-specific download + for (const ext of extensions) { + const asset = assets.find(asset => + asset && + asset.name && + typeof asset.name === 'string' && + asset.name.toLowerCase().endsWith(ext.toLowerCase()) && + asset.browser_download_url + ); + + if (asset) { + logger.info(`Found platform-specific download: ${asset.name}`); + return asset.browser_download_url; + } + } + + // Fallback to releases page + logger.info('No platform-specific download found, using releases page'); + return `https://github.com/${this.githubRepo}/releases/latest`; + } + + /** + * Robust version comparison with detailed logging + */ + isVersionNewer(newVersion, currentVersion) { + try { + const parseVersion = (version) => { + return version.split('.').map(num => { + const parsed = parseInt(num, 10); + return isNaN(parsed) ? 0 : parsed; + }); + }; + + const newParts = parseVersion(newVersion); + const currentParts = parseVersion(currentVersion); + const maxLength = Math.max(newParts.length, currentParts.length); + + for (let i = 0; i < maxLength; i++) { + const newPart = newParts[i] || 0; + const currentPart = currentParts[i] || 0; + + if (newPart > currentPart) { + logger.info(`Version ${newVersion} is newer than ${currentVersion}`); + return true; + } + if (newPart < currentPart) { + logger.info(`Version ${newVersion} is older than ${currentVersion}`); + return false; + } + } + + logger.info(`Version ${newVersion} is same as ${currentVersion}`); + return false; + } catch (error) { + logger.error('Error comparing versions:', error); + return false; // Safe fallback + } + } + + /** + * Show platform-specific update dialog with error handling + */ + async showUpdateDialog(updateInfo) { + try { + const { hasUpdate, latestVersion, downloadUrl, releaseNotes } = updateInfo; + + if (!hasUpdate) { + return await dialog.showMessageBox({ + type: 'info', + title: 'No Updates Available', + message: 'You are running the latest version of Clara.', + detail: `Current version: ${this.currentVersion}`, + buttons: ['OK'], + defaultId: 0 + }); + } + + const truncatedNotes = releaseNotes && releaseNotes.length > 200 + ? releaseNotes.substring(0, 200) + '...' + : releaseNotes; + + if (this.isOTASupported()) { + // Mac: Show OTA update dialog + return await dialog.showMessageBox({ + type: 'info', + title: 'Update Available', + message: `Clara ${latestVersion} is available`, + detail: `You have ${this.currentVersion}. Would you like to download and install the update automatically?\n\n${truncatedNotes ? `What's new:\n${truncatedNotes}` : 'Click "View Release Notes" for details.'}`, + buttons: ['Download & Install', 'View Release Notes', 'Later'], + defaultId: 0, + cancelId: 2 + }).then(({ response }) => { + try { + if (response === 0) { + // Start OTA update + autoUpdater.downloadUpdate(); + return { action: 'download' }; + } else if (response === 1) { + // Open release notes + shell.openExternal(updateInfo.releaseUrl); + return { action: 'view_notes' }; + } + return { action: 'later' }; + } catch (error) { + logger.error('Error handling dialog response:', error); + return { action: 'error', error: error.message }; + } + }); + } else { + // Windows/Linux: Show manual update dialog + const platformName = this.platform === 'win32' ? 'Windows' : 'Linux'; + + return await dialog.showMessageBox({ + type: 'info', + title: 'Update Available', + message: `Clara ${latestVersion} is available`, + detail: `You have ${this.currentVersion}. A new version is available for download.\n\nOn ${platformName}, updates need to be installed manually for security reasons.\n\n${truncatedNotes ? `What's new:\n${truncatedNotes}` : 'Click "View Release Notes" for details.'}`, + buttons: ['Download Now', 'View Release Notes', 'Later'], + defaultId: 0, + cancelId: 2 + }).then(({ response }) => { + try { + if (response === 0) { + // Open download page + shell.openExternal(downloadUrl); + return { action: 'download' }; + } else if (response === 1) { + // Open release notes + shell.openExternal(updateInfo.releaseUrl); + return { action: 'view_notes' }; + } + return { action: 'later' }; + } catch (error) { + logger.error('Error handling dialog response:', error); + return { action: 'error', error: error.message }; + } + }); + } + } catch (error) { + logger.error('Error showing update dialog:', error); + + // Show fallback error dialog + try { + await dialog.showErrorBox( + 'Update Dialog Error', + `Failed to show update information: ${error.message}` + ); + } catch (dialogError) { + logger.error('Failed to show error dialog:', dialogError); + } + + return { action: 'error', error: error.message }; + } + } +} + +// Global instance with error protection +let platformUpdateService; +try { + platformUpdateService = new PlatformUpdateService(); +} catch (error) { + logger.error('Failed to initialize update service:', error); + platformUpdateService = null; +} + +// Enhanced auto-updater setup with comprehensive error handling +function setupAutoUpdater(mainWindow) { + if (!platformUpdateService || !platformUpdateService.isOTASupported()) { + logger.info('OTA updates not supported on this platform'); + return; + } + + try { + // Check for updates when the app starts (Mac only) + autoUpdater.checkForUpdatesAndNotify().catch(error => { + logger.error('Initial update check failed:', error); + }); + + // Check for updates every hour (Mac only) with error handling + setInterval(() => { + autoUpdater.checkForUpdatesAndNotify().catch(error => { + logger.error('Periodic update check failed:', error); + }); + }, 60 * 60 * 1000); + + // Update available + autoUpdater.on('update-available', (info) => { + try { + dialog.showMessageBox({ + type: 'info', + title: 'Update Available', + message: `A new version (${info.version}) is available. Would you like to download and update now?`, + buttons: ['Update', 'Later'], + defaultId: 0, + cancelId: 1 + }).then(({ response }) => { + if (response === 0) { + autoUpdater.downloadUpdate().catch(error => { + logger.error('Download update failed:', error); + dialog.showErrorBox('Download Failed', `Failed to download update: ${error.message}`); + }); + } + }).catch(error => { + logger.error('Error showing update available dialog:', error); + }); + } catch (error) { + logger.error('Error in update-available handler:', error); + } + }); + + // Update downloaded + autoUpdater.on('update-downloaded', () => { + try { + dialog.showMessageBox({ + type: 'info', + title: 'Update Ready', + message: 'The update has been downloaded. The application will restart to apply the update.', + buttons: ['Restart Now'], + defaultId: 0 + }).then(() => { + try { + autoUpdater.quitAndInstall(); + } catch (error) { + logger.error('Error during quit and install:', error); + dialog.showErrorBox('Installation Failed', `Failed to install update: ${error.message}`); + } + }).catch(error => { + logger.error('Error showing update downloaded dialog:', error); + }); + } catch (error) { + logger.error('Error in update-downloaded handler:', error); + } + }); + + // Enhanced error handling + autoUpdater.on('error', (err) => { + logger.error('Auto-updater error:', err); + + try { + // Fallback to GitHub-based updates on error + dialog.showErrorBox('Update Error', + `Automatic update failed: ${err.message}\n\nYou can manually download the latest version from GitHub.` + ); + } catch (dialogError) { + logger.error('Failed to show error dialog:', dialogError); + } + }); + + // Progress updates with error handling + autoUpdater.on('download-progress', (progressObj) => { + try { + if (mainWindow && mainWindow.webContents) { + mainWindow.webContents.send('update-progress', progressObj); + } + } catch (error) { + logger.error('Error sending progress update:', error); + } + }); + + // No update available + autoUpdater.on('update-not-available', (info, isManualCheck) => { + try { + if (isManualCheck) { + dialog.showMessageBox({ + type: 'info', + title: 'No Updates Available', + message: 'You are running the latest version of Clara.', + buttons: ['OK'], + defaultId: 0 + }).catch(error => { + logger.error('Error showing no update dialog:', error); + }); + } + } catch (error) { + logger.error('Error in update-not-available handler:', error); + } + }); + + logger.info('Auto-updater setup completed successfully'); + } catch (error) { + logger.error('Failed to setup auto-updater:', error); + } +} + +// Universal update check with comprehensive error handling +async function checkForUpdates() { + if (!platformUpdateService) { + const error = 'Update service not available'; + logger.error(error); + try { + dialog.showErrorBox('Update Service Error', error); + } catch (dialogError) { + logger.error('Failed to show error dialog:', dialogError); + } + return; + } + + try { + if (platformUpdateService.isOTASupported()) { + // Mac: Use electron-updater first, fallback to GitHub + try { + return await autoUpdater.checkForUpdates(); + } catch (error) { + logger.warn('OTA update check failed, falling back to GitHub:', error); + const updateInfo = await platformUpdateService.checkGitHubReleases(); + return await platformUpdateService.showUpdateDialog(updateInfo); + } + } else { + // Windows/Linux: Use GitHub releases + const updateInfo = await platformUpdateService.checkGitHubReleases(); + return await platformUpdateService.showUpdateDialog(updateInfo); + } + } catch (error) { + logger.error('Error checking for updates:', error); + + let userMessage = 'Could not check for updates. Please check your internet connection and try again.'; + + if (error instanceof UpdateError) { + switch (error.type) { + case 'RATE_LIMIT': + userMessage = error.message; + break; + case 'NO_FETCH': + userMessage = 'Network functionality is not available. Please restart the application.'; + break; + case 'CONCURRENT_CHECK': + userMessage = 'Update check is already in progress. Please wait.'; + break; + default: + userMessage = `Update check failed: ${error.message}`; + } + } + + try { + dialog.showErrorBox('Update Check Failed', userMessage); + } catch (dialogError) { + logger.error('Failed to show error dialog:', dialogError); + } + } +} + +// Safe update info retrieval for UI +async function getUpdateInfo() { + if (!platformUpdateService) { + return { + hasUpdate: false, + error: 'Update service not available', + platform: process.platform, + isOTASupported: false, + currentVersion: getSafeCurrentVersion() + }; + } + + try { + const updateInfo = await platformUpdateService.checkGitHubReleases(); + return { + ...updateInfo, + platform: platformUpdateService.platform, + isOTASupported: platformUpdateService.isOTASupported() + }; + } catch (error) { + logger.error('Error getting update info:', error); + + let errorMessage = 'Failed to check for updates'; + + if (error instanceof UpdateError) { + errorMessage = error.message; + } else { + errorMessage = error.message || 'Unknown error occurred'; + } + + return { + hasUpdate: false, + error: errorMessage, + platform: platformUpdateService.platform, + isOTASupported: platformUpdateService.isOTASupported(), + currentVersion: platformUpdateService.currentVersion + }; + } +} + +// Llama.cpp Binary Update Service +class LlamacppUpdateService { + constructor() { + this.platform = process.platform; + this.arch = process.arch; + this.githubRepo = 'ggerganov/llama.cpp'; + this.isUpdating = false; + this.binariesPath = this.getBinariesPath(); + } + + getBinariesPath() { + const path = require('path'); + const { app } = require('electron'); + + const isDevelopment = process.env.NODE_ENV === 'development'; + + if (isDevelopment) { + return path.join(__dirname, 'llamacpp-binaries'); + } else { + // Production paths + const possiblePaths = [ + path.join(process.resourcesPath, 'electron', 'llamacpp-binaries'), + path.join(app.getAppPath(), 'electron', 'llamacpp-binaries'), + path.join(__dirname, 'llamacpp-binaries') + ]; + + for (const possiblePath of possiblePaths) { + if (require('fs').existsSync(possiblePath)) { + return possiblePath; + } + } + + return path.join(app.getPath('userData'), 'llamacpp-binaries'); + } + } + + getCurrentVersion() { + const path = require('path'); + const fs = require('fs'); + + try { + const versionFile = path.join(this.binariesPath, 'version.txt'); + if (fs.existsSync(versionFile)) { + return fs.readFileSync(versionFile, 'utf8').trim(); + } + } catch (error) { + logger.warn('Could not read current llama.cpp version:', error); + } + + return 'Unknown'; + } + + getPlatformInfo() { + switch (this.platform) { + case 'darwin': + return { + platform: 'darwin', + arch: this.arch === 'arm64' ? 'arm64' : 'x64', + platformDir: this.arch === 'arm64' ? 'darwin-arm64' : 'darwin-x64', + assetPattern: this.arch === 'arm64' ? 'llama-.*-bin-macos-arm64.zip' : 'llama-.*-bin-macos-x64.zip' + }; + case 'linux': + return { + platform: 'linux', + arch: 'x64', + platformDir: 'linux-x64', + assetPattern: 'llama-.*-bin-ubuntu-vulkan-x64.zip' + }; + case 'win32': + return { + platform: 'win32', + arch: 'x64', + platformDir: 'win32-x64', + assetPattern: 'llama-.*-bin-win-.*-x64.zip' + }; + default: + throw new Error(`Unsupported platform: ${this.platform}-${this.arch}`); + } + } + + async checkForUpdates() { + if (!fetch) { + throw new UpdateError('Network functionality not available', 'NO_FETCH', false); + } + + try { + const response = await makeRobustRequest(`https://api.github.com/repos/${this.githubRepo}/releases/latest`); + const release = await response.json(); + + validateReleaseData(release); + + const currentVersion = this.getCurrentVersion(); + const latestVersion = release.tag_name; + const hasUpdate = currentVersion === 'Unknown' || currentVersion !== latestVersion; + + const platformInfo = this.getPlatformInfo(); + const matchingAsset = release.assets.find(asset => + new RegExp(platformInfo.assetPattern, 'i').test(asset.name) + ); + + return { + hasUpdate, + currentVersion, + latestVersion, + platform: this.platform, + downloadSize: matchingAsset ? this.formatFileSize(matchingAsset.size) : 'Unknown size', + releaseUrl: release.html_url, + downloadUrl: matchingAsset?.browser_download_url, + publishedAt: release.published_at, + error: matchingAsset ? null : `No compatible binary found for ${this.platform}-${this.arch}` + }; + } catch (error) { + logger.error('Error checking for llama.cpp updates:', error); + throw error; + } + } + + async updateBinaries() { + if (this.isUpdating) { + throw new Error('Binary update already in progress'); + } + + this.isUpdating = true; + + try { + const updateInfo = await this.checkForUpdates(); + + if (!updateInfo.hasUpdate) { + return { success: true, message: 'Binaries are already up to date' }; + } + + if (!updateInfo.downloadUrl) { + throw new Error(updateInfo.error || 'No download URL available'); + } + + const result = await this.downloadAndInstallBinaries(updateInfo); + return result; + } catch (error) { + logger.error('Error updating llama.cpp binaries:', error); + return { + success: false, + error: error.message || 'Failed to update binaries' + }; + } finally { + this.isUpdating = false; + } + } + + async downloadAndInstallBinaries(updateInfo) { + const path = require('path'); + const fs = require('fs').promises; + const fsSync = require('fs'); + + // Robust require for adm-zip with fallback + let AdmZip; + try { + AdmZip = require('adm-zip'); + } catch (requireError) { + // Try alternative require paths + try { + const mainPath = require.resolve('adm-zip', { paths: [process.cwd(), __dirname, path.join(__dirname, '..', 'node_modules')] }); + AdmZip = require(mainPath); + } catch (fallbackError) { + // If adm-zip is not available, try using built-in modules for Windows + if (this.platform === 'win32') { + return await this.downloadAndInstallBinariesWindows(updateInfo); + } + throw new Error(`Cannot find 'adm-zip' module. Please install it: npm install adm-zip. Error: ${requireError.message}`); + } + } + + try { + logger.info(`Downloading llama.cpp binaries from: ${updateInfo.downloadUrl}`); + + // Download the main zip file + const response = await makeRobustRequest(updateInfo.downloadUrl); + const buffer = Buffer.from(await response.arrayBuffer()); + + // Create temporary file + const tempDir = path.join(require('os').tmpdir(), 'clara-llamacpp-update'); + await fs.mkdir(tempDir, { recursive: true }); + const tempZipPath = path.join(tempDir, 'llamacpp-binaries.zip'); + + await fs.writeFile(tempZipPath, buffer); + logger.info(`Downloaded binaries to: ${tempZipPath}`); + + // Extract main zip file + const zip = new AdmZip(tempZipPath); + const extractDir = path.join(tempDir, 'extracted'); + zip.extractAllTo(extractDir, true); + + // For Windows, also download CUDA runtime binaries if available + let cudaRuntimeFiles = []; + if (this.platform === 'win32') { + try { + cudaRuntimeFiles = await this.downloadCudaRuntimeBinaries(updateInfo, tempDir); + } catch (cudaError) { + logger.warn('Failed to download CUDA runtime binaries:', cudaError.message); + logger.info('Continuing with main binary update. CUDA acceleration may not work optimally.'); + } + } + + // ===== STOP ALL SERVICES BEFORE UPDATING ===== + logger.info('🛑 Stopping all services before binary update...'); + const servicesWereStopped = await this.stopAllServicesForUpdate(); + + try { + // Find the platform directory to update + const platformInfo = this.getPlatformInfo(); + const targetPlatformDir = path.join(this.binariesPath, platformInfo.platformDir); + + // Backup current official llama.cpp files only (preserving Clara's custom files) + const backupDir = path.join(this.binariesPath, `${platformInfo.platformDir}-backup-${Date.now()}`); + if (fsSync.existsSync(targetPlatformDir)) { + await fs.mkdir(backupDir, { recursive: true }); + + // Only backup official llama.cpp files, leave Clara's custom files alone + const files = await fs.readdir(targetPlatformDir); + for (const file of files) { + // Only backup official llama.cpp files (not Clara's custom binaries) + if (this.isOfficialLlamacppFile(file)) { + const sourcePath = path.join(targetPlatformDir, file); + const backupPath = path.join(backupDir, file); + + if (fsSync.existsSync(sourcePath)) { + await fs.copyFile(sourcePath, backupPath); + logger.info(`Backed up ${file} to backup directory`); + } + } + } + } + + // Ensure target directory exists + await fs.mkdir(targetPlatformDir, { recursive: true }); + + // Find and copy official llama.cpp files from extracted files + const extractedFiles = await this.findOfficialFilesInExtracted(extractDir); + + if (extractedFiles.length === 0) { + throw new Error('No official llama.cpp files found in the downloaded archive'); + } + + for (const [sourceFile, targetName] of extractedFiles) { + const targetPath = path.join(targetPlatformDir, targetName); + await fs.copyFile(sourceFile, targetPath); + + // Make executable for binary files on Unix systems + if (this.platform !== 'win32' && this.isExecutableFile(targetName)) { + await fs.chmod(targetPath, 0o755); + } + + logger.info(`Installed official file: ${targetName}`); + } + + // Install CUDA runtime files for Windows if downloaded + if (cudaRuntimeFiles.length > 0) { + for (const [sourceFile, targetName] of cudaRuntimeFiles) { + const targetPath = path.join(targetPlatformDir, targetName); + await fs.copyFile(sourceFile, targetPath); + logger.info(`Installed CUDA runtime file: ${targetName}`); + } + logger.info(`Successfully installed ${cudaRuntimeFiles.length} CUDA runtime files for GPU acceleration`); + } + + // Save version info + const versionFile = path.join(this.binariesPath, 'version.txt'); + await fs.writeFile(versionFile, updateInfo.latestVersion); + + // Validate the installation by testing the main binary + await this.validateInstallation(targetPlatformDir); + + logger.info(`Successfully updated llama.cpp binaries and libraries to version ${updateInfo.latestVersion}`); + logger.info(`Clara's custom binaries (like llama-swap) were preserved and not modified`); + if (cudaRuntimeFiles.length > 0) { + logger.info(`CUDA runtime binaries were also updated for optimal GPU acceleration`); + } + + } finally { + // ===== RESTART SERVICES AFTER UPDATE ===== + if (servicesWereStopped) { + logger.info('🔄 Restarting services after binary update...'); + await this.restartAllServicesAfterUpdate(); + } + } + + // Cleanup + await fs.rm(tempDir, { recursive: true, force: true }); + + const successMessage = `Successfully updated llama.cpp binaries and libraries to version ${updateInfo.latestVersion}. Clara's custom binaries were preserved. Services have been restarted.`; + const cudaMessage = cudaRuntimeFiles.length > 0 ? ` CUDA runtime binaries were also updated for optimal GPU acceleration.` : ''; + + return { + success: true, + message: successMessage + cudaMessage, + version: updateInfo.latestVersion + }; + + } catch (error) { + logger.error('Error during binary installation:', error); + throw new Error(`Installation failed: ${error.message}`); + } + } + + // Windows-specific binary update using built-in modules if adm-zip fails + async downloadAndInstallBinariesWindows(updateInfo) { + const path = require('path'); + const fs = require('fs').promises; + const fsSync = require('fs'); + const { spawn } = require('child_process'); + + try { + logger.info(`Downloading llama.cpp binaries for Windows from: ${updateInfo.downloadUrl}`); + + // Download the main zip file + const response = await makeRobustRequest(updateInfo.downloadUrl); + const buffer = Buffer.from(await response.arrayBuffer()); + + // Create temporary file + const tempDir = path.join(require('os').tmpdir(), 'clara-llamacpp-update'); + await fs.mkdir(tempDir, { recursive: true }); + const tempZipPath = path.join(tempDir, 'llamacpp-binaries.zip'); + + await fs.writeFile(tempZipPath, buffer); + logger.info(`Downloaded binaries to: ${tempZipPath}`); + + // Use Windows built-in extract if available, or try PowerShell + const extractDir = path.join(tempDir, 'extracted'); + await fs.mkdir(extractDir, { recursive: true }); + + // Try PowerShell extraction first + const success = await this.extractWithPowerShell(tempZipPath, extractDir); + + if (!success) { + // Fallback to manual extraction + throw new Error('Could not extract zip file. Please ensure adm-zip package is properly installed.'); + } + + // Also download CUDA runtime binaries if available (Windows fallback method) + let cudaRuntimeFiles = []; + try { + cudaRuntimeFiles = await this.downloadCudaRuntimeBinariesWindows(updateInfo, tempDir); + } catch (cudaError) { + logger.warn('Failed to download CUDA runtime binaries (Windows fallback):', cudaError.message); + logger.info('Continuing with main binary update. CUDA acceleration may not work optimally.'); + } + + // ===== STOP ALL SERVICES BEFORE UPDATING ===== + logger.info('🛑 Stopping all services before binary update (Windows fallback)...'); + const servicesWereStopped = await this.stopAllServicesForUpdate(); + + try { + // Continue with the same logic as the main function + const platformInfo = this.getPlatformInfo(); + const targetPlatformDir = path.join(this.binariesPath, platformInfo.platformDir); + + // Backup current official llama.cpp files only (preserving Clara's custom files) + const backupDir = path.join(this.binariesPath, `${platformInfo.platformDir}-backup-${Date.now()}`); + if (fsSync.existsSync(targetPlatformDir)) { + await fs.mkdir(backupDir, { recursive: true }); + + // Only backup official llama.cpp files, leave Clara's custom files alone + const files = await fs.readdir(targetPlatformDir); + for (const file of files) { + if (this.isOfficialLlamacppFile(file)) { + const sourcePath = path.join(targetPlatformDir, file); + const backupPath = path.join(backupDir, file); + + if (fsSync.existsSync(sourcePath)) { + await fs.copyFile(sourcePath, backupPath); + logger.info(`Backed up ${file} to backup directory`); + } + } + } + } + + // Ensure target directory exists + await fs.mkdir(targetPlatformDir, { recursive: true }); + + // Find and copy official llama.cpp files from extracted files + const extractedFiles = await this.findOfficialFilesInExtracted(extractDir); + + if (extractedFiles.length === 0) { + throw new Error('No official llama.cpp files found in the downloaded archive'); + } + + for (const [sourceFile, targetName] of extractedFiles) { + const targetPath = path.join(targetPlatformDir, targetName); + await fs.copyFile(sourceFile, targetPath); + logger.info(`Installed official file: ${targetName}`); + } + + // Install CUDA runtime files for Windows if downloaded (Windows fallback) + if (cudaRuntimeFiles.length > 0) { + for (const [sourceFile, targetName] of cudaRuntimeFiles) { + const targetPath = path.join(targetPlatformDir, targetName); + await fs.copyFile(sourceFile, targetPath); + logger.info(`Installed CUDA runtime file: ${targetName}`); + } + logger.info(`Successfully installed ${cudaRuntimeFiles.length} CUDA runtime files for GPU acceleration (Windows fallback)`); + } + + // Save version info + const versionFile = path.join(this.binariesPath, 'version.txt'); + await fs.writeFile(versionFile, updateInfo.latestVersion); + + // Validate the installation + await this.validateInstallation(targetPlatformDir); + + logger.info(`Successfully updated llama.cpp binaries and libraries to version ${updateInfo.latestVersion} (Windows fallback method)`); + if (cudaRuntimeFiles.length > 0) { + logger.info(`CUDA runtime binaries were also updated for optimal GPU acceleration (Windows fallback)`); + } + + } finally { + // ===== RESTART SERVICES AFTER UPDATE ===== + if (servicesWereStopped) { + logger.info('🔄 Restarting services after binary update (Windows fallback)...'); + await this.restartAllServicesAfterUpdate(); + } + } + + // Cleanup + await fs.rm(tempDir, { recursive: true, force: true }); + + const successMessage = `Successfully updated llama.cpp binaries and libraries to version ${updateInfo.latestVersion}. Clara's custom binaries were preserved. Services have been restarted.`; + const cudaMessage = cudaRuntimeFiles.length > 0 ? ` CUDA runtime binaries were also updated for optimal GPU acceleration (Windows fallback).` : ''; + + return { + success: true, + message: successMessage + cudaMessage, + version: updateInfo.latestVersion + }; + + } catch (error) { + logger.error('Error during Windows binary installation:', error); + throw new Error(`Windows installation failed: ${error.message}`); + } + } + + // Extract zip using PowerShell on Windows + async extractWithPowerShell(zipPath, extractPath) { + return new Promise((resolve) => { + const powerShellCommand = ` + try { + Add-Type -AssemblyName System.IO.Compression.FileSystem + [System.IO.Compression.ZipFile]::ExtractToDirectory('${zipPath}', '${extractPath}') + Write-Output "SUCCESS" + } catch { + Write-Output "ERROR: $($_.Exception.Message)" + exit 1 + } + `; + + const powerShell = spawn('powershell', [ + '-Command', powerShellCommand + ], { + stdio: ['ignore', 'pipe', 'pipe'] + }); + + let output = ''; + let errorOutput = ''; + + powerShell.stdout.on('data', (data) => { + output += data.toString(); + }); + + powerShell.stderr.on('data', (data) => { + errorOutput += data.toString(); + }); + + powerShell.on('close', (code) => { + if (code === 0 && output.includes('SUCCESS')) { + logger.info('Successfully extracted using PowerShell'); + resolve(true); + } else { + logger.warn('PowerShell extraction failed:', errorOutput || output); + resolve(false); + } + }); + + powerShell.on('error', (error) => { + logger.warn('PowerShell extraction error:', error.message); + resolve(false); + }); + }); + } + + // Stop all services that might be using llama.cpp binaries + async stopAllServicesForUpdate() { + const stoppedServices = []; + + try { + // Get IPC access to communicate with main process + const { ipcMain } = require('electron'); + + // Try to stop llama swap service + try { + logger.info('Stopping LlamaSwap service...'); + const llamaSwapService = require('./llamaSwapService.cjs'); + if (llamaSwapService && typeof llamaSwapService.stop === 'function') { + await llamaSwapService.stop(); + stoppedServices.push('llamaSwap'); + logger.info('✅ LlamaSwap service stopped'); + } + } catch (error) { + logger.warn('Could not stop LlamaSwap service:', error.message); + } + + // Try to stop any Python backend services + try { + logger.info('Stopping Python backend services...'); + // Send signal to main process to stop Python services + const { BrowserWindow } = require('electron'); + const mainWindow = BrowserWindow.getAllWindows()[0]; + if (mainWindow && mainWindow.webContents) { + mainWindow.webContents.send('stop-python-services'); + stoppedServices.push('python'); + logger.info('✅ Python backend services stop signal sent'); + } + } catch (error) { + logger.warn('Could not stop Python services:', error.message); + } + + // Try to stop any model server processes + try { + logger.info('Stopping model server processes...'); + + // Kill any llama-server processes that might be running + if (this.platform === 'win32') { + const { spawn } = require('child_process'); + + // Try to gracefully stop llama-server processes + const taskkill = spawn('taskkill', ['/F', '/IM', 'llama-server.exe'], { + stdio: 'ignore', + windowsHide: true + }); + + await new Promise((resolve) => { + taskkill.on('close', () => resolve()); + setTimeout(resolve, 3000); // Don't wait forever + }); + + stoppedServices.push('llama-server'); + logger.info('✅ Windows llama-server processes stopped'); + } else { + // Unix-like systems + const { spawn } = require('child_process'); + + const pkill = spawn('pkill', ['-f', 'llama-server'], { + stdio: 'ignore' + }); + + await new Promise((resolve) => { + pkill.on('close', () => resolve()); + setTimeout(resolve, 3000); // Don't wait forever + }); + + stoppedServices.push('llama-server'); + logger.info('✅ Unix llama-server processes stopped'); + } + } catch (error) { + logger.warn('Could not stop model server processes:', error.message); + } + + // Wait a bit for processes to fully release file handles + logger.info('⏳ Waiting for processes to release file handles...'); + await new Promise(resolve => setTimeout(resolve, 2000)); + + logger.info(`🛑 Stopped services: ${stoppedServices.join(', ')}`); + return stoppedServices.length > 0; + + } catch (error) { + logger.error('Error stopping services:', error); + return false; + } + } + + // Restart services after update + async restartAllServicesAfterUpdate() { + try { + // Wait a moment for the system to settle + await new Promise(resolve => setTimeout(resolve, 1000)); + + // Repair binary paths after update + try { + logger.info('🔧 Repairing binary paths after update...'); + const LlamaSwapService = require('./llamaSwapService.cjs'); + await LlamaSwapService.repairBinariesAfterUpdate(); + logger.info('✅ Binary paths repaired after update'); + } catch (repairError) { + logger.warn('Binary repair failed, but continuing with restart:', repairError.message); + } + + // Restart LlamaSwap service + try { + logger.info('Restarting LlamaSwap service...'); + const llamaSwapService = require('./llamaSwapService.cjs'); + if (llamaSwapService && typeof llamaSwapService.restart === 'function') { + await llamaSwapService.restart(); + logger.info('✅ LlamaSwap service restarted'); + } + } catch (error) { + logger.warn('Could not restart LlamaSwap service:', error.message); + } + + // Send signal to restart Python services + try { + logger.info('Restarting Python backend services...'); + const { BrowserWindow } = require('electron'); + const mainWindow = BrowserWindow.getAllWindows()[0]; + if (mainWindow && mainWindow.webContents) { + mainWindow.webContents.send('restart-python-services'); + logger.info('✅ Python backend services restart signal sent'); + } + } catch (error) { + logger.warn('Could not restart Python services:', error.message); + } + + // The services will automatically start when needed, so we don't need to explicitly start model servers + logger.info('🔄 Service restart completed'); + + } catch (error) { + logger.error('Error restarting services:', error); + } + } + + // Check if a file is an official llama.cpp file (binaries, libraries, headers, etc.) + // Now includes ALL files except Clara's custom binaries + isOfficialLlamacppFile(fileName) { + // Exclude Clara's custom binaries - these are the ONLY files we don't want to touch + const claraCustomBinaries = [ + 'llama-swap', + 'llama-swap.exe', + 'llama-swap-darwin', + 'llama-swap-darwin-arm64', + 'llama-swap-linux', + 'llama-swap-win32-x64.exe' + ]; + + // Don't touch Clara's custom binaries (llamaswap files) + if (claraCustomBinaries.some(custom => fileName.includes(custom))) { + return false; + } + + // Everything else from the llama.cpp zip is considered official and should be updated + return true; + } + + // Check if a file should be made executable + isExecutableFile(fileName) { + // On Windows, executables have .exe extension + if (this.platform === 'win32') { + return fileName.endsWith('.exe'); + } + + // On Unix systems (macOS/Linux), check for common binary patterns + // Look for files without extensions or with common binary names + const hasNoExtension = !fileName.includes('.') || + fileName.endsWith('.bin') || + fileName.endsWith('.out'); + + // Common patterns for llama.cpp binaries + const isBinaryPattern = fileName.startsWith('llama-') || + fileName.startsWith('ggml-') || + fileName.includes('server') || + fileName.includes('cli') || + fileName.includes('quantize') || + fileName.includes('perplexity') || + fileName.includes('bench') || + fileName.includes('eval') || + fileName.includes('embedding') || + fileName.includes('convert') || + fileName.includes('finetune') || + fileName.includes('export'); + + // Make executable if it looks like a binary and doesn't have a typical non-executable extension + const nonExecutableExtensions = ['.so', '.dylib', '.dll', '.h', '.hpp', '.c', '.cpp', '.txt', '.md', '.json', '.xml', '.metal', '.py', '.sh']; + const hasNonExecutableExtension = nonExecutableExtensions.some(ext => fileName.toLowerCase().endsWith(ext)); + + return (hasNoExtension && isBinaryPattern) && !hasNonExecutableExtension; + } + + async findOfficialFilesInExtracted(extractDir) { + const path = require('path'); + const fs = require('fs').promises; + + const filesMap = []; + + // Required core files that must be present for a valid update + const requiredFiles = this.platform === 'win32' + ? ['llama-server.exe'] + : ['llama-server']; + + // Helper function to search directory recursively and include ALL files except llamaswap + async function searchDirectory(dir) { + const entries = await fs.readdir(dir, { withFileTypes: true }); + + for (const entry of entries) { + const fullPath = path.join(dir, entry.name); + + if (entry.isDirectory()) { + await searchDirectory(fullPath); + } else if (entry.isFile()) { + // Skip Clara's custom llamaswap binaries + const claraCustomBinaries = [ + 'llama-swap', + 'llama-swap.exe', + 'llama-swap-darwin', + 'llama-swap-darwin-arm64', + 'llama-swap-linux', + 'llama-swap-win32-x64.exe' + ]; + + // Don't include llamaswap files + if (claraCustomBinaries.some(custom => entry.name.includes(custom))) { + logger.info(`Skipping Clara custom binary: ${entry.name}`); + continue; + } + + // Include ALL other files from the llama.cpp archive + filesMap.push([fullPath, entry.name]); + logger.info(`Found file for update: ${entry.name}`); + } + } + } + + await searchDirectory(extractDir); + + // Validate that we have the essential files for a complete update + const foundFiles = filesMap.map(([, targetName]) => targetName); + const missingRequired = requiredFiles.filter(required => !foundFiles.includes(required)); + + if (missingRequired.length > 0) { + logger.warn(`Missing required files for complete update: ${missingRequired.join(', ')}`); + logger.warn(`Found files: ${foundFiles.slice(0, 20).join(', ')}${foundFiles.length > 20 ? '...' : ''}`); + logger.warn('This might cause compatibility issues. Skipping update.'); + throw new Error(`Incomplete update package: missing required files ${missingRequired.join(', ')}`); + } + + logger.info(`Found ${filesMap.length} official files for update: ${foundFiles.slice(0, 10).join(', ')}${foundFiles.length > 10 ? '...' : ''}`); + + return filesMap; + } + + formatFileSize(bytes) { + if (bytes === 0) return '0 Bytes'; + const k = 1024; + const sizes = ['Bytes', 'KB', 'MB', 'GB']; + const i = Math.floor(Math.log(bytes) / Math.log(k)); + return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i]; + } + + async validateInstallation(targetPlatformDir) { + const path = require('path'); + const fs = require('fs').promises; + const fsSync = require('fs'); + + try { + // Check that the main binary exists and is executable + const mainBinaryPath = path.join(targetPlatformDir, this.platform === 'win32' ? 'llama-server.exe' : 'llama-server'); + + if (!fsSync.existsSync(mainBinaryPath)) { + throw new Error(`Main binary not found: ${mainBinaryPath}`); + } + + // Check that required libraries exist + const requiredLibs = this.platform === 'darwin' + ? ['libllama.dylib'] + : this.platform === 'win32' + ? ['llama.dll'] + : ['libllama.so']; + + for (const lib of requiredLibs) { + const libPath = path.join(targetPlatformDir, lib); + if (!fsSync.existsSync(libPath)) { + throw new Error(`Required library not found: ${libPath}`); + } + } + + logger.info('Installation validation completed successfully'); + } catch (error) { + logger.error('Installation validation failed:', error); + throw new Error(`Installation validation failed: ${error.message}`); + } + } + + // Download CUDA runtime binaries for Windows + async downloadCudaRuntimeBinaries(updateInfo, tempDir) { + const path = require('path'); + const fs = require('fs').promises; + + try { + // Get the latest release info to find CUDA runtime assets + const response = await makeRobustRequest(`https://api.github.com/repos/${this.githubRepo}/releases/latest`); + const release = await response.json(); + + // Look for CUDA runtime assets (e.g., cudart-llama-bin-win-cuda-12.4-x64.zip) + const cudaAssets = release.assets.filter(asset => + asset.name.match(/cudart.*llama.*bin.*win.*cuda.*x64\.zip/i) + ); + + if (cudaAssets.length === 0) { + logger.info('No CUDA runtime binaries found in release assets'); + return []; + } + + // Sort by name to get the latest CUDA version (usually the highest version number) + cudaAssets.sort((a, b) => b.name.localeCompare(a.name)); + const cudaAsset = cudaAssets[0]; + + logger.info(`Found CUDA runtime asset: ${cudaAsset.name}`); + logger.info(`Downloading CUDA runtime binaries from: ${cudaAsset.browser_download_url}`); + + // Download the CUDA runtime zip + const cudaResponse = await makeRobustRequest(cudaAsset.browser_download_url); + const cudaBuffer = Buffer.from(await cudaResponse.arrayBuffer()); + + // Save CUDA runtime zip + const cudaTempZipPath = path.join(tempDir, 'cuda-runtime.zip'); + await fs.writeFile(cudaTempZipPath, cudaBuffer); + logger.info(`Downloaded CUDA runtime to: ${cudaTempZipPath}`); + + // Extract CUDA runtime zip + let AdmZip; + try { + AdmZip = require('adm-zip'); + } catch (requireError) { + // Try alternative require paths + const mainPath = require.resolve('adm-zip', { paths: [process.cwd(), __dirname, path.join(__dirname, '..', 'node_modules')] }); + AdmZip = require(mainPath); + } + + const cudaZip = new AdmZip(cudaTempZipPath); + const cudaExtractDir = path.join(tempDir, 'cuda-extracted'); + cudaZip.extractAllTo(cudaExtractDir, true); + + // Find all CUDA runtime files (DLLs) in the extracted directory + const cudaFiles = await this.findCudaRuntimeFilesInExtracted(cudaExtractDir); + + if (cudaFiles.length === 0) { + logger.warn('No CUDA runtime files found in extracted CUDA archive'); + return []; + } + + logger.info(`Found ${cudaFiles.length} CUDA runtime files to install`); + return cudaFiles; + + } catch (error) { + logger.error('Error downloading CUDA runtime binaries:', error); + throw new Error(`Failed to download CUDA runtime: ${error.message}`); + } + } + + // Find CUDA runtime files in extracted directory + async findCudaRuntimeFilesInExtracted(extractDir) { + const path = require('path'); + const fs = require('fs').promises; + + const cudaFiles = []; + + // CUDA runtime files we're looking for + const cudaRuntimePatterns = [ + /^cudart64_\d+\.dll$/i, // CUDA runtime (e.g., cudart64_124.dll) + /^cublas64_\d+\.dll$/i, // cuBLAS (e.g., cublas64_12.dll) + /^cublasLt64_\d+\.dll$/i, // cuBLAS LT (e.g., cublasLt64_12.dll) + /^curand64_\d+\.dll$/i, // cuRAND (e.g., curand64_10.dll) + /^cusolver64_\d+\.dll$/i, // cuSOLVER (e.g., cusolver64_11.dll) + /^cusparse64_\d+\.dll$/i, // cuSPARSE (e.g., cusparse64_12.dll) + /^nvrtc64_\d+\.dll$/i, // NVRTC (e.g., nvrtc64_120_0.dll) + /^nvrtc-builtins64_\d+\.dll$/i // NVRTC builtins + ]; + + // Helper function to search directory recursively + async function searchDirectory(dir) { + const entries = await fs.readdir(dir, { withFileTypes: true }); + + for (const entry of entries) { + const fullPath = path.join(dir, entry.name); + + if (entry.isDirectory()) { + await searchDirectory(fullPath); + } else if (entry.isFile()) { + // Check if this file matches any CUDA runtime pattern + const matchesCudaPattern = cudaRuntimePatterns.some(pattern => + pattern.test(entry.name) + ); + + if (matchesCudaPattern) { + cudaFiles.push([fullPath, entry.name]); + logger.info(`Found CUDA runtime file: ${entry.name}`); + } + } + } + } + + await searchDirectory(extractDir); + + return cudaFiles; + } + + // Download CUDA runtime binaries for Windows using PowerShell extraction (fallback method) + async downloadCudaRuntimeBinariesWindows(updateInfo, tempDir) { + const path = require('path'); + const fs = require('fs').promises; + + try { + // Get the latest release info to find CUDA runtime assets + const response = await makeRobustRequest(`https://api.github.com/repos/${this.githubRepo}/releases/latest`); + const release = await response.json(); + + // Look for CUDA runtime assets (e.g., cudart-llama-bin-win-cuda-12.4-x64.zip) + const cudaAssets = release.assets.filter(asset => + asset.name.match(/cudart.*llama.*bin.*win.*cuda.*x64\.zip/i) + ); + + if (cudaAssets.length === 0) { + logger.info('No CUDA runtime binaries found in release assets (Windows fallback)'); + return []; + } + + // Sort by name to get the latest CUDA version (usually the highest version number) + cudaAssets.sort((a, b) => b.name.localeCompare(a.name)); + const cudaAsset = cudaAssets[0]; + + logger.info(`Found CUDA runtime asset: ${cudaAsset.name} (Windows fallback)`); + logger.info(`Downloading CUDA runtime binaries from: ${cudaAsset.browser_download_url}`); + + // Download the CUDA runtime zip + const cudaResponse = await makeRobustRequest(cudaAsset.browser_download_url); + const cudaBuffer = Buffer.from(await cudaResponse.arrayBuffer()); + + // Save CUDA runtime zip + const cudaTempZipPath = path.join(tempDir, 'cuda-runtime.zip'); + await fs.writeFile(cudaTempZipPath, cudaBuffer); + logger.info(`Downloaded CUDA runtime to: ${cudaTempZipPath} (Windows fallback)`); + + // Extract CUDA runtime zip using PowerShell + const cudaExtractDir = path.join(tempDir, 'cuda-extracted'); + await fs.mkdir(cudaExtractDir, { recursive: true }); + + const extractSuccess = await this.extractWithPowerShell(cudaTempZipPath, cudaExtractDir); + + if (!extractSuccess) { + throw new Error('Could not extract CUDA runtime zip using PowerShell'); + } + + // Find all CUDA runtime files (DLLs) in the extracted directory + const cudaFiles = await this.findCudaRuntimeFilesInExtracted(cudaExtractDir); + + if (cudaFiles.length === 0) { + logger.warn('No CUDA runtime files found in extracted CUDA archive (Windows fallback)'); + return []; + } + + logger.info(`Found ${cudaFiles.length} CUDA runtime files to install (Windows fallback)`); + return cudaFiles; + + } catch (error) { + logger.error('Error downloading CUDA runtime binaries (Windows fallback):', error); + throw new Error(`Failed to download CUDA runtime (Windows fallback): ${error.message}`); + } + } +} + +// Create llama.cpp update service instance +const llamacppUpdateService = new LlamacppUpdateService(); + +// Safe llama.cpp update check for UI +async function checkLlamacppUpdates() { + try { + return await llamacppUpdateService.checkForUpdates(); + } catch (error) { + logger.error('Error checking llama.cpp updates:', error); + + let errorMessage = 'Failed to check for llama.cpp updates'; + + if (error instanceof UpdateError) { + errorMessage = error.message; + } else { + errorMessage = error.message || 'Unknown error occurred'; + } + + return { + hasUpdate: false, + error: errorMessage, + platform: llamacppUpdateService.platform, + currentVersion: llamacppUpdateService.getCurrentVersion() + }; + } +} + +// Safe llama.cpp binary update +async function updateLlamacppBinaries() { + try { + return await llamacppUpdateService.updateBinaries(); + } catch (error) { + logger.error('Error updating llama.cpp binaries:', error); + return { + success: false, + error: error.message || 'Failed to update binaries' + }; + } +} + +module.exports = { + setupAutoUpdater, + checkForUpdates, + getUpdateInfo, + checkLlamacppUpdates, + updateLlamacppBinaries, + platformUpdateService +}; \ No newline at end of file diff --git a/electron/updateServiceEnhanced.cjs b/electron/updateServiceEnhanced.cjs new file mode 100644 index 00000000..631eef9a --- /dev/null +++ b/electron/updateServiceEnhanced.cjs @@ -0,0 +1,1160 @@ +const { autoUpdater } = require('electron-updater'); +const { dialog, BrowserWindow } = require('electron'); +const { shell } = require('electron'); +const fs = require('fs'); +const path = require('path'); + +// Use node-fetch for HTTP requests with comprehensive fallback +let fetch; +let AbortController; + +try { + // Try to use global fetch first (Node.js 18+) + fetch = globalThis.fetch; + AbortController = globalThis.AbortController; +} catch (error) { + // Fallback to node-fetch for older versions + try { + const nodeFetch = require('node-fetch'); + fetch = nodeFetch.default || nodeFetch; + AbortController = require('abort-controller').AbortController; + } catch (fetchError) { + console.warn('No fetch implementation available. Update checking will not work.'); + fetch = null; + AbortController = null; + } +} + +// Configure logging with error boundaries +let logger; +try { + logger = require('electron-log'); + autoUpdater.logger = logger; + autoUpdater.logger.transports.file.level = 'info'; +} catch (error) { + console.warn('Electron log not available, using console'); + logger = console; +} + +// Enhanced constants for robust update handling with UX improvements +const UPDATE_CONSTANTS = { + GITHUB_API_TIMEOUT: 15000, + MAX_RETRIES: 3, + RETRY_DELAY: 2000, + RATE_LIMIT_DELAY: 60000, + MAX_RELEASE_NOTES_LENGTH: 2000, // Increased for better release notes + // Enhanced UX constants + NOTIFICATION_DELAY: 1500, + PROGRESS_UPDATE_INTERVAL: 500, + BACKGROUND_CHECK_INTERVAL: 24 * 60 * 60 * 1000, // 24 hours + AUTO_CHECK_STARTUP_DELAY: 30000, // 30 seconds after startup + DOWNLOAD_CHUNK_SIZE: 1024 * 1024, // 1MB chunks +}; + +// Update preferences management +const UPDATE_PREFERENCES_KEY = 'clara-update-preferences'; +const DEFAULT_UPDATE_PREFERENCES = { + autoCheck: true, + checkFrequency: 'daily', // 'daily', 'weekly', 'monthly', 'manual' + notifyOnAvailable: true, + backgroundDownload: false, // For manual platforms + quietHours: { + enabled: false, + start: '22:00', + end: '08:00' + }, + betaChannel: false, + lastAutoCheck: null, + dismissedVersions: [] // Versions user chose to skip +}; + +// Enhanced release notes processing with markdown support +function processReleaseNotes(notes) { + if (!notes || typeof notes !== 'string') { + return { + plain: 'No release notes available.', + formatted: 'No release notes available.', + categories: {} + }; + } + + // Enhanced sanitization while preserving markdown structure + let sanitized = notes + .replace(/]*>.*?<\/script>/gis, '') // Remove scripts + .replace(/]*>.*?<\/iframe>/gis, '') // Remove iframes + .replace(/javascript:/gi, '') // Remove javascript: URLs + .trim(); + + // Parse and categorize content + const categories = { + 'New Features': [], + 'Improvements': [], + 'Bug Fixes': [], + 'Breaking Changes': [], + 'Other': [] + }; + + // Simple markdown-aware categorization + const lines = sanitized.split('\n'); + let currentCategory = 'Other'; + + for (const line of lines) { + const trimmed = line.trim(); + if (!trimmed) continue; + + // Detect category headers + if (trimmed.match(/^#+\s*(new features?|features?)/i)) { + currentCategory = 'New Features'; + continue; + } else if (trimmed.match(/^#+\s*(improvements?|enhancements?)/i)) { + currentCategory = 'Improvements'; + continue; + } else if (trimmed.match(/^#+\s*(bug fixes?|fixes?|bugfixes?)/i)) { + currentCategory = 'Bug Fixes'; + continue; + } else if (trimmed.match(/^#+\s*(breaking changes?|breaking)/i)) { + currentCategory = 'Breaking Changes'; + continue; + } + + // Add content to current category + if (trimmed.startsWith('- ') || trimmed.startsWith('* ') || trimmed.startsWith('+ ')) { + categories[currentCategory].push(trimmed.substring(2).trim()); + } else if (trimmed && !trimmed.startsWith('#')) { + categories[currentCategory].push(trimmed); + } + } + + // Create formatted version + let formatted = ''; + for (const [category, items] of Object.entries(categories)) { + if (items.length > 0) { + formatted += `**${category}:**\n`; + for (const item of items) { + formatted += `• ${item}\n`; + } + formatted += '\n'; + } + } + + // Limit length if needed + const plainText = sanitized.replace(/[#*_`]/g, ''); + if (plainText.length > UPDATE_CONSTANTS.MAX_RELEASE_NOTES_LENGTH) { + sanitized = plainText.substring(0, UPDATE_CONSTANTS.MAX_RELEASE_NOTES_LENGTH) + '...'; + formatted = formatted.substring(0, UPDATE_CONSTANTS.MAX_RELEASE_NOTES_LENGTH) + '...'; + } + + return { + plain: plainText || 'No release notes available.', + formatted: formatted || plainText || 'No release notes available.', + categories, + hasBreakingChanges: categories['Breaking Changes'].length > 0 + }; +} + +// Update preferences management with error handling +function getUpdatePreferences() { + try { + // Try localStorage first (for renderer process) + if (typeof localStorage !== 'undefined') { + const stored = localStorage.getItem(UPDATE_PREFERENCES_KEY); + if (stored) { + return { ...DEFAULT_UPDATE_PREFERENCES, ...JSON.parse(stored) }; + } + } + + // Fallback to file-based storage (for main process) + const { app } = require('electron'); + const prefsPath = path.join(app.getPath('userData'), 'update-preferences.json'); + + if (fs.existsSync(prefsPath)) { + const stored = JSON.parse(fs.readFileSync(prefsPath, 'utf8')); + return { ...DEFAULT_UPDATE_PREFERENCES, ...stored }; + } + } catch (error) { + logger.warn('Failed to load update preferences:', error); + } + return { ...DEFAULT_UPDATE_PREFERENCES }; +} + +function saveUpdatePreferences(preferences) { + try { + const current = getUpdatePreferences(); + const updated = { ...current, ...preferences, lastUpdated: new Date().toISOString() }; + + // Try localStorage first (for renderer process) + if (typeof localStorage !== 'undefined') { + localStorage.setItem(UPDATE_PREFERENCES_KEY, JSON.stringify(updated)); + } else { + // Fallback to file-based storage (for main process) + const { app } = require('electron'); + const prefsPath = path.join(app.getPath('userData'), 'update-preferences.json'); + fs.writeFileSync(prefsPath, JSON.stringify(updated, null, 2)); + } + + logger.info('Update preferences saved:', updated); + return updated; + } catch (error) { + logger.error('Failed to save update preferences:', error); + return null; + } +} + +// Smart timing for update checks +function isQuietTime(preferences = getUpdatePreferences()) { + if (!preferences.quietHours.enabled) return false; + + const now = new Date(); + const currentTime = now.getHours() * 60 + now.getMinutes(); + + const [startHour, startMin] = preferences.quietHours.start.split(':').map(Number); + const [endHour, endMin] = preferences.quietHours.end.split(':').map(Number); + + const startTime = startHour * 60 + startMin; + const endTime = endHour * 60 + endMin; + + if (startTime <= endTime) { + return currentTime >= startTime && currentTime <= endTime; + } else { + // Quiet hours span midnight + return currentTime >= startTime || currentTime <= endTime; + } +} + +function shouldAutoCheck(preferences = getUpdatePreferences()) { + if (!preferences.autoCheck) return false; + if (isQuietTime(preferences)) return false; + + const lastCheck = preferences.lastAutoCheck ? new Date(preferences.lastAutoCheck) : null; + if (!lastCheck) return true; + + const now = new Date(); + const timeDiff = now.getTime() - lastCheck.getTime(); + + switch (preferences.checkFrequency) { + case 'daily': + return timeDiff >= 24 * 60 * 60 * 1000; + case 'weekly': + return timeDiff >= 7 * 24 * 60 * 60 * 1000; + case 'monthly': + return timeDiff >= 30 * 24 * 60 * 60 * 1000; + default: + return false; + } +} + +// Robust version validation +function validateVersion(version) { + if (!version || typeof version !== 'string') { + throw new Error('Invalid version: must be a non-empty string'); + } + + const cleanVersion = version.replace(/^v/, '').trim(); + const versionRegex = /^\d+(\.\d+){0,3}(-[a-zA-Z0-9-]+)?$/; + + if (!versionRegex.test(cleanVersion)) { + throw new Error(`Invalid version format: ${version}`); + } + + return cleanVersion; +} + +// Safe package.json reading with validation +function getSafeCurrentVersion() { + try { + const packagePath = path.join(__dirname, '../package.json'); + + if (!fs.existsSync(packagePath)) { + throw new Error('Package.json not found'); + } + + const packageContent = fs.readFileSync(packagePath, 'utf8'); + const packageData = JSON.parse(packageContent); + + if (!packageData.version) { + throw new Error('Version not found in package.json'); + } + + return validateVersion(packageData.version); + } catch (error) { + logger.error('Error reading version from package.json:', error); + // Fallback version to prevent crashes + return '1.0.0'; + } +} + +// Enhanced error classification +class UpdateError extends Error { + constructor(message, type = 'UNKNOWN', retryable = false) { + super(message); + this.name = 'UpdateError'; + this.type = type; + this.retryable = retryable; + } +} + +// Robust network request with retry logic and progress tracking +async function makeRobustRequest(url, options = {}) { + if (!fetch) { + throw new UpdateError('Network functionality not available', 'NO_FETCH', false); + } + + const controller = AbortController ? new AbortController() : null; + const timeoutId = controller ? setTimeout(() => controller.abort(), UPDATE_CONSTANTS.GITHUB_API_TIMEOUT) : null; + + const requestOptions = { + ...options, + signal: controller?.signal, + headers: { + 'User-Agent': 'Clara-App-Updater', + 'Accept': 'application/vnd.github.v3+json', + ...options.headers + } + }; + + let lastError; + + for (let attempt = 1; attempt <= UPDATE_CONSTANTS.MAX_RETRIES; attempt++) { + try { + const response = await fetch(url, requestOptions); + + if (timeoutId) clearTimeout(timeoutId); + + // Handle rate limiting + if (response.status === 403) { + const rateLimitReset = response.headers.get('X-RateLimit-Reset'); + if (rateLimitReset) { + const resetTime = new Date(parseInt(rateLimitReset) * 1000); + const waitTime = Math.min(resetTime - Date.now(), UPDATE_CONSTANTS.RATE_LIMIT_DELAY); + throw new UpdateError( + `GitHub API rate limit exceeded. Try again in ${Math.ceil(waitTime / 1000)} seconds.`, + 'RATE_LIMIT', + false + ); + } + } + + if (!response.ok) { + throw new UpdateError( + `GitHub API error: ${response.status} ${response.statusText}`, + 'API_ERROR', + response.status >= 500 || response.status === 429 + ); + } + + return response; + } catch (error) { + lastError = error; + + if (timeoutId) clearTimeout(timeoutId); + + // Don't retry for non-retryable errors + if (error instanceof UpdateError && !error.retryable) { + throw error; + } + + // Don't retry on the last attempt + if (attempt === UPDATE_CONSTANTS.MAX_RETRIES) { + break; + } + + // Wait before retrying with exponential backoff + const delay = UPDATE_CONSTANTS.RETRY_DELAY * Math.pow(2, attempt - 1); + await new Promise(resolve => setTimeout(resolve, delay)); + } + } + + throw lastError || new UpdateError('All retry attempts failed', 'NETWORK_ERROR', false); +} + +// Validate GitHub release data structure +function validateReleaseData(release) { + if (!release || typeof release !== 'object') { + throw new UpdateError('Invalid release data structure', 'INVALID_DATA', false); + } + + const requiredFields = ['tag_name', 'html_url', 'assets']; + for (const field of requiredFields) { + if (!release[field]) { + throw new UpdateError(`Missing required field: ${field}`, 'INVALID_DATA', false); + } + } + + if (!Array.isArray(release.assets)) { + throw new UpdateError('Release assets must be an array', 'INVALID_DATA', false); + } + + return true; +} + +// Enhanced platform-specific update service with comprehensive UX improvements +class EnhancedPlatformUpdateService { + constructor() { + this.platform = process.platform; + this.currentVersion = getSafeCurrentVersion(); + this.githubRepo = 'badboysm890/ClaraVerse'; + this.isChecking = false; // Prevent concurrent checks + this.downloadProgress = null; // Track download progress + this.backgroundDownload = null; // Background download state + this.notificationCallbacks = new Set(); // UI notification callbacks + this.preferences = getUpdatePreferences(); + this.autoCheckTimer = null; + + // Initialize background checking if enabled + this.initializeAutoCheck(); + } + + /** + * Initialize automatic update checking + */ + initializeAutoCheck() { + // Clear any existing timer + if (this.autoCheckTimer) { + clearTimeout(this.autoCheckTimer); + } + + const preferences = getUpdatePreferences(); + + if (preferences.autoCheck) { + // Check on startup (delayed) + this.autoCheckTimer = setTimeout(() => { + this.performBackgroundCheck(); + }, UPDATE_CONSTANTS.AUTO_CHECK_STARTUP_DELAY); + + // Set up periodic checks + setInterval(() => { + if (shouldAutoCheck()) { + this.performBackgroundCheck(); + } + }, UPDATE_CONSTANTS.BACKGROUND_CHECK_INTERVAL); + } + } + + /** + * Perform background update check + */ + async performBackgroundCheck() { + try { + const preferences = getUpdatePreferences(); + + if (!preferences.autoCheck || isQuietTime(preferences)) { + return; + } + + logger.info('Performing background update check...'); + + const updateInfo = await this.checkGitHubReleases(); + + // Update last check time + saveUpdatePreferences({ lastAutoCheck: new Date().toISOString() }); + + if (updateInfo.hasUpdate && preferences.notifyOnAvailable) { + // Check if this version was dismissed + if (!preferences.dismissedVersions.includes(updateInfo.latestVersion)) { + this.notify('update-available', updateInfo); + + // Show native notification if supported + this.showNativeNotification(updateInfo); + } + } + + logger.info('Background update check completed:', { + hasUpdate: updateInfo.hasUpdate, + version: updateInfo.latestVersion + }); + + } catch (error) { + logger.error('Background update check failed:', error); + } + } + + /** + * Show native system notification + */ + showNativeNotification(updateInfo) { + try { + const { Notification } = require('electron'); + + if (Notification.isSupported()) { + const notification = new Notification({ + title: `Clara ${updateInfo.latestVersion} Available`, + body: `A new version of Clara is ready to download. Click to view details.`, + icon: path.join(__dirname, '../assets/icons/icon.png'), // Adjust path as needed + silent: false + }); + + notification.on('click', () => { + // Open settings to updates tab + const windows = BrowserWindow.getAllWindows(); + if (windows.length > 0) { + const mainWindow = windows[0]; + mainWindow.show(); + mainWindow.webContents.send('navigate-to-updates'); + } + }); + + notification.show(); + } + } catch (error) { + logger.warn('Failed to show native notification:', error); + } + } + + /** + * Register callback for update notifications + */ + onNotification(callback) { + this.notificationCallbacks.add(callback); + return () => this.notificationCallbacks.delete(callback); + } + + /** + * Send notification to all registered callbacks + */ + notify(type, data) { + for (const callback of this.notificationCallbacks) { + try { + callback(type, data); + } catch (error) { + logger.error('Error in notification callback:', error); + } + } + } + + /** + * Update preferences and reinitialize if needed + */ + updatePreferences(newPreferences) { + const updated = saveUpdatePreferences(newPreferences); + if (updated) { + this.preferences = updated; + this.initializeAutoCheck(); // Reinitialize with new settings + } + return updated; + } + + /** + * Dismiss a specific version (user chose to skip) + */ + dismissVersion(version) { + const preferences = getUpdatePreferences(); + const dismissedVersions = [...preferences.dismissedVersions, version]; + return this.updatePreferences({ dismissedVersions }); + } + + /** + * Check if OTA updates are supported for the current platform + */ + isOTASupported() { + // Only Mac supports OTA updates because it's signed + return this.platform === 'darwin'; + } + + /** + * Safe GitHub releases check with comprehensive validation and enhanced UX + */ + async checkGitHubReleases() { + // Prevent concurrent update checks + if (this.isChecking) { + throw new UpdateError('Update check already in progress', 'CONCURRENT_CHECK', false); + } + + this.isChecking = true; + this.notify('check-started', { timestamp: new Date() }); + + try { + const url = `https://api.github.com/repos/${this.githubRepo}/releases/latest`; + logger.info(`Checking for updates at: ${url}`); + + const response = await makeRobustRequest(url); + const release = await response.json(); + + // Validate release data structure + validateReleaseData(release); + + const latestVersion = validateVersion(release.tag_name); + const hasUpdate = this.isVersionNewer(latestVersion, this.currentVersion); + + // Enhanced release notes processing + const processedNotes = processReleaseNotes(release.body); + + const updateInfo = { + hasUpdate, + latestVersion, + currentVersion: this.currentVersion, + releaseUrl: release.html_url, + downloadUrl: this.getDownloadUrlForPlatform(release.assets), + releaseNotes: processedNotes.plain, + releaseNotesFormatted: processedNotes.formatted, + releaseNotesCategories: processedNotes.categories, + hasBreakingChanges: processedNotes.hasBreakingChanges, + publishedAt: release.published_at, + assetSize: this.getAssetSize(release.assets), + downloadEstimate: this.estimateDownloadTime(release.assets) + }; + + logger.info('Update check completed successfully:', { + hasUpdate, + currentVersion: this.currentVersion, + latestVersion, + hasBreakingChanges: processedNotes.hasBreakingChanges + }); + + this.notify('check-completed', updateInfo); + return updateInfo; + + } catch (error) { + logger.error('Error checking GitHub releases:', error); + + const errorInfo = { + hasUpdate: false, + error: error instanceof UpdateError ? error.message : `Failed to check for updates: ${error.message}`, + currentVersion: this.currentVersion, + platform: this.platform, + isOTASupported: this.isOTASupported() + }; + + this.notify('check-failed', errorInfo); + + if (error instanceof UpdateError) { + throw error; + } + + throw new UpdateError( + `Failed to check for updates: ${error.message}`, + 'UNKNOWN_ERROR', + true + ); + } finally { + this.isChecking = false; + } + } + + /** + * Get asset size for download estimation + */ + getAssetSize(assets) { + if (!Array.isArray(assets)) return null; + + const platformAsset = this.findPlatformAsset(assets); + if (platformAsset && platformAsset.size) { + return this.formatFileSize(platformAsset.size); + } + + return null; + } + + /** + * Estimate download time based on asset size + */ + estimateDownloadTime(assets) { + if (!Array.isArray(assets)) return null; + + const platformAsset = this.findPlatformAsset(assets); + if (platformAsset && platformAsset.size) { + // Assume average download speed of 10 Mbps (conservative estimate) + const avgSpeedBytesPerSecond = (10 * 1024 * 1024) / 8; // 10 Mbps to bytes/sec + const estimatedSeconds = platformAsset.size / avgSpeedBytesPerSecond; + + if (estimatedSeconds < 60) { + return `< 1 minute`; + } else if (estimatedSeconds < 3600) { + return `~ ${Math.ceil(estimatedSeconds / 60)} minutes`; + } else { + return `~ ${Math.ceil(estimatedSeconds / 3600)} hours`; + } + } + + return null; + } + + /** + * Find platform-specific asset + */ + findPlatformAsset(assets) { + const platformExtensions = { + darwin: ['.dmg', '-arm64.dmg', '-mac.dmg'], + win32: ['.exe', '-win.exe', '-windows.exe'], + linux: ['.AppImage', '.deb', '-linux.AppImage'] + }; + + const extensions = platformExtensions[this.platform] || []; + + for (const ext of extensions) { + const asset = assets.find(asset => + asset && + asset.name && + typeof asset.name === 'string' && + asset.name.toLowerCase().endsWith(ext.toLowerCase()) && + asset.browser_download_url + ); + + if (asset) { + return asset; + } + } + + return null; + } + + /** + * Format file size in human readable format + */ + formatFileSize(bytes) { + if (bytes === 0) return '0 Bytes'; + const k = 1024; + const sizes = ['Bytes', 'KB', 'MB', 'GB']; + const i = Math.floor(Math.log(bytes) / Math.log(k)); + return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i]; + } + + /** + * Get the appropriate download URL for the current platform with validation + */ + getDownloadUrlForPlatform(assets) { + if (!Array.isArray(assets)) { + logger.warn('Invalid assets array, using fallback URL'); + return `https://github.com/${this.githubRepo}/releases/latest`; + } + + const platformAsset = this.findPlatformAsset(assets); + + if (platformAsset) { + logger.info(`Found platform-specific download: ${platformAsset.name}`); + return platformAsset.browser_download_url; + } + + // Fallback to releases page + logger.info('No platform-specific download found, using releases page'); + return `https://github.com/${this.githubRepo}/releases/latest`; + } + + /** + * Robust version comparison with detailed logging + */ + isVersionNewer(newVersion, currentVersion) { + try { + const parseVersion = (version) => { + return version.split('.').map(num => { + const parsed = parseInt(num, 10); + return isNaN(parsed) ? 0 : parsed; + }); + }; + + const newParts = parseVersion(newVersion); + const currentParts = parseVersion(currentVersion); + const maxLength = Math.max(newParts.length, currentParts.length); + + for (let i = 0; i < maxLength; i++) { + const newPart = newParts[i] || 0; + const currentPart = currentParts[i] || 0; + + if (newPart > currentPart) { + logger.info(`Version ${newVersion} is newer than ${currentVersion}`); + return true; + } + if (newPart < currentPart) { + logger.info(`Version ${newVersion} is older than ${currentVersion}`); + return false; + } + } + + logger.info(`Version ${newVersion} is same as ${currentVersion}`); + return false; + } catch (error) { + logger.error('Error comparing versions:', error); + return false; // Safe fallback + } + } + + /** + * Enhanced platform-specific update dialog with beautiful UX + */ + async showEnhancedUpdateDialog(updateInfo) { + try { + const { hasUpdate, latestVersion, downloadUrl, releaseNotesFormatted, hasBreakingChanges } = updateInfo; + + if (!hasUpdate) { + return await dialog.showMessageBox({ + type: 'info', + title: '✅ You\'re Up to Date!', + message: 'Clara is current', + detail: `You're running Clara ${this.currentVersion}, which is the latest version available.`, + buttons: ['Perfect!'], + defaultId: 0 + }); + } + + // Build enhanced message with categorized release notes + let detailMessage = `Current version: Clara ${this.currentVersion}\nNew version: Clara ${latestVersion}\n\n`; + + if (hasBreakingChanges) { + detailMessage += `⚠️ This update contains breaking changes. Please review the release notes.\n\n`; + } + + if (updateInfo.assetSize) { + detailMessage += `Download size: ${updateInfo.assetSize}`; + if (updateInfo.downloadEstimate) { + detailMessage += ` (${updateInfo.downloadEstimate})`; + } + detailMessage += '\n\n'; + } + + if (releaseNotesFormatted && releaseNotesFormatted !== 'No release notes available.') { + const truncated = releaseNotesFormatted.length > 400 + ? releaseNotesFormatted.substring(0, 400) + '...\n\nClick "Release Notes" for full details.' + : releaseNotesFormatted; + detailMessage += `What's new:\n${truncated}`; + } + + if (this.isOTASupported()) { + // Mac: Enhanced OTA update dialog + return await dialog.showMessageBox({ + type: 'info', + title: hasBreakingChanges ? '⚠️ Important Update Available' : '🎉 Update Available', + message: `Clara ${latestVersion} is ready to install`, + detail: detailMessage, + buttons: ['Download & Install Now', 'View Release Notes', 'Remind Me Later', 'Skip This Version'], + defaultId: 0, + cancelId: 2 + }).then(({ response }) => { + try { + switch (response) { + case 0: + // Start OTA update with progress tracking + this.startOTAUpdateWithProgress(); + return { action: 'download' }; + case 1: + // Open release notes + shell.openExternal(updateInfo.releaseUrl); + return { action: 'view_notes' }; + case 2: + return { action: 'later' }; + case 3: + // Skip this version + this.dismissVersion(latestVersion); + return { action: 'dismissed' }; + default: + return { action: 'later' }; + } + } catch (error) { + logger.error('Error handling dialog response:', error); + return { action: 'error', error: error.message }; + } + }); + } else { + // Windows/Linux: Enhanced manual update dialog + const platformName = this.platform === 'win32' ? 'Windows' : 'Linux'; + + detailMessage += `\n🔒 On ${platformName}, updates are installed manually for security. The download will open in your browser.`; + + return await dialog.showMessageBox({ + type: 'info', + title: hasBreakingChanges ? '⚠️ Important Update Available' : '📦 Update Available', + message: `Clara ${latestVersion} is ready to download`, + detail: detailMessage, + buttons: ['Download Now', 'View Release Notes', 'Remind Me Later', 'Skip This Version'], + defaultId: 0, + cancelId: 2 + }).then(({ response }) => { + try { + switch (response) { + case 0: + // Open download page + shell.openExternal(downloadUrl); + return { action: 'download' }; + case 1: + // Open release notes + shell.openExternal(updateInfo.releaseUrl); + return { action: 'view_notes' }; + case 2: + return { action: 'later' }; + case 3: + // Skip this version + this.dismissVersion(latestVersion); + return { action: 'dismissed' }; + default: + return { action: 'later' }; + } + } catch (error) { + logger.error('Error handling dialog response:', error); + return { action: 'error', error: error.message }; + } + }); + } + } catch (error) { + logger.error('Error showing enhanced update dialog:', error); + + // Show fallback error dialog + try { + await dialog.showErrorBox( + '❌ Update Dialog Error', + `Failed to show update information: ${error.message}\n\nPlease check for updates manually at: https://github.com/${this.githubRepo}/releases` + ); + } catch (dialogError) { + logger.error('Failed to show error dialog:', dialogError); + } + + return { action: 'error', error: error.message }; + } + } + + /** + * Start OTA update with progress tracking + */ + startOTAUpdateWithProgress() { + try { + this.notify('download-started', { timestamp: new Date() }); + autoUpdater.downloadUpdate(); + } catch (error) { + logger.error('Failed to start OTA update:', error); + this.notify('download-error', { error: error.message }); + } + } +} + +// Create enhanced global instance with error protection +let enhancedPlatformUpdateService; +try { + enhancedPlatformUpdateService = new EnhancedPlatformUpdateService(); +} catch (error) { + logger.error('Failed to initialize enhanced update service:', error); + enhancedPlatformUpdateService = null; +} + +// Enhanced auto-updater setup with comprehensive error handling and progress tracking +function setupEnhancedAutoUpdater(mainWindow) { + if (!enhancedPlatformUpdateService || !enhancedPlatformUpdateService.isOTASupported()) { + logger.info('OTA updates not supported on this platform'); + return; + } + + try { + // Enhanced progress tracking + autoUpdater.on('download-progress', (progressObj) => { + try { + const progress = { + percent: Math.round(progressObj.percent), + transferred: enhancedPlatformUpdateService.formatFileSize(progressObj.transferred), + total: enhancedPlatformUpdateService.formatFileSize(progressObj.total), + bytesPerSecond: enhancedPlatformUpdateService.formatFileSize(progressObj.bytesPerSecond) + '/s' + }; + + enhancedPlatformUpdateService.notify('download-progress', progress); + + if (mainWindow && mainWindow.webContents) { + mainWindow.webContents.send('update-download-progress', progress); + } + } catch (error) { + logger.error('Error processing download progress:', error); + } + }); + + // Update available with enhanced dialog + autoUpdater.on('update-available', async (info) => { + try { + logger.info('OTA update available:', info); + + const updateInfo = { + hasUpdate: true, + latestVersion: info.version, + currentVersion: enhancedPlatformUpdateService.currentVersion, + releaseNotes: info.releaseNotes || 'Release notes not available', + publishedAt: info.releaseDate + }; + + enhancedPlatformUpdateService.notify('update-available', updateInfo); + + } catch (error) { + logger.error('Error in update-available handler:', error); + } + }); + + // Update downloaded with enhanced dialog + autoUpdater.on('update-downloaded', () => { + try { + enhancedPlatformUpdateService.notify('download-completed', { timestamp: new Date() }); + + dialog.showMessageBox({ + type: 'info', + title: '🎉 Update Ready!', + message: 'Clara has been updated successfully', + detail: 'The update has been downloaded and verified. Clara will restart to complete the installation.', + buttons: ['Restart Now', 'Restart Later'], + defaultId: 0, + cancelId: 1 + }).then(({ response }) => { + if (response === 0) { + try { + autoUpdater.quitAndInstall(); + } catch (error) { + logger.error('Error during quit and install:', error); + dialog.showErrorBox('Installation Failed', `Failed to install update: ${error.message}`); + } + } + }).catch(error => { + logger.error('Error showing update downloaded dialog:', error); + }); + } catch (error) { + logger.error('Error in update-downloaded handler:', error); + } + }); + + // Enhanced error handling + autoUpdater.on('error', (err) => { + logger.error('Auto-updater error:', err); + + enhancedPlatformUpdateService.notify('download-error', { error: err.message }); + + try { + // Fallback to GitHub-based updates on error + dialog.showErrorBox('❌ Update Error', + `Automatic update failed: ${err.message}\n\nYou can manually download the latest version from:\nhttps://github.com/${enhancedPlatformUpdateService.githubRepo}/releases` + ); + } catch (dialogError) { + logger.error('Failed to show error dialog:', dialogError); + } + }); + + // No update available + autoUpdater.on('update-not-available', (info) => { + try { + enhancedPlatformUpdateService.notify('no-update-available', { + currentVersion: enhancedPlatformUpdateService.currentVersion, + timestamp: new Date() + }); + } catch (error) { + logger.error('Error in update-not-available handler:', error); + } + }); + + logger.info('Enhanced auto-updater setup completed successfully'); + } catch (error) { + logger.error('Failed to setup enhanced auto-updater:', error); + } +} + +// Enhanced universal update check with comprehensive error handling +async function checkForUpdatesEnhanced() { + if (!enhancedPlatformUpdateService) { + const error = 'Enhanced update service not available'; + logger.error(error); + try { + dialog.showErrorBox('Update Service Error', error); + } catch (dialogError) { + logger.error('Failed to show error dialog:', dialogError); + } + return null; + } + + try { + if (enhancedPlatformUpdateService.isOTASupported()) { + // Mac: Use electron-updater first, fallback to GitHub + try { + return await autoUpdater.checkForUpdates(); + } catch (error) { + logger.warn('OTA update check failed, falling back to GitHub:', error); + const updateInfo = await enhancedPlatformUpdateService.checkGitHubReleases(); + return await enhancedPlatformUpdateService.showEnhancedUpdateDialog(updateInfo); + } + } else { + // Windows/Linux: Use enhanced GitHub releases + const updateInfo = await enhancedPlatformUpdateService.checkGitHubReleases(); + return await enhancedPlatformUpdateService.showEnhancedUpdateDialog(updateInfo); + } + } catch (error) { + logger.error('Error checking for updates:', error); + + let userMessage = 'Could not check for updates. Please check your internet connection and try again.'; + + if (error instanceof UpdateError) { + switch (error.type) { + case 'RATE_LIMIT': + userMessage = error.message; + break; + case 'NO_FETCH': + userMessage = 'Network functionality is not available. Please restart the application.'; + break; + case 'CONCURRENT_CHECK': + userMessage = 'Update check is already in progress. Please wait.'; + break; + default: + userMessage = `Update check failed: ${error.message}`; + } + } + + try { + dialog.showErrorBox('❌ Update Check Failed', userMessage); + } catch (dialogError) { + logger.error('Failed to show error dialog:', dialogError); + } + + return null; + } +} + +// Enhanced update info retrieval for UI with comprehensive data +async function getEnhancedUpdateInfo() { + if (!enhancedPlatformUpdateService) { + return { + hasUpdate: false, + error: 'Enhanced update service not available', + platform: process.platform, + isOTASupported: false, + currentVersion: getSafeCurrentVersion(), + preferences: DEFAULT_UPDATE_PREFERENCES + }; + } + + try { + const updateInfo = await enhancedPlatformUpdateService.checkGitHubReleases(); + const preferences = getUpdatePreferences(); + + return { + ...updateInfo, + platform: enhancedPlatformUpdateService.platform, + isOTASupported: enhancedPlatformUpdateService.isOTASupported(), + preferences, + lastAutoCheck: preferences.lastAutoCheck, + dismissedVersions: preferences.dismissedVersions + }; + } catch (error) { + logger.error('Error getting enhanced update info:', error); + + let errorMessage = 'Failed to check for updates'; + + if (error instanceof UpdateError) { + errorMessage = error.message; + } else { + errorMessage = error.message || 'Unknown error occurred'; + } + + return { + hasUpdate: false, + error: errorMessage, + platform: enhancedPlatformUpdateService.platform, + isOTASupported: enhancedPlatformUpdateService.isOTASupported(), + currentVersion: enhancedPlatformUpdateService.currentVersion, + preferences: getUpdatePreferences() + }; + } +} + +// Export both enhanced and legacy functions for compatibility +module.exports = { + // Enhanced functions (new) + setupEnhancedAutoUpdater, + checkForUpdatesEnhanced, + getEnhancedUpdateInfo, + enhancedPlatformUpdateService, + + // Preferences management + getUpdatePreferences, + saveUpdatePreferences, + + // Legacy functions (for backward compatibility) + setupAutoUpdater: setupEnhancedAutoUpdater, + checkForUpdates: checkForUpdatesEnhanced, + getUpdateInfo: getEnhancedUpdateInfo, + platformUpdateService: enhancedPlatformUpdateService +}; diff --git a/electron/vendor/gsap.min.js b/electron/vendor/gsap.min.js new file mode 100644 index 00000000..0c38bc45 --- /dev/null +++ b/electron/vendor/gsap.min.js @@ -0,0 +1,11 @@ +/*! + * GSAP 3.12.2 + * https://greensock.com + * + * @license Copyright 2023, GreenSock. All rights reserved. + * Subject to the terms at https://greensock.com/standard-license or for Club GreenSock members, the agreement issued with that membership. + * @author: Jack Doyle, jack@greensock.com + */ + +!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?e(exports):"function"==typeof define&&define.amd?define(["exports"],e):e((t=t||self).window=t.window||{})}(this,function(e){"use strict";function _inheritsLoose(t,e){t.prototype=Object.create(e.prototype),(t.prototype.constructor=t).__proto__=e}function _assertThisInitialized(t){if(void 0===t)throw new ReferenceError("this hasn't been initialised - super() hasn't been called");return t}function r(t){return"string"==typeof t}function s(t){return"function"==typeof t}function t(t){return"number"==typeof t}function u(t){return void 0===t}function v(t){return"object"==typeof t}function w(t){return!1!==t}function x(){return"undefined"!=typeof window}function y(t){return s(t)||r(t)}function P(t){return(i=yt(t,ot))&&Ee}function Q(t,e){return console.warn("Invalid property",t,"set to",e,"Missing plugin? gsap.registerPlugin()")}function R(t,e){return!e&&console.warn(t)}function S(t,e){return t&&(ot[t]=e)&&i&&(i[t]=e)||ot}function T(){return 0}function ea(t){var e,r,i=t[0];if(v(i)||s(i)||(t=[t]),!(e=(i._gsap||{}).harness)){for(r=gt.length;r--&&!gt[r].targetTest(i););e=gt[r]}for(r=t.length;r--;)t[r]&&(t[r]._gsap||(t[r]._gsap=new Vt(t[r],e)))||t.splice(r,1);return t}function fa(t){return t._gsap||ea(Ot(t))[0]._gsap}function ga(t,e,r){return(r=t[e])&&s(r)?t[e]():u(r)&&t.getAttribute&&t.getAttribute(e)||r}function ha(t,e){return(t=t.split(",")).forEach(e)||t}function ia(t){return Math.round(1e5*t)/1e5||0}function ja(t){return Math.round(1e7*t)/1e7||0}function ka(t,e){var r=e.charAt(0),i=parseFloat(e.substr(2));return t=parseFloat(t),"+"===r?t+i:"-"===r?t-i:"*"===r?t*i:t/i}function la(t,e){for(var r=e.length,i=0;t.indexOf(e[i])<0&&++ia;)s=s._prev;return s?(e._next=s._next,s._next=e):(e._next=t[r],t[r]=e),e._next?e._next._prev=e:t[i]=e,e._prev=s,e.parent=e._dp=t,e}function ya(t,e,r,i){void 0===r&&(r="_first"),void 0===i&&(i="_last");var n=e._prev,a=e._next;n?n._next=a:t[r]===e&&(t[r]=a),a?a._prev=n:t[i]===e&&(t[i]=n),e._next=e._prev=e.parent=null}function za(t,e){t.parent&&(!e||t.parent.autoRemoveChildren)&&t.parent.remove&&t.parent.remove(t),t._act=0}function Aa(t,e){if(t&&(!e||e._end>t._dur||e._start<0))for(var r=t;r;)r._dirty=1,r=r.parent;return t}function Ca(t,e,r,i){return t._startAt&&(L?t._startAt.revert(ht):t.vars.immediateRender&&!t.vars.autoRevert||t._startAt.render(e,!0,i))}function Ea(t){return t._repeat?Tt(t._tTime,t=t.duration()+t._rDelay)*t:0}function Ga(t,e){return(t-e._start)*e._ts+(0<=e._ts?0:e._dirty?e.totalDuration():e._tDur)}function Ha(t){return t._end=ja(t._start+(t._tDur/Math.abs(t._ts||t._rts||X)||0))}function Ia(t,e){var r=t._dp;return r&&r.smoothChildTiming&&t._ts&&(t._start=ja(r._time-(0X)&&e.render(r,!0)),Aa(t,e)._dp&&t._initted&&t._time>=t._dur&&t._ts){if(t._dur(n=Math.abs(n))&&(a=i,o=n);return a}function tb(t){return za(t),t.scrollTrigger&&t.scrollTrigger.kill(!!L),t.progress()<1&&At(t,"onInterrupt"),t}function wb(t){if(x()&&t){var e=(t=!t.name&&t.default||t).name,r=s(t),i=e&&!r&&t.init?function(){this._props=[]}:t,n={init:T,render:he,add:Qt,kill:ce,modifier:fe,rawVars:0},a={targetTest:0,get:0,getSetter:ne,aliases:{},register:0};if(Ft(),t!==i){if(pt[e])return;qa(i,qa(ua(t,n),a)),yt(i.prototype,yt(n,ua(t,a))),pt[i.prop=e]=i,t.targetTest&&(gt.push(i),ft[e]=1),e=("css"===e?"CSS":e.charAt(0).toUpperCase()+e.substr(1))+"Plugin"}S(e,i),t.register&&t.register(Ee,i,_e)}else t&&Ct.push(t)}function zb(t,e,r){return(6*(t+=t<0?1:1>16,e>>8&St,e&St]:0:Et.black;if(!p){if(","===e.substr(-1)&&(e=e.substr(0,e.length-1)),Et[e])p=Et[e];else if("#"===e.charAt(0)){if(e.length<6&&(e="#"+(n=e.charAt(1))+n+(a=e.charAt(2))+a+(s=e.charAt(3))+s+(5===e.length?e.charAt(4)+e.charAt(4):"")),9===e.length)return[(p=parseInt(e.substr(1,6),16))>>16,p>>8&St,p&St,parseInt(e.substr(7),16)/255];p=[(e=parseInt(e.substr(1),16))>>16,e>>8&St,e&St]}else if("hsl"===e.substr(0,3))if(p=d=e.match(tt),r){if(~e.indexOf("="))return p=e.match(et),i&&p.length<4&&(p[3]=1),p}else o=+p[0]%360/360,u=p[1]/100,n=2*(h=p[2]/100)-(a=h<=.5?h*(u+1):h+u-h*u),3=U?u.endTime(!1):t._dur;return r(e)&&(isNaN(e)||e in o)?(a=e.charAt(0),s="%"===e.substr(-1),n=e.indexOf("="),"<"===a||">"===a?(0<=n&&(e=e.replace(/=/,"")),("<"===a?u._start:u.endTime(0<=u._repeat))+(parseFloat(e.substr(1))||0)*(s?(n<0?u:i).totalDuration()/100:1)):n<0?(e in o||(o[e]=h),o[e]):(a=parseFloat(e.charAt(n-1)+e.substr(n+1)),s&&i&&(a=a/100*($(i)?i[0]:i).totalDuration()),1=r&&te)return i;i=i._next}else for(i=t._last;i&&i._start>=r;){if("isPause"===i.data&&i._start=n._start)&&n._ts&&h!==n){if(n.parent!==this)return this.render(t,e,r);if(n.render(0=this.totalDuration()||!v&&_)&&(f!==this._start&&Math.abs(l)===Math.abs(this._ts)||this._lock||(!t&&g||!(v===m&&0=i&&(a instanceof Zt?e&&n.push(a):(r&&n.push(a),t&&n.push.apply(n,a.getChildren(!0,e,r)))),a=a._next;return n},e.getById=function getById(t){for(var e=this.getChildren(1,1,1),r=e.length;r--;)if(e[r].vars.id===t)return e[r]},e.remove=function remove(t){return r(t)?this.removeLabel(t):s(t)?this.killTweensOf(t):(ya(this,t),t===this._recent&&(this._recent=this._last),Aa(this))},e.totalTime=function totalTime(t,e){return arguments.length?(this._forcing=1,!this._dp&&this._ts&&(this._start=ja(Rt.time-(0r:!r||s.isActive())&&n.push(s):(i=s.getTweensOf(a,r)).length&&n.push.apply(n,i),s=s._next;return n},e.tweenTo=function tweenTo(t,e){e=e||{};var r,i=this,n=xt(i,t),a=e.startAt,s=e.onStart,o=e.onStartParams,u=e.immediateRender,h=Zt.to(i,qa({ease:e.ease||"none",lazy:!1,immediateRender:!1,time:n,overwrite:"auto",duration:e.duration||Math.abs((n-(a&&"time"in a?a.time:i._time))/i.timeScale())||X,onStart:function onStart(){if(i.pause(),!r){var t=e.duration||Math.abs((n-(a&&"time"in a?a.time:i._time))/i.timeScale());h._dur!==t&&Ra(h,t,0,1).render(h._time,!0,!0),r=1}s&&s.apply(h,o||[])}},e));return u?h.render(0):h},e.tweenFromTo=function tweenFromTo(t,e,r){return this.tweenTo(e,qa({startAt:{time:xt(this,t)}},r))},e.recent=function recent(){return this._recent},e.nextLabel=function nextLabel(t){return void 0===t&&(t=this._time),rb(this,xt(this,t))},e.previousLabel=function previousLabel(t){return void 0===t&&(t=this._time),rb(this,xt(this,t),1)},e.currentLabel=function currentLabel(t){return arguments.length?this.seek(t,!0):this.previousLabel(this._time+X)},e.shiftChildren=function shiftChildren(t,e,r){void 0===r&&(r=0);for(var i,n=this._first,a=this.labels;n;)n._start>=r&&(n._start+=t,n._end+=t),n=n._next;if(e)for(i in a)a[i]>=r&&(a[i]+=t);return Aa(this)},e.invalidate=function invalidate(t){var e=this._first;for(this._lock=0;e;)e.invalidate(t),e=e._next;return i.prototype.invalidate.call(this,t)},e.clear=function clear(t){void 0===t&&(t=!0);for(var e,r=this._first;r;)e=r._next,this.remove(r),r=e;return this._dp&&(this._time=this._tTime=this._pTime=0),t&&(this.labels={}),Aa(this)},e.totalDuration=function totalDuration(t){var e,r,i,n=0,a=this,s=a._last,o=U;if(arguments.length)return a.timeScale((a._repeat<0?a.duration():a.totalDuration())/(a.reversed()?-t:t));if(a._dirty){for(i=a.parent;s;)e=s._prev,s._dirty&&s.totalDuration(),o<(r=s._start)&&a._sort&&s._ts&&!a._lock?(a._lock=1,Ka(a,s,r-s._delay,1)._lock=0):o=r,r<0&&s._ts&&(n-=r,(!i&&!a._dp||i&&i.smoothChildTiming)&&(a._start+=r/a._ts,a._time-=r,a._tTime-=r),a.shiftChildren(-r,!1,-Infinity),o=0),s._end>n&&s._ts&&(n=s._end),s=e;Ra(a,a===I&&a._time>n?a._time:n,1,1),a._dirty=0}return a._tDur},Timeline.updateRoot=function updateRoot(t){if(I._ts&&(na(I,Ga(t,I)),f=Rt.frame),Rt.frame>=mt){mt+=q.autoSleep||120;var e=I._first;if((!e||!e._ts)&&q.autoSleep&&Rt._listeners.length<2){for(;e&&!e._ts;)e=e._next;e||Rt.sleep()}}},Timeline}(Ut);qa(Xt.prototype,{_lock:0,_hasPause:0,_forcing:0});function ac(t,e,i,n,a,o){var u,h,l,f;if(pt[t]&&!1!==(u=new pt[t]).init(a,u.rawVars?e[t]:function _processVars(t,e,i,n,a){if(s(t)&&(t=Kt(t,a,e,i,n)),!v(t)||t.style&&t.nodeType||$(t)||Z(t))return r(t)?Kt(t,a,e,i,n):t;var o,u={};for(o in t)u[o]=Kt(t[o],a,e,i,n);return u}(e[t],n,a,o,i),i,n,o)&&(i._pt=h=new _e(i._pt,a,t,0,1,u.render,u,0,u.priority),i!==c))for(l=i._ptLookup[i._targets.indexOf(a)],f=u._props.length;f--;)l[u._props[f]]=h;return u}function gc(t,r,e,i){var n,a,s=r.ease||i||"power1.inOut";if($(r))a=e[t]||(e[t]=[]),r.forEach(function(t,e){return a.push({t:e/(r.length-1)*100,v:t,e:s})});else for(n in r)a=e[n]||(e[n]=[]),"ease"===n||a.push({t:parseFloat(t),v:r[n],e:s})}var Nt,Wt,Qt=function _addPropTween(t,e,i,n,a,o,u,h,l,f){s(n)&&(n=n(a||0,t,o));var c,d=t[e],p="get"!==i?i:s(d)?l?t[e.indexOf("set")||!s(t["get"+e.substr(3)])?e:"get"+e.substr(3)](l):t[e]():d,_=s(d)?l?re:te:$t;if(r(n)&&(~n.indexOf("random(")&&(n=ob(n)),"="===n.charAt(1)&&(!(c=ka(p,n)+(Ya(p)||0))&&0!==c||(n=c))),!f||p!==n||Wt)return isNaN(p*n)||""===n?(d||e in t||Q(e,n),function _addComplexStringPropTween(t,e,r,i,n,a,s){var o,u,h,l,f,c,d,p,_=new _e(this._pt,t,e,0,1,ue,null,n),m=0,g=0;for(_.b=r,_.e=i,r+="",(d=~(i+="").indexOf("random("))&&(i=ob(i)),a&&(a(p=[r,i],t,e),r=p[0],i=p[1]),u=r.match(it)||[];o=it.exec(i);)l=o[0],f=i.substring(m,o.index),h?h=(h+1)%5:"rgba("===f.substr(-5)&&(h=1),l!==u[g++]&&(c=parseFloat(u[g-1])||0,_._pt={_next:_._pt,p:f||1===g?f:",",s:c,c:"="===l.charAt(1)?ka(c,l)-c:parseFloat(l)-c,m:h&&h<4?Math.round:0},m=it.lastIndex);return _.c=m")}),s.duration();else{for(l in u={},x)"ease"===l||"easeEach"===l||gc(l,x[l],u,x.easeEach);for(l in u)for(C=u[l].sort(function(t,e){return t.t-e.t}),o=D=0;o=t._tDur||e<0)&&t.ratio===u&&(u&&za(t,1),r||L||(At(t,u?"onComplete":"onReverseComplete",!0),t._prom&&t._prom()))}else t._zTime||(t._zTime=e)}(this,t,e,r);return this},e.targets=function targets(){return this._targets},e.invalidate=function invalidate(t){return t&&this.vars.runBackwards||(this._startAt=0),this._pt=this._op=this._onUpdate=this._lazy=this.ratio=0,this._ptLookup=[],this.timeline&&this.timeline.invalidate(t),z.prototype.invalidate.call(this,t)},e.resetTo=function resetTo(t,e,r,i){d||Rt.wake(),this._ts||this.play();var n,a=Math.min(this._dur,(this._dp._time-this._start)*this._ts);return this._initted||Gt(this,a),n=this._ease(a/this._dur),function _updatePropTweens(t,e,r,i,n,a,s){var o,u,h,l,f=(t._pt&&t._ptCache||(t._ptCache={}))[e];if(!f)for(f=t._ptCache[e]=[],h=t._ptLookup,l=t._targets.length;l--;){if((o=h[l][e])&&o.d&&o.d._pt)for(o=o.d._pt;o&&o.p!==e&&o.fp!==e;)o=o._next;if(!o)return Wt=1,t.vars[e]="+=0",Gt(t,s),Wt=0,1;f.push(o)}for(l=f.length;l--;)(o=(u=f[l])._pt||u).s=!i&&0!==i||n?o.s+(i||0)+a*o.c:i,o.c=r-o.s,u.e&&(u.e=ia(r)+Ya(u.e)),u.b&&(u.b=o.s+Ya(u.b))}(this,t,e,r,i,n,a)?this.resetTo(t,e,r,i):(Ia(this,0),this.parent||xa(this._dp,this,"_first","_last",this._dp._sort?"_start":0),this.render(0))},e.kill=function kill(t,e){if(void 0===e&&(e="all"),!(t||e&&"all"!==e))return this._lazy=this._pt=0,this.parent?tb(this):this;if(this.timeline){var i=this.timeline.totalDuration();return this.timeline.killTweensOf(t,e,Nt&&!0!==Nt.vars.overwrite)._first||tb(this),this.parent&&i!==this.timeline.totalDuration()&&Ra(this,this._dur*this.timeline._tDur/i,0,1),this}var n,a,s,o,u,h,l,f=this._targets,c=t?Ot(t):f,d=this._ptLookup,p=this._pt;if((!e||"all"===e)&&function _arraysMatch(t,e){for(var r=t.length,i=r===e.length;i&&r--&&t[r]===e[r];);return r<0}(f,c))return"all"===e&&(this._pt=0),tb(this);for(n=this._op=this._op||[],"all"!==e&&(r(e)&&(u={},ha(e,function(t){return u[t]=1}),e=u),e=function _addAliasesToVars(t,e){var r,i,n,a,s=t[0]?fa(t[0]).harness:0,o=s&&s.aliases;if(!o)return e;for(i in r=yt({},e),o)if(i in r)for(n=(a=o[i].split(",")).length;n--;)r[a[n]]=r[i];return r}(f,e)),l=f.length;l--;)if(~c.indexOf(f[l]))for(u in a=d[l],"all"===e?(n[l]=e,o=a,s={}):(s=n[l]=n[l]||{},o=e),o)(h=a&&a[u])&&("kill"in h.d&&!0!==h.d.kill(u)||ya(this,h,"_pt"),delete a[u]),"all"!==s&&(s[u]=1);return this._initted&&!this._pt&&p&&tb(this),this},Tween.to=function to(t,e,r){return new Tween(t,e,r)},Tween.from=function from(t,e){return Va(1,arguments)},Tween.delayedCall=function delayedCall(t,e,r,i){return new Tween(e,0,{immediateRender:!1,lazy:!1,overwrite:!1,delay:t,onComplete:e,onReverseComplete:e,onCompleteParams:r,onReverseCompleteParams:r,callbackScope:i})},Tween.fromTo=function fromTo(t,e,r){return Va(2,arguments)},Tween.set=function set(t,e){return e.duration=0,e.repeatDelay||(e.repeat=0),new Tween(t,e)},Tween.killTweensOf=function killTweensOf(t,e,r){return I.killTweensOf(t,e,r)},Tween}(Ut);qa(Zt.prototype,{_targets:[],_lazy:0,_startAt:0,_op:0,_onInit:0}),ha("staggerTo,staggerFrom,staggerFromTo",function(r){Zt[r]=function(){var t=new Xt,e=Mt.call(arguments,0);return e.splice("staggerFromTo"===r?5:4,0,0),t[r].apply(t,e)}});function oc(t,e,r){return t.setAttribute(e,r)}function wc(t,e,r,i){i.mSet(t,e,i.m.call(i.tween,r,i.mt),i)}var $t=function _setterPlain(t,e,r){return t[e]=r},te=function _setterFunc(t,e,r){return t[e](r)},re=function _setterFuncWithParam(t,e,r,i){return t[e](i.fp,r)},ne=function _getSetter(t,e){return s(t[e])?te:u(t[e])&&t.setAttribute?oc:$t},ae=function _renderPlain(t,e){return e.set(e.t,e.p,Math.round(1e6*(e.s+e.c*t))/1e6,e)},se=function _renderBoolean(t,e){return e.set(e.t,e.p,!!(e.s+e.c*t),e)},ue=function _renderComplexString(t,e){var r=e._pt,i="";if(!t&&e.b)i=e.b;else if(1===t&&e.e)i=e.e;else{for(;r;)i=r.p+(r.m?r.m(r.s+r.c*t):Math.round(1e4*(r.s+r.c*t))/1e4)+i,r=r._next;i+=e.c}e.set(e.t,e.p,i,e)},he=function _renderPropTweens(t,e){for(var r=e._pt;r;)r.r(t,r.d),r=r._next},fe=function _addPluginModifier(t,e,r,i){for(var n,a=this._pt;a;)n=a._next,a.p===i&&a.modifier(t,e,r),a=n},ce=function _killPropTweensOf(t){for(var e,r,i=this._pt;i;)r=i._next,i.p===t&&!i.op||i.op===t?ya(this,i,"_pt"):i.dep||(e=1),i=r;return!e},pe=function _sortPropTweensByPriority(t){for(var e,r,i,n,a=t._pt;a;){for(e=a._next,r=i;r&&r.pr>a.pr;)r=r._next;(a._prev=r?r._prev:n)?a._prev._next=a:i=a,(a._next=r)?r._prev=a:n=a,a=e}t._pt=i},_e=(PropTween.prototype.modifier=function modifier(t,e,r){this.mSet=this.mSet||this.set,this.set=wc,this.m=t,this.mt=r,this.tween=e},PropTween);function PropTween(t,e,r,i,n,a,s,o,u){this.t=e,this.s=i,this.c=n,this.p=r,this.r=a||ae,this.d=s||this,this.set=o||$t,this.pr=u||0,(this._next=t)&&(t._prev=this)}ha(vt+"parent,duration,ease,delay,overwrite,runBackwards,startAt,yoyo,immediateRender,repeat,repeatDelay,data,paused,reversed,lazy,callbackScope,stringFilter,id,yoyoEase,stagger,inherit,repeatRefresh,keyframes,autoRevert,scrollTrigger",function(t){return ft[t]=1}),ot.TweenMax=ot.TweenLite=Zt,ot.TimelineLite=ot.TimelineMax=Xt,I=new Xt({sortChildren:!1,defaults:V,autoRemoveChildren:!0,id:"root",smoothChildTiming:!0}),q.stringFilter=Fb;function Ec(t){return(ye[t]||Te).map(function(t){return t()})}function Fc(){var t=Date.now(),o=[];2":t.push(e.outerHTML),e=e.nextSibling;for(E=t.length;E--;)"&"===t[E]&&t.splice(E,1,"&");return t}function emojiSafeSplit(D,u,F,C){if(D+="",F&&(D=D.trim?D.trim():D.replace(i,"")),u&&""!==u)return D.replace(/>/g,">").replace(/"===e?">":"<"===e?"<":!C||" "!==e||" "!==D.charAt(B-1)&&" "!==D.charAt(B+1)?e:" ");return t}var u,g,F={version:"3.12.2",name:"text",init:function init(D,u,F){"object"!=typeof u&&(u={value:u});var C,E,e,t,n,B,i,r,A=D.nodeName.toUpperCase(),s=this,l=u.newClass,o=u.oldClass,a=u.preserveSpaces,p=u.rtl,f=s.delimiter=u.delimiter||"",d=s.fillChar=u.fillChar||(u.padSpace?" ":"");if(s.svg=D.getBBox&&("TEXT"===A||"TSPAN"===A),!("innerHTML"in D||s.svg))return!1;if(s.target=D,"value"in u){for(e=splitInnerHTML(D,f,!1,a),(g=g||document.createElement("div")).innerHTML=u.value,E=splitInnerHTML(g,f,!1,a),s.from=F._from,!s.from&&!p||p&&s.from||(A=e,e=E,E=A),s.hasClass=!(!l&&!o),s.newClass=p?o:l,s.oldClass=p?l:o,C=(A=e.length-E.length)<0?e:E,A<0&&(A=-A);-1<--A;)C.push(d);if("diff"===u.type){for(n=[],B=[],i="",A=t=0;A":"")+e.slice(0,a).join(i)+(F?"":"")+(C?"":"")+i+s.slice(a).join(i)+(C?"":"")):e.slice(0,a).join(i)+i+s.slice(a).join(i),u.svg?r.textContent=E:r.innerHTML=" "===A&&~E.indexOf(" ")?E.split(" ").join("  "):E}};F.splitInnerHTML=splitInnerHTML,F.emojiSafeSplit=emojiSafeSplit,F.getText=function getText(D){var u=D.nodeType,F="";if(1===u||9===u||11===u){if("string"==typeof D.textContent)return D.textContent;for(D=D.firstChild;D;D=D.nextSibling)F+=getText(D)}else if(3===u||4===u)return D.nodeValue;return F},function _getGSAP(){return u||"undefined"!=typeof window&&(u=window.gsap)&&u.registerPlugin&&u}()&&u.registerPlugin(F),D.TextPlugin=F,D.default=F;if (typeof(window)==="undefined"||window!==D){Object.defineProperty(D,"__esModule",{value:!0})} else {delete D.default}}); + diff --git a/electron/watchdogService.cjs b/electron/watchdogService.cjs new file mode 100644 index 00000000..8953dd8e --- /dev/null +++ b/electron/watchdogService.cjs @@ -0,0 +1,1219 @@ +const { EventEmitter } = require('events'); +const { Notification } = require('electron'); +const log = require('electron-log'); +const { getAdaptiveHealthCheckManager } = require('./adaptiveHealthCheckManager.cjs'); + +class WatchdogService extends EventEmitter { + constructor(dockerSetup, mcpService, ipcLogger = null) { + super(); + this.dockerSetup = dockerSetup; + this.mcpService = mcpService; + this.ipcLogger = ipcLogger; + + // Get adaptive health check manager + this.adaptiveHealthManager = getAdaptiveHealthCheckManager(); + + // Professional logging configuration + this.logger = this.initializeLogger(); + this.sessionId = this.generateSessionId(); + this.serviceStates = new Map(); // Track service state changes + + // Logging configuration for enterprise-grade output + this.loggingConfig = { + level: 'INFO', // DEBUG, INFO, WARN, ERROR, FATAL + logOnlyStateChanges: true, + enableMetrics: true, + structuredOutput: true, + adminFriendly: true + }; + + // Service state tracking for change detection + this.lastKnownStates = new Map(); + this.lastLoggedStates = new Map(); + this.serviceMetrics = new Map(); + + // Get user's feature selections + const selectedFeatures = global.selectedFeatures || { + comfyUI: true, + n8n: true, + ragAndTts: true, + claraCore: true + }; + + // Watchdog configuration (now adaptive) + this.config = { + baseCheckInterval: 30000, // Base: 30 seconds when active + startupDelay: 60000, // Wait 60 seconds before starting health checks after startup + retryAttempts: 3, + retryDelay: 10000, // 10 seconds between retries + notificationTimeout: 5000, // Auto-dismiss notifications after 5 seconds + maxNotificationAttempts: 3, // Stop showing notifications after this many attempts + gracePeriod: 30 * 60 * 1000, // 30 minutes grace period after service is confirmed healthy + useAdaptiveChecks: true // Enable adaptive health checking + }; + + // Service status tracking - only include selected services + this.services = {}; + + // Clara Core monitoring disabled - llamaSwapService has been removed + // Keeping the structure commented for future reference + /* + this.services.clarasCore = { + name: "Clara's Core", + status: 'unknown', + lastCheck: null, + lastHealthyTime: null, + failureCount: 0, + isRetrying: false, + enabled: false, // Disabled - service removed + healthCheck: () => this.checkClarasCoreHealth(), + restart: () => this.restartClarasCore() + }; + */ + + // Python backend is always enabled (core service) + this.services.python = { + name: 'Python Backend Service', + status: 'unknown', + lastCheck: null, + lastHealthyTime: null, // Track when service was last confirmed healthy + failureCount: 0, + isRetrying: false, + enabled: selectedFeatures.ragAndTts, // Only if RAG & TTS is selected + healthCheck: () => this.checkPythonHealth(), + restart: () => this.restartPythonService() + }; + + // N8N only if selected + if (selectedFeatures.n8n) { + this.services.n8n = { + name: 'n8n Workflow Engine', + status: 'unknown', + lastCheck: null, + lastHealthyTime: null, // Track when service was last confirmed healthy + failureCount: 0, + isRetrying: false, + enabled: true, + healthCheck: () => this.checkN8nHealth(), + restart: () => this.restartN8nService() + }; + } + + // ComfyUI only if selected + if (selectedFeatures.comfyUI) { + this.services.comfyui = { + name: 'ComfyUI Image Generation', + status: 'unknown', + lastCheck: null, + lastHealthyTime: null, // Track when service was last confirmed healthy + failureCount: 0, + isRetrying: false, + enabled: true, // Will be updated based on user consent + healthCheck: () => this.checkComfyUIHealth(), + restart: () => this.restartComfyUIService() + }; + } + + // Watchdog state + this.isRunning = false; + this.isStarting = false; + this.checkTimer = null; + this.startupTimer = null; + this.activeNotifications = new Map(); + + this.logEvent('WATCHDOG_INIT', 'INFO', 'Watchdog Service initialized', { + selectedFeatures, + sessionId: this.sessionId, + servicesCount: Object.keys(this.services).length + }); + } + + // Initialize professional logging system + initializeLogger() { + const logger = { + format: (level, operation, message, metadata = {}) => { + const timestamp = new Date().toISOString(); + const context = { + timestamp, + sessionId: this.sessionId, + component: 'WatchdogService', + level, + operation, + message, + ...metadata + }; + + if (this.loggingConfig.structuredOutput) { + return JSON.stringify(context); + } else { + // Human-readable format for admins + const metaStr = Object.keys(metadata).length > 0 ? + ` | ${JSON.stringify(metadata)}` : ''; + return `[${timestamp}] [WATCHDOG:${level}] [${operation}] ${message}${metaStr}`; + } + } + }; + return logger; + } + + // Generate unique session ID for tracking + generateSessionId() { + return `wd_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`; + } + + // Professional logging method + logEvent(operation, level, message, metadata = {}) { + const logLevel = this.loggingConfig.level; + const levelPriority = { DEBUG: 0, INFO: 1, WARN: 2, ERROR: 3, FATAL: 4 }; + + // Only log if meets minimum level + if (levelPriority[level] < levelPriority[logLevel]) { + return; + } + + const formattedLog = this.logger.format(level, operation, message, metadata); + + // Route to appropriate log level + switch (level) { + case 'DEBUG': + log.debug(formattedLog); + break; + case 'INFO': + log.info(formattedLog); + break; + case 'WARN': + log.warn(formattedLog); + break; + case 'ERROR': + case 'FATAL': + log.error(formattedLog); + break; + } + + // Send to IPC logger for critical events + if (this.ipcLogger && (level === 'ERROR' || level === 'FATAL' || level === 'WARN')) { + this.ipcLogger.logWatchdogEvent(operation, level, { message, ...metadata }); + } + } + + // Track service state changes with metrics + trackServiceStateChange(serviceKey, oldState, newState, metadata = {}) { + const timestamp = Date.now(); + const stateChange = { + serviceKey, + oldState, + newState, + timestamp, + ...metadata + }; + + // Update service metrics + if (!this.serviceMetrics.has(serviceKey)) { + this.serviceMetrics.set(serviceKey, { + stateChanges: 0, + totalDowntime: 0, + lastHealthyTime: null, + restartCount: 0 + }); + } + + const metrics = this.serviceMetrics.get(serviceKey); + metrics.stateChanges++; + + // Calculate downtime if recovering + if (oldState !== 'healthy' && newState === 'healthy' && metrics.lastHealthyTime) { + const downtime = timestamp - metrics.lastHealthyTime; + metrics.totalDowntime += downtime; + } + + if (newState === 'healthy') { + metrics.lastHealthyTime = timestamp; + } + + this.serviceMetrics.set(serviceKey, metrics); + this.lastKnownStates.set(serviceKey, newState); + + return stateChange; + } + + // Start the watchdog monitoring + start() { + if (this.isRunning) { + this.logEvent('WATCHDOG_START', 'WARN', 'Attempted to start already running watchdog service'); + return; + } + + // Check for user consent before starting any service monitoring + if (!this.checkUserConsent()) { + this.logEvent('WATCHDOG_START', 'INFO', 'User consent not found, watchdog service will not monitor services'); + return; + } + + this.isRunning = true; + this.isStarting = true; + + this.logEvent('WATCHDOG_START', 'INFO', 'Starting Watchdog Service (health check errors for stopped services are suppressed)', { + checkInterval: this.config.checkInterval, + startupDelay: this.config.startupDelay, + gracePeriod: this.config.gracePeriod, + note: 'ECONNREFUSED errors are expected when services are not running and will not be logged' + }); + + // Set enabled services to "starting" state during startup (after consent checks) + for (const [serviceKey, service] of Object.entries(this.services)) { + if (service.enabled) { + const oldState = service.status; + service.status = 'starting'; + this.trackServiceStateChange(serviceKey, oldState, 'starting', { reason: 'watchdog_startup' }); + } else { + // Ensure disabled services remain disabled + service.status = 'disabled'; + } + } + + // Wait for startup delay before beginning health checks + this.logEvent('WATCHDOG_STARTUP_DELAY', 'INFO', 'Waiting for startup delay before health checks', { + delaySeconds: this.config.startupDelay / 1000 + }); + + this.startupTimer = setTimeout(() => { + this.isStarting = false; + this.logEvent('WATCHDOG_HEALTH_CHECKS_BEGIN', 'INFO', 'Startup delay complete, beginning adaptive health checks'); + + // Start adaptive health monitoring + if (this.config.useAdaptiveChecks) { + this.adaptiveHealthManager.startMonitoring(); + this.logEvent('ADAPTIVE_MONITORING', 'INFO', 'Adaptive health monitoring enabled', + this.adaptiveHealthManager.getStatus()); + } + + // Perform initial health checks + this.performHealthChecks(); + + // Schedule adaptive health checks + this.scheduleNextHealthCheck(); + + }, this.config.startupDelay); + + this.emit('started'); + this.logEvent('WATCHDOG_STARTED', 'INFO', 'Watchdog Service started successfully'); + } + + checkUserConsent() { + const fs = require('fs'); + const path = require('path'); + const { app } = require('electron'); + + try { + const userDataPath = app.getPath('userData'); + const consentFile = path.join(userDataPath, 'user-service-consent.json'); + + if (fs.existsSync(consentFile)) { + const consentData = JSON.parse(fs.readFileSync(consentFile, 'utf8')); + this.logEvent('USER_CONSENT_CHECK', 'INFO', 'User service consent status loaded', { + hasConsented: consentData.hasConsented, + servicesConsented: consentData.services, + onboardingMode: consentData.onboardingMode, + autoStartServices: consentData.autoStartServices, + consentDate: consentData.timestamp + }); + + // Check if this is from onboarding mode with auto-start disabled + const isOnboardingMode = consentData.onboardingMode === true; + const autoStartDisabled = consentData.autoStartServices === false; + + if (isOnboardingMode && autoStartDisabled) { + this.logEvent('ONBOARDING_MODE_DETECTED', 'INFO', 'Onboarding mode with auto-start disabled - only Clara Core will be managed'); + + // In onboarding mode with auto-start disabled: + // - Clara Core is always managed (it's essential) + // - Other services are disabled for auto-management but user preferences are stored + for (const [serviceKey, service] of Object.entries(this.services)) { + if (serviceKey === 'clarasCore') { + // Always enable Clara Core monitoring (using correct service key) + service.enabled = true; + this.logEvent('SERVICE_CLARA_CORE_ENABLED', 'INFO', 'Clara Core monitoring enabled - essential service'); + } else { + // Disable auto-management for other services during onboarding mode + service.enabled = false; + service.status = 'disabled'; + this.logEvent('SERVICE_ONBOARDING_DISABLED', 'INFO', `Service ${serviceKey} disabled - onboarding mode with auto-start off`); + } + } + + return true; // Return true so watchdog runs, but only for Clara Core + } + + // Check if user has auto-start enabled for post-onboarding use + let autoStartEnabled = false; + try { + // Since we can't easily access the frontend db from main process, + // and the default is false (which is what we want for security), + // we'll default to false unless explicitly set via a dedicated file + + // Check if autoStartServices is explicitly set to true (for future use) + autoStartEnabled = consentData.autoStartServices === true; + } catch (error) { + this.logEvent('AUTO_START_CHECK', 'WARN', 'Could not check auto-start preference', { + error: error.message + }); + } + + if (!autoStartEnabled && !isOnboardingMode) { + this.logEvent('AUTO_START_DISABLED', 'INFO', 'Auto-start disabled - watchdog will not monitor services'); + // Disable all services if auto-start is disabled + for (const [serviceKey, service] of Object.entries(this.services)) { + service.enabled = false; + service.status = 'disabled'; + } + return false; + } + + // Only enable services that user has explicitly consented to (for full auto-start mode) + if (consentData.hasConsented && consentData.services && autoStartEnabled) { + for (const [serviceKey, service] of Object.entries(this.services)) { + if (serviceKey === 'clarasCore') { + // Clara Core is always enabled when user has consented + service.enabled = true; + this.logEvent('SERVICE_CLARA_CORE_ENABLED', 'INFO', 'Clara Core monitoring enabled - essential service'); + } else if (consentData.services[serviceKey] === true) { + service.enabled = true; + this.logEvent('SERVICE_CONSENT_ENABLED', 'INFO', `Service ${serviceKey} enabled by user consent`); + } else { + service.enabled = false; + service.status = 'disabled'; + this.logEvent('SERVICE_CONSENT_DISABLED', 'INFO', `Service ${serviceKey} disabled - no user consent`); + } + } + } + + return consentData.hasConsented === true; + } else { + this.logEvent('USER_CONSENT_CHECK', 'INFO', 'No user consent file found - only Clara Core will be managed'); + + // Without consent file, only enable Clara Core (essential service) + for (const [serviceKey, service] of Object.entries(this.services)) { + if (serviceKey === 'clarasCore') { + service.enabled = true; + this.logEvent('SERVICE_CLARA_CORE_ENABLED', 'INFO', 'Clara Core monitoring enabled - essential service (no consent file)'); + } else { + service.enabled = false; + service.status = 'disabled'; + } + } + + return true; // Return true so watchdog runs for Clara Core + } + } catch (error) { + this.logEvent('USER_CONSENT_ERROR', 'ERROR', 'Failed to read user consent status - only Clara Core will be managed', { + error: error.message + }); + + // On error, only enable Clara Core (essential service) + for (const [serviceKey, service] of Object.entries(this.services)) { + if (serviceKey === 'clarasCore') { + service.enabled = true; + this.logEvent('SERVICE_CLARA_CORE_ENABLED', 'INFO', 'Clara Core monitoring enabled - essential service (error fallback)'); + } else { + service.enabled = false; + service.status = 'disabled'; + } + } + + return true; // Return true so watchdog runs for Clara Core + } + } + + // Stop the watchdog monitoring + stop() { + if (!this.isRunning) { + this.logEvent('WATCHDOG_STOP', 'WARN', 'Attempted to stop already stopped watchdog service'); + return; + } + + this.isRunning = false; + this.isStarting = false; + + this.logEvent('WATCHDOG_STOP', 'INFO', 'Stopping Watchdog Service'); + + // Stop adaptive monitoring + if (this.adaptiveHealthManager) { + this.adaptiveHealthManager.stopMonitoring(); + } + + if (this.checkTimer) { + clearTimeout(this.checkTimer); // Changed to clearTimeout for adaptive scheduling + this.checkTimer = null; + } + + if (this.startupTimer) { + clearTimeout(this.startupTimer); + this.startupTimer = null; + } + + // Clear any active notifications + this.activeNotifications.forEach((notification, id) => { + notification.close(); + }); + this.activeNotifications.clear(); + + this.emit('stopped'); + this.logEvent('WATCHDOG_STOPPED', 'INFO', 'Watchdog Service stopped successfully'); + } + + // Check if a service is within its grace period + isServiceInGracePeriod(service) { + if (!service.lastHealthyTime) { + return false; // No grace period if never been healthy + } + + const timeSinceHealthy = Date.now() - service.lastHealthyTime; + const inGracePeriod = timeSinceHealthy < this.config.gracePeriod; + + if (inGracePeriod) { + // Only log once when grace period starts + if (!service.gracePeriodLogged) { + this.logEvent('SERVICE_GRACE_PERIOD', 'DEBUG', `${service.name} is in grace period`, { + timeRemainingMs: this.config.gracePeriod - timeSinceHealthy, + timeRemainingMin: Math.round((this.config.gracePeriod - timeSinceHealthy) / 60000) + }); + service.gracePeriodLogged = true; + } + } else { + service.gracePeriodLogged = false; + } + + return inGracePeriod; + } + + // Schedule next health check with adaptive interval + scheduleNextHealthCheck() { + if (!this.isRunning || this.isStarting) { + return; + } + + // Get adaptive interval + let nextInterval = this.config.baseCheckInterval; + + if (this.config.useAdaptiveChecks && this.adaptiveHealthManager) { + // Use the most conservative interval (longest) from all services + for (const serviceKey of Object.keys(this.services)) { + const serviceInterval = this.adaptiveHealthManager.getHealthCheckInterval( + serviceKey, + this.config.baseCheckInterval + ); + nextInterval = Math.max(nextInterval, serviceInterval); + } + + // Log adaptive interval changes + const status = this.adaptiveHealthManager.getStatus(); + if (status.mode !== 'ACTIVE') { + this.logEvent('ADAPTIVE_INTERVAL', 'DEBUG', `Next health check in ${nextInterval / 1000}s`, { + mode: status.mode, + minutesIdle: status.minutesSinceActivity, + onBattery: status.isOnBattery + }); + } + } + + // Clear existing timer + if (this.checkTimer) { + clearTimeout(this.checkTimer); + } + + // Schedule next check + this.checkTimer = setTimeout(() => { + this.performHealthChecks(); + this.scheduleNextHealthCheck(); // Schedule next one after this completes + }, nextInterval); + } + + // Perform health checks on all services + async performHealthChecks() { + // Skip health checks during startup phase + if (this.isStarting) { + return; + } + + const timestamp = new Date(); + let healthCheckSummary = { + totalServices: 0, + checkedServices: 0, + healthyServices: 0, + unhealthyServices: 0, + skippedServices: 0, + stateChanges: [] + }; + + for (const [serviceKey, service] of Object.entries(this.services)) { + healthCheckSummary.totalServices++; + + // Skip disabled services + if (!service.enabled) { + healthCheckSummary.skippedServices++; + continue; + } + + // Skip services that are in grace period (recently confirmed healthy) + if (this.isServiceInGracePeriod(service)) { + healthCheckSummary.skippedServices++; + continue; + } + + // Skip services during deep idle if adaptive checking is enabled + if (this.config.useAdaptiveChecks && + this.adaptiveHealthManager && + this.adaptiveHealthManager.shouldSkipHealthCheck(serviceKey)) { + healthCheckSummary.skippedServices++; + this.logEvent('HEALTH_CHECK_SKIPPED', 'DEBUG', `Skipping ${serviceKey} - deep idle mode`); + continue; + } + + healthCheckSummary.checkedServices++; + const previousStatus = service.status; + + try { + const isHealthy = await service.healthCheck(); + service.lastCheck = timestamp; + + if (isHealthy) { + // Record service activity for adaptive checking + if (this.adaptiveHealthManager) { + this.adaptiveHealthManager.recordServiceActivity(serviceKey); + } + + if (service.status !== 'healthy') { + // Service recovered - important state change + const stateChange = this.trackServiceStateChange(serviceKey, service.status, 'healthy', { + reason: 'health_check_recovery', + failureCount: service.failureCount + }); + healthCheckSummary.stateChanges.push(stateChange); + + service.status = 'healthy'; + service.lastHealthyTime = Date.now(); + service.failureCount = 0; + service.isRetrying = false; + + this.logEvent('SERVICE_RECOVERY', 'INFO', `${service.name} has recovered and is now healthy`, { + previousStatus, + downtimeMs: stateChange.downtimeMs || 0, + failureCount: service.failureCount + }); + + this.emit('serviceRestored', serviceKey, service); + healthCheckSummary.healthyServices++; + } else { + // Service was already healthy - just refresh grace period silently + service.lastHealthyTime = Date.now(); + healthCheckSummary.healthyServices++; + } + } else { + // Service is unhealthy + const wasHealthy = service.status === 'healthy'; + service.lastHealthyTime = null; + service.failureCount++; + + if (wasHealthy) { + // New failure - log immediately + const stateChange = this.trackServiceStateChange(serviceKey, 'healthy', 'degraded', { + reason: 'health_check_failure', + failureCount: service.failureCount + }); + healthCheckSummary.stateChanges.push(stateChange); + + this.logEvent('SERVICE_DEGRADED', 'WARN', `${service.name} health check failed - service degraded`, { + failureCount: service.failureCount, + maxRetries: this.config.retryAttempts + }); + } else if (service.failureCount === this.config.retryAttempts && !service.isRetrying) { + // Critical failure threshold reached + const stateChange = this.trackServiceStateChange(serviceKey, service.status, 'failed', { + reason: 'critical_failure', + failureCount: service.failureCount + }); + healthCheckSummary.stateChanges.push(stateChange); + + service.status = 'failed'; + service.isRetrying = true; + + this.logEvent('SERVICE_CRITICAL_FAILURE', 'ERROR', `${service.name} has failed critically - initiating restart`, { + failureCount: service.failureCount, + maxRetries: this.config.retryAttempts + }); + + // Attempt restart in background + this.attemptServiceRestart(serviceKey, service); + } else { + service.status = 'degraded'; + } + + healthCheckSummary.unhealthyServices++; + } + } catch (error) { + // Health check errors are always critical + const stateChange = this.trackServiceStateChange(serviceKey, service.status, 'error', { + reason: 'health_check_error', + error: error.message + }); + healthCheckSummary.stateChanges.push(stateChange); + + service.status = 'error'; + service.lastCheck = timestamp; + service.lastHealthyTime = null; + + this.logEvent('SERVICE_ERROR', 'ERROR', `${service.name} health check encountered an error`, { + error: error.message, + previousStatus, + stack: error.stack + }); + + healthCheckSummary.unhealthyServices++; + } + } + + // Only log health check summary if there were state changes or if in debug mode + if (healthCheckSummary.stateChanges.length > 0 || this.loggingConfig.level === 'DEBUG') { + this.logEvent('HEALTH_CHECK_COMPLETE', + healthCheckSummary.stateChanges.length > 0 ? 'INFO' : 'DEBUG', + 'Health check cycle completed', + healthCheckSummary + ); + } + } + + // Separate restart logic with proper logging + async attemptServiceRestart(serviceKey, service) { + this.logEvent('SERVICE_RESTART_ATTEMPT', 'INFO', `Attempting to restart ${service.name}`, { + failureCount: service.failureCount, + previousStatus: service.status + }); + + try { + await service.restart(); + + // Give service time to start before checking health + setTimeout(async () => { + try { + const restartHealthy = await service.healthCheck(); + if (restartHealthy) { + const stateChange = this.trackServiceStateChange(serviceKey, 'failed', 'healthy', { + reason: 'restart_success' + }); + + service.status = 'healthy'; + service.lastHealthyTime = Date.now(); + service.failureCount = 0; + service.isRetrying = false; + + this.logEvent('SERVICE_RESTART_SUCCESS', 'INFO', `${service.name} restart completed successfully`, { + downtimeMs: stateChange.downtimeMs || 0 + }); + + this.emit('serviceRestarted', serviceKey, service); + this.showRestartSuccessNotification(service); + } else { + service.isRetrying = false; + service.lastHealthyTime = null; + + this.logEvent('SERVICE_RESTART_FAILED', 'ERROR', `${service.name} restart completed but service is still unhealthy`); + + this.emit('serviceFailed', serviceKey, service); + this.showRestartFailureNotification(service); + } + } catch (healthCheckError) { + service.isRetrying = false; + service.lastHealthyTime = null; + + this.logEvent('SERVICE_RESTART_HEALTH_CHECK_ERROR', 'ERROR', `${service.name} restart health check failed`, { + error: healthCheckError.message + }); + + this.emit('serviceFailed', serviceKey, service); + this.showRestartFailureNotification(service); + } + }, this.config.retryDelay); + + } catch (restartError) { + service.isRetrying = false; + service.lastHealthyTime = null; + + this.logEvent('SERVICE_RESTART_ERROR', 'ERROR', `${service.name} restart operation failed`, { + error: restartError.message, + stack: restartError.stack + }); + + this.emit('serviceFailed', serviceKey, service); + this.showRestartFailureNotification(service); + } + } + + // Individual service health check methods + async checkClarasCoreHealth() { + // Clara's Core service (llamaSwapService) has been removed + // This method is kept for compatibility but always returns true + this.logEvent('SERVICE_HEALTH_CHECK', 'DEBUG', 'Clara\'s Core health check skipped (service removed)'); + return true; + } + + async checkN8nHealth() { + try { + const result = await this.dockerSetup.checkN8NHealth(); + + this.logEvent('SERVICE_HEALTH_CHECK', 'DEBUG', 'n8n health check completed', { + success: result.success, + details: result + }); + + return result.success === true; + } catch (error) { + this.logEvent('SERVICE_HEALTH_CHECK_ERROR', 'ERROR', 'n8n health check failed', { + error: error.message, + stack: error.stack + }); + return false; + } + } + + async checkPythonHealth() { + try { + const isRunning = await this.dockerSetup.isPythonRunning(); + + this.logEvent('SERVICE_HEALTH_CHECK', 'DEBUG', 'Python Backend health check completed', { + isRunning: isRunning + }); + + return isRunning; + } catch (error) { + this.logEvent('SERVICE_HEALTH_CHECK_ERROR', 'ERROR', 'Python Backend health check failed', { + error: error.message, + stack: error.stack + }); + return false; + } + } + + async checkComfyUIHealth() { + try { + const isRunning = await this.dockerSetup.isComfyUIRunning(); + + this.logEvent('SERVICE_HEALTH_CHECK', 'DEBUG', 'ComfyUI health check completed', { + isRunning: isRunning + }); + + return isRunning; + } catch (error) { + this.logEvent('SERVICE_HEALTH_CHECK_ERROR', 'ERROR', 'ComfyUI health check failed', { + error: error.message, + stack: error.stack + }); + return false; + } + } + + // Individual service restart methods + async restartClarasCore() { + // Clara's Core service (llamaSwapService) has been removed + // This method is kept for compatibility but does nothing + this.logEvent('SERVICE_RESTART', 'INFO', 'Clara\'s Core restart skipped (service removed)'); + return; + } + + async restartN8nService() { + this.logEvent('SERVICE_RESTART', 'INFO', 'Initiating n8n service restart'); + + try { + let n8nConfig = this.dockerSetup.containers.n8n; + + // If N8N config is not available, create it + if (!n8nConfig) { + this.logEvent('SERVICE_RESTART_WARNING', 'WARN', 'N8N configuration not found, creating default configuration'); + n8nConfig = { + name: 'clara_n8n', + image: this.dockerSetup.getArchSpecificImage('n8nio/n8n', 'latest'), + port: 5678, + internalPort: 5678, + healthCheck: this.dockerSetup.checkN8NHealth.bind(this.dockerSetup), + volumes: [ + `${require('path').join(require('os').homedir(), '.clara', 'n8n')}:/home/node/.n8n` + ] + }; + this.dockerSetup.containers.n8n = n8nConfig; + } + + await this.dockerSetup.startContainer(n8nConfig); + this.logEvent('SERVICE_RESTART_OPERATION', 'INFO', 'n8n restart operation completed'); + } catch (error) { + this.logEvent('SERVICE_RESTART_ERROR', 'ERROR', 'n8n restart operation failed', { + error: error.message, + stack: error.stack + }); + throw error; + } + } + + async restartPythonService() { + this.logEvent('SERVICE_RESTART', 'INFO', 'Initiating Python Backend service restart'); + + try { + let pythonConfig = this.dockerSetup.containers.python; + + // If Python config is not available, create it + if (!pythonConfig) { + this.logEvent('SERVICE_RESTART_WARNING', 'WARN', 'Python configuration not found, creating default configuration'); + pythonConfig = { + name: 'clara_python', + image: this.dockerSetup.getArchSpecificImage('clara17verse/clara-backend', 'latest'), + // On Linux (host network mode), use port 5000. On Windows/Mac (bridge mode), use port 5001 + port: process.platform === 'linux' ? 5000 : 5001, + internalPort: 5000, + healthCheck: this.dockerSetup.isPythonRunning.bind(this.dockerSetup), + volumes: [ + `${this.dockerSetup.pythonBackendDataPath}:/home/clara`, + 'clara_python_models:/app/models' + ], + volumeNames: ['clara_python_models'] + }; + this.dockerSetup.containers.python = pythonConfig; + } + + await this.dockerSetup.startContainer(pythonConfig); + this.logEvent('SERVICE_RESTART_OPERATION', 'INFO', 'Python Backend restart operation completed'); + } catch (error) { + this.logEvent('SERVICE_RESTART_ERROR', 'ERROR', 'Python Backend restart operation failed', { + error: error.message, + stack: error.stack + }); + throw error; + } + } + + async restartComfyUIService() { + this.logEvent('SERVICE_RESTART', 'INFO', 'Initiating ComfyUI service restart'); + + try { + if (this.dockerSetup.containers.comfyui) { + await this.dockerSetup.startContainer(this.dockerSetup.containers.comfyui); + this.logEvent('SERVICE_RESTART_OPERATION', 'INFO', 'ComfyUI restart operation completed'); + } + } catch (error) { + this.logEvent('SERVICE_RESTART_ERROR', 'ERROR', 'ComfyUI restart operation failed', { + error: error.message, + stack: error.stack + }); + throw error; + } + } + + // Notification methods for restart success/failure + showRestartSuccessNotification(service) { + this.logEvent('NOTIFICATION_SENT', 'INFO', `Showing restart success notification for ${service.name}`); + this.showNotification( + `${service.name} Restarted`, + `${service.name} has been successfully restarted and is now healthy.`, + 'success' + ); + } + + showRestartFailureNotification(service) { + // Only show failure notifications for the first few attempts to avoid spam + if (service.failureCount <= this.config.maxNotificationAttempts) { + this.logEvent('NOTIFICATION_SENT', 'WARN', `Showing restart failure notification for ${service.name}`, { + failureCount: service.failureCount, + maxAttempts: this.config.maxNotificationAttempts + }); + this.showNotification( + `${service.name} Restart Failed`, + `Failed to restart ${service.name}. Manual intervention may be required.`, + 'error' + ); + } else { + this.logEvent('NOTIFICATION_SUPPRESSED', 'INFO', `Restart failure notification suppressed for ${service.name}`, { + failureCount: service.failureCount, + maxAttempts: this.config.maxNotificationAttempts, + reason: 'spam_prevention' + }); + } + } + + // Show non-persistent notification + showNotification(title, body, type = 'info') { + try { + const notification = new Notification({ + title, + body, + icon: this.getNotificationIcon(type), + silent: false, + urgency: type === 'error' ? 'critical' : type === 'warning' ? 'normal' : 'low' + }); + + // Auto-dismiss notification after configured timeout + const notificationId = Date.now().toString(); + this.activeNotifications.set(notificationId, notification); + + notification.show(); + + notification.on('click', () => { + notification.close(); + this.activeNotifications.delete(notificationId); + }); + + notification.on('close', () => { + this.activeNotifications.delete(notificationId); + }); + + // Auto-close after timeout + setTimeout(() => { + if (this.activeNotifications.has(notificationId)) { + notification.close(); + this.activeNotifications.delete(notificationId); + } + }, this.config.notificationTimeout); + + log.info(`Notification shown: ${title} - ${body}`); + } catch (error) { + log.error('Error showing notification:', error); + if (this.ipcLogger) { + this.ipcLogger.logError('WatchdogService.showNotification', error); + } + } + } + + // Get appropriate icon for notification type + getNotificationIcon(type) { + // You can customize these paths to your app's icon files + switch (type) { + case 'success': + return null; // Use default app icon + case 'warning': + return null; // Use default app icon + case 'error': + return null; // Use default app icon + default: + return null; // Use default app icon + } + } + + // Get current status of all services + getServicesStatus() { + const status = {}; + for (const [key, service] of Object.entries(this.services)) { + const inGracePeriod = this.isServiceInGracePeriod(service); + const gracePeriodRemainingMinutes = service.lastHealthyTime + ? Math.max(0, Math.ceil((this.config.gracePeriod - (Date.now() - service.lastHealthyTime)) / (60 * 1000))) + : 0; + + status[key] = { + name: service.name, + status: service.status, + lastCheck: service.lastCheck, + lastHealthyTime: service.lastHealthyTime, + failureCount: service.failureCount, + isRetrying: service.isRetrying, + inGracePeriod: inGracePeriod, + gracePeriodRemainingMinutes: gracePeriodRemainingMinutes + }; + } + return status; + } + + // Get overall system health + getOverallHealth() { + const statuses = Object.values(this.services).map(service => service.status); + const healthyCount = statuses.filter(status => status === 'healthy').length; + const totalCount = statuses.length; + + if (healthyCount === totalCount) { + return 'healthy'; + } else if (healthyCount === 0) { + return 'critical'; + } else { + return 'degraded'; + } + } + + // Update configuration + updateConfig(newConfig) { + this.config = { ...this.config, ...newConfig }; + log.info('Watchdog configuration updated:', this.config); + + // Restart monitoring with new interval if it changed and service is running + if (this.isRunning && newConfig.checkInterval) { + this.stop(); + this.start(); + } + } + + // Signal that Docker setup is complete and watchdog can start monitoring + signalSetupComplete() { + if (!this.isRunning || !this.isStarting) { + return; + } + + log.info('Docker setup complete signal received, starting health checks early...'); + + // Clear the startup timer if it's still running + if (this.startupTimer) { + clearTimeout(this.startupTimer); + this.startupTimer = null; + } + + this.isStarting = false; + + // Perform initial health checks + this.performHealthChecks(); + + // Schedule regular health checks + this.checkTimer = setInterval(() => { + this.performHealthChecks(); + }, this.config.checkInterval); + + log.info('Watchdog health checks started early due to setup completion'); + } + + // Manual health check trigger + async performManualHealthCheck() { + this.logEvent('MANUAL_HEALTH_CHECK', 'INFO', 'Manual health check triggered by admin'); + await this.performHealthChecks(); + const status = this.getServicesStatus(); + this.logEvent('MANUAL_HEALTH_CHECK_COMPLETE', 'INFO', 'Manual health check completed', status); + return status; + } + + // Reset failure counts for all services + resetFailureCounts() { + for (const [serviceKey, service] of Object.entries(this.services)) { + const oldFailureCount = service.failureCount; + service.failureCount = 0; + service.isRetrying = false; + // Keep grace period intact when resetting failure counts + } + this.logEvent('FAILURE_COUNTS_RESET', 'INFO', 'All service failure counts reset by admin', { + preservedGracePeriods: true + }); + } + + // Enable monitoring for a specific service (useful after onboarding) + enableServiceMonitoring(serviceKey) { + if (this.services[serviceKey]) { + this.services[serviceKey].enabled = true; + this.services[serviceKey].status = 'unknown'; // Reset status to trigger fresh check + this.logEvent('SERVICE_MONITORING_ENABLED', 'INFO', `Monitoring enabled for service: ${serviceKey}`); + + // Trigger immediate health check for this service + if (this.isRunning) { + setTimeout(() => { + this.performHealthChecks(); + }, 1000); + } + + return true; + } else { + this.logEvent('SERVICE_MONITORING_ERROR', 'ERROR', `Cannot enable monitoring - service not found: ${serviceKey}`); + return false; + } + } + + // Disable monitoring for a specific service + disableServiceMonitoring(serviceKey) { + if (this.services[serviceKey]) { + this.services[serviceKey].enabled = false; + this.services[serviceKey].status = 'disabled'; + this.logEvent('SERVICE_MONITORING_DISABLED', 'INFO', `Monitoring disabled for service: ${serviceKey}`); + return true; + } else { + this.logEvent('SERVICE_MONITORING_ERROR', 'ERROR', `Cannot disable monitoring - service not found: ${serviceKey}`); + return false; + } + } + + // Get list of services and their monitoring status + getServiceMonitoringStatus() { + const status = {}; + for (const [serviceKey, service] of Object.entries(this.services)) { + status[serviceKey] = { + name: service.name, + enabled: service.enabled, + status: service.status, + lastCheck: service.lastCheck, + lastHealthyTime: service.lastHealthyTime, + failureCount: service.failureCount + }; + } + return status; + } + + // Force end grace period for a specific service (for manual intervention) + endGracePeriod(serviceKey) { + if (this.services[serviceKey]) { + this.services[serviceKey].lastHealthyTime = null; + this.logEvent('GRACE_PERIOD_ENDED', 'INFO', `Grace period manually ended for ${this.services[serviceKey].name}`, { + serviceKey: serviceKey, + reason: 'admin_intervention' + }); + } + } + + // Force end grace period for all services + endAllGracePeriods() { + const endedServices = []; + for (const [serviceKey, service] of Object.entries(this.services)) { + if (service.lastHealthyTime) { + service.lastHealthyTime = null; + endedServices.push(service.name); + } + } + this.logEvent('ALL_GRACE_PERIODS_ENDED', 'INFO', 'All service grace periods manually ended by admin', { + affectedServices: endedServices, + reason: 'admin_intervention' + }); + } + + // Enable verbose logging for debugging + enableVerboseLogging() { + this.loggingConfig.level = 'DEBUG'; + this.logEvent('VERBOSE_LOGGING_ENABLED', 'INFO', 'Verbose logging enabled for debugging'); + } + + // Disable verbose logging + disableVerboseLogging() { + this.loggingConfig.level = 'INFO'; + this.logEvent('VERBOSE_LOGGING_DISABLED', 'INFO', 'Verbose logging disabled'); + } + + // Get comprehensive system health report + getSystemHealthReport() { + const report = { + timestamp: new Date().toISOString(), + sessionId: this.sessionId, + watchdogStatus: this.isRunning ? 'running' : 'stopped', + totalServices: Object.keys(this.services).length, + enabledServices: Object.values(this.services).filter(s => s.enabled).length, + healthyServices: Object.values(this.services).filter(s => s.status === 'healthy').length, + services: {}, + serviceMetrics: Object.fromEntries(this.serviceMetrics), + configuration: { + checkInterval: this.config.checkInterval, + gracePeriod: this.config.gracePeriod, + retryAttempts: this.config.retryAttempts + } + }; + + for (const [serviceKey, service] of Object.entries(this.services)) { + report.services[serviceKey] = { + name: service.name, + status: service.status, + enabled: service.enabled, + failureCount: service.failureCount, + isRetrying: service.isRetrying, + lastCheck: service.lastCheck, + lastHealthyTime: service.lastHealthyTime, + inGracePeriod: this.isServiceInGracePeriod(service) + }; + } + + return report; + } +} + +module.exports = WatchdogService; \ No newline at end of file diff --git a/electron/widgetService.cjs b/electron/widgetService.cjs new file mode 100644 index 00000000..c252496b --- /dev/null +++ b/electron/widgetService.cjs @@ -0,0 +1,452 @@ +const { spawn } = require('child_process'); +const path = require('path'); +const fs = require('fs'); +const { platform } = require('os'); +const log = require('electron-log'); + +class WidgetService { + constructor() { + this.process = null; + this.port = 8765; + this.isStarting = false; + this.requiredWidgets = new Set(['gpu-monitor', 'system-monitor', 'process-monitor', 'system-resources']); + this.activeWidgets = new Set(); + this.healthCheckInterval = null; + + // Platform-specific auto-start settings + this.platformSettings = { + darwin: false, // Disable auto-start on Mac by default (reduces heat) + win32: true, // Enable on Windows + linux: true // Enable on Linux + }; + } + + /** + * Get the correct executable name for the current platform + */ + getExecutableName() { + const currentPlatform = platform(); + switch (currentPlatform) { + case 'win32': + return 'widgets-service-windows.exe'; + case 'darwin': + return 'widgets-service-macos'; + case 'linux': + return 'widgets-service-linux'; + default: + return 'widgets-service'; + } + } + + /** + * Get the path to the widget service executable + */ + getServicePath() { + const execName = this.getExecutableName(); + + // Try electron app resources first (production) + const resourcesPath = process.resourcesPath + ? path.join(process.resourcesPath, 'electron', 'services', execName) + : null; + + // Try local development path + const devPath = path.join(__dirname, 'services', execName); + + // Try fallback to widgets_service_app folder (development) + const fallbackPath = path.join(__dirname, '..', 'widgets_service_app', execName); + + // Check which path exists + if (resourcesPath && fs.existsSync(resourcesPath)) { + log.info(`Using production service path: ${resourcesPath}`); + return resourcesPath; + } else if (fs.existsSync(devPath)) { + log.info(`Using development service path: ${devPath}`); + return devPath; + } else if (fs.existsSync(fallbackPath)) { + log.info(`Using fallback service path: ${fallbackPath}`); + return fallbackPath; + } else { + throw new Error(`Widget service executable not found. Checked paths: + - Resources: ${resourcesPath} + - Dev: ${devPath} + - Fallback: ${fallbackPath}`); + } + } + + /** + * Check if the service is currently running + */ + async isServiceRunning() { + if (this.process && !this.process.killed) { + return true; + } + + // Try to ping the service + try { + const response = await fetch(`http://localhost:${this.port}/api/health`); + return response.ok; + } catch (error) { + return false; + } + } + + /** + * Check if auto-start is enabled for the current platform + */ + isAutoStartEnabled() { + const currentPlatform = platform(); + return this.platformSettings[currentPlatform] !== false; + } + + /** + * Check if any active widgets require the service + */ + shouldServiceRun() { + // Don't auto-start service if disabled for this platform + if (!this.isAutoStartEnabled()) { + log.info(`Widget service auto-start disabled for platform: ${platform()}`); + return false; + } + + for (const widget of this.activeWidgets) { + if (this.requiredWidgets.has(widget)) { + return true; + } + } + return false; + } + + /** + * Register a widget as active + */ + registerWidget(widgetType) { + log.info(`Registering widget: ${widgetType}`); + this.activeWidgets.add(widgetType); + + if (this.requiredWidgets.has(widgetType)) { + log.info(`Widget ${widgetType} requires service, starting if needed`); + this.manageService(); + } + } + + /** + * Unregister a widget + */ + unregisterWidget(widgetType) { + log.info(`Unregistering widget: ${widgetType}`); + this.activeWidgets.delete(widgetType); + + if (this.requiredWidgets.has(widgetType)) { + log.info(`Widget ${widgetType} no longer active, checking if service still needed`); + this.manageService(); + } + } + + /** + * Start the widget service + */ + async startService() { + // If already starting, wait for it to complete + if (this.isStarting) { + log.info('Service is already starting, waiting for completion...'); + // Wait for the current startup to complete + while (this.isStarting) { + await new Promise(resolve => setTimeout(resolve, 100)); + } + // Check if service is now running + if (await this.isServiceRunning()) { + return { + success: true, + port: this.port, + pid: this.process?.pid || null, + message: 'Service started by another request' + }; + } + } + + if (await this.isServiceRunning()) { + log.info('Widget service already running'); + return { + success: true, + port: this.port, + pid: this.process?.pid || null, + message: 'Service already running' + }; + } + + this.isStarting = true; + + try { + const servicePath = this.getServicePath(); + + log.info(`Starting widget service: ${servicePath}`); + + this.process = spawn(servicePath, [this.port.toString()], { + detached: false, + stdio: ['ignore', 'pipe', 'pipe'], + windowsHide: true + }); + + // Handle process events + this.process.on('error', (error) => { + log.error('Widget service error:', error); + this.process = null; + this.isStarting = false; + }); + + this.process.on('exit', (code, signal) => { + log.info(`Widget service exited with code ${code}, signal ${signal}`); + this.process = null; + this.isStarting = false; + this.stopHealthCheck(); + }); + + // Log service output + if (this.process.stdout) { + this.process.stdout.on('data', (data) => { + log.info(`Widget Service: ${data.toString().trim()}`); + }); + } + + if (this.process.stderr) { + this.process.stderr.on('data', (data) => { + const message = data.toString().trim(); + if (message && !message.includes('Widget Service starting')) { + log.warn(`Widget Service Error: ${message}`); + } + }); + } + + // Wait for the service to start + await new Promise(resolve => setTimeout(resolve, 2000)); + + // Verify the service is running + const isRunning = await this.isServiceRunning(); + + if (!isRunning) { + this.process = null; + throw new Error('Service failed to start properly'); + } + + // Start health check + this.startHealthCheck(); + + log.info(`Widget service started successfully on port ${this.port}`); + + return { + success: true, + port: this.port, + pid: this.process?.pid || null, + message: 'Service started successfully' + }; + + } catch (error) { + log.error('Failed to start widget service:', error); + this.process = null; + return { + success: false, + port: this.port, + error: error.message + }; + } finally { + this.isStarting = false; + } + } + + /** + * Stop the widget service + */ + async stopService() { + // Check if already stopping or not running + if (this.isStopping) { + log.debug('Service is already stopping, waiting for completion...'); + while (this.isStopping) { + await new Promise(resolve => setTimeout(resolve, 100)); + } + return { success: true, message: 'Service stopped (was already stopping)' }; + } + + if (!this.process) { + log.info('Widget service not running'); + return { success: true, message: 'Service not running' }; + } + + this.isStopping = true; + + try { + log.info('Stopping widget service...'); + + // Stop health check + this.stopHealthCheck(); + + // Try graceful shutdown first + this.process.kill('SIGTERM'); + + // Wait for graceful shutdown + await new Promise((resolve) => { + const timeout = setTimeout(() => { + // Force kill if graceful shutdown takes too long + if (this.process && !this.process.killed) { + log.info('Force killing widget service...'); + this.process.kill('SIGKILL'); + } + resolve(); + }, 5000); + + if (this.process) { + this.process.on('exit', () => { + clearTimeout(timeout); + resolve(); + }); + } else { + clearTimeout(timeout); + resolve(); + } + }); + + this.process = null; + log.info('Widget service stopped'); + + return { success: true, message: 'Service stopped successfully' }; + + } catch (error) { + log.error('Error stopping widget service:', error); + this.process = null; + return { success: false, error: error.message }; + } finally { + this.isStopping = false; + } + } + + /** + * Enable auto-start for current platform + */ + enableAutoStart() { + const currentPlatform = platform(); + this.platformSettings[currentPlatform] = true; + log.info(`Widget service auto-start enabled for platform: ${currentPlatform}`); + } + + /** + * Disable auto-start for current platform + */ + disableAutoStart() { + const currentPlatform = platform(); + this.platformSettings[currentPlatform] = false; + log.info(`Widget service auto-start disabled for platform: ${currentPlatform}`); + } + + /** + * Get current service status + */ + async getStatus() { + const running = await this.isServiceRunning(); + const currentPlatform = platform(); + return { + running, + port: this.port, + pid: this.process?.pid, + activeWidgets: Array.from(this.activeWidgets), + shouldRun: this.shouldServiceRun(), + autoStartEnabled: this.isAutoStartEnabled(), + platform: currentPlatform + }; + } + + /** + * Manage service based on widget requirements + */ + async manageService() { + // Check if we're already managing the service + if (this.isStarting || this.isStopping) { + log.debug('Service management already in progress, waiting for completion...'); + + // Wait for the current operation to complete + while (this.isStarting || this.isStopping) { + await new Promise(resolve => setTimeout(resolve, 100)); + } + return await this.getStatus(); + } + + const shouldRun = this.shouldServiceRun(); + const isRunning = await this.isServiceRunning(); + + if (shouldRun && !isRunning) { + log.info('Starting widget service (required by active widgets)'); + return await this.startService(); + } else if (!shouldRun && isRunning) { + log.info('Stopping widget service (no longer needed)'); + return await this.stopService(); + } else { + return await this.getStatus(); + } + } + + /** + * Start health check interval + */ + startHealthCheck() { + if (this.healthCheckInterval) { + clearInterval(this.healthCheckInterval); + } + + this.healthCheckInterval = setInterval(async () => { + const isRunning = await this.isServiceRunning(); + if (!isRunning && this.process) { + log.warn('Widget service health check failed, process may have crashed'); + this.process = null; + + // Restart if widgets still need it + if (this.shouldServiceRun()) { + log.info('Attempting to restart widget service...'); + try { + await this.startService(); + } catch (error) { + log.error('Failed to restart widget service:', error); + } + } + } + }, 10000); // Check every 10 seconds + } + + /** + * Stop health check interval + */ + stopHealthCheck() { + if (this.healthCheckInterval) { + clearInterval(this.healthCheckInterval); + this.healthCheckInterval = null; + } + } + + /** + * Force restart the service + */ + async restartService() { + log.info('Force restarting widget service...'); + await this.stopService(); + await new Promise(resolve => setTimeout(resolve, 1000)); + return await this.startService(); + } + + /** + * Cleanup on application exit + */ + cleanup() { + this.stopHealthCheck(); + + if (this.process && !this.process.killed) { + log.info('Cleaning up widget service...'); + this.process.kill('SIGTERM'); + + // Force kill after 3 seconds if still running + setTimeout(() => { + if (this.process && !this.process.killed) { + this.process.kill('SIGKILL'); + } + }, 3000); + } + } +} + +module.exports = WidgetService; diff --git a/eslint.config.js b/eslint.config.js new file mode 100644 index 00000000..82c2e20c --- /dev/null +++ b/eslint.config.js @@ -0,0 +1,28 @@ +import js from '@eslint/js'; +import globals from 'globals'; +import reactHooks from 'eslint-plugin-react-hooks'; +import reactRefresh from 'eslint-plugin-react-refresh'; +import tseslint from 'typescript-eslint'; + +export default tseslint.config( + { ignores: ['dist'] }, + { + extends: [js.configs.recommended, ...tseslint.configs.recommended], + files: ['**/*.{ts,tsx}'], + languageOptions: { + ecmaVersion: 2020, + globals: globals.browser, + }, + plugins: { + 'react-hooks': reactHooks, + 'react-refresh': reactRefresh, + }, + rules: { + ...reactHooks.configs.recommended.rules, + 'react-refresh/only-export-components': [ + 'warn', + { allowConstantExport: true }, + ], + }, + } +); diff --git a/example-flows/research-assistant-flow.json b/example-flows/research-assistant-flow.json new file mode 100644 index 00000000..fdaa65ce --- /dev/null +++ b/example-flows/research-assistant-flow.json @@ -0,0 +1,157 @@ +{ + "format": "clara-native", + "version": "1.0.0", + "name": "Research Assistant Flow", + "description": "An advanced flow that takes a research topic, calls an API to gather information, and then uses LLM to analyze and summarize the findings", + "nodes": [ + { + "id": "input-topic", + "type": "input", + "position": { "x": 50, "y": 100 }, + "data": { + "label": "Research Topic", + "inputType": "text", + "placeholder": "Enter a topic to research...", + "required": true + } + }, + { + "id": "input-depth", + "type": "input", + "position": { "x": 50, "y": 250 }, + "data": { + "label": "Analysis Depth", + "inputType": "text", + "placeholder": "Basic, Detailed, or Expert level", + "required": false + } + }, + { + "id": "api-search", + "type": "api-request", + "position": { "x": 350, "y": 100 }, + "data": { + "label": "Search API", + "url": "https://api.duckduckgo.com/?q={{topic}}&format=json&no_html=1&skip_disambig=1", + "method": "GET", + "headers": { + "User-Agent": "Clara Research Assistant" + } + } + }, + { + "id": "llm-analyzer", + "type": "llm", + "position": { "x": 650, "y": 100 }, + "data": { + "label": "Research Analyzer", + "model": "llama3.2", + "prompt": "You are a research analyst. I've gathered some information about: {{topic}}\n\nSearched data: {{searchResults}}\n\nAnalysis depth requested: {{depth}}\n\nPlease provide a comprehensive analysis of this topic based on the available information. Structure your response with:\n1. Key findings\n2. Important insights\n3. Practical implications\n4. Recommendations for further research\n\nTailor the depth of analysis to the requested level.", + "temperature": 0.3, + "maxTokens": 1000 + } + }, + { + "id": "structured-summary", + "type": "structured-llm", + "position": { "x": 650, "y": 300 }, + "data": { + "label": "Structured Summary", + "model": "llama3.2", + "prompt": "Create a structured summary of this research analysis: {{analysis}}\n\nFormat the output as JSON with: title, summary, keyPoints (array), confidence (0-1), and sources (array).", + "outputSchema": { + "type": "object", + "properties": { + "title": { "type": "string" }, + "summary": { "type": "string" }, + "keyPoints": { + "type": "array", + "items": { "type": "string" } + }, + "confidence": { "type": "number", "minimum": 0, "maximum": 1 }, + "sources": { + "type": "array", + "items": { "type": "string" } + } + } + } + } + }, + { + "id": "output-analysis", + "type": "output", + "position": { "x": 950, "y": 100 }, + "data": { + "label": "Research Analysis", + "format": "text" + } + }, + { + "id": "output-summary", + "type": "output", + "position": { "x": 950, "y": 300 }, + "data": { + "label": "Structured Summary", + "format": "json" + } + } + ], + "connections": [ + { + "id": "topic-to-api", + "sourceNodeId": "input-topic", + "sourcePortId": "output", + "targetNodeId": "api-search", + "targetPortId": "topic" + }, + { + "id": "topic-to-llm", + "sourceNodeId": "input-topic", + "sourcePortId": "output", + "targetNodeId": "llm-analyzer", + "targetPortId": "topic" + }, + { + "id": "depth-to-llm", + "sourceNodeId": "input-depth", + "sourcePortId": "output", + "targetNodeId": "llm-analyzer", + "targetPortId": "depth" + }, + { + "id": "api-to-llm", + "sourceNodeId": "api-search", + "sourcePortId": "output", + "targetNodeId": "llm-analyzer", + "targetPortId": "searchResults" + }, + { + "id": "llm-to-output", + "sourceNodeId": "llm-analyzer", + "sourcePortId": "output", + "targetNodeId": "output-analysis", + "targetPortId": "input" + }, + { + "id": "llm-to-structured", + "sourceNodeId": "llm-analyzer", + "sourcePortId": "output", + "targetNodeId": "structured-summary", + "targetPortId": "analysis" + }, + { + "id": "structured-to-output", + "sourceNodeId": "structured-summary", + "sourcePortId": "output", + "targetNodeId": "output-summary", + "targetPortId": "input" + } + ], + "metadata": { + "createdAt": "2024-01-15T10:30:00Z", + "exportedBy": "Clara Agent Studio", + "hasCustomNodes": false, + "complexity": "advanced", + "estimatedExecutionTime": "10-30 seconds" + } +} \ No newline at end of file diff --git a/example-flows/simple-llm-flow.json b/example-flows/simple-llm-flow.json new file mode 100644 index 00000000..b2e3df4a --- /dev/null +++ b/example-flows/simple-llm-flow.json @@ -0,0 +1,61 @@ +{ + "format": "clara-native", + "version": "1.0.0", + "name": "Simple LLM Chat Flow", + "description": "A basic flow that takes user input and processes it through an LLM to generate a response", + "nodes": [ + { + "id": "input-1", + "type": "input", + "position": { "x": 100, "y": 100 }, + "data": { + "label": "Tell me what to do", + "inputType": "textarea", + "placeholder": "Ask me anything...", + "required": true + } + }, + { + "id": "llm-1", + "type": "llm", + "position": { "x": 400, "y": 100 }, + "data": { + "label": "AI Assistant", + "model": "llama3.2", + "prompt": "You are a helpful AI assistant. Please respond to the user's question or request in a clear and helpful way:\n\n{{input}}", + "temperature": 0.7, + "maxTokens": 500 + } + }, + { + "id": "output-1", + "type": "output", + "position": { "x": 700, "y": 100 }, + "data": { + "label": "AI Response", + "format": "text" + } + } + ], + "connections": [ + { + "id": "connection-1", + "sourceNodeId": "input-1", + "sourcePortId": "output", + "targetNodeId": "llm-1", + "targetPortId": "input" + }, + { + "id": "connection-2", + "sourceNodeId": "llm-1", + "sourcePortId": "output", + "targetNodeId": "output-1", + "targetPortId": "input" + } + ], + "metadata": { + "createdAt": "2024-01-15T10:00:00Z", + "exportedBy": "Clara Agent Studio", + "hasCustomNodes": false + } +} \ No newline at end of file diff --git a/frontend/.dockerignore b/frontend/.dockerignore deleted file mode 100644 index cbb133c5..00000000 --- a/frontend/.dockerignore +++ /dev/null @@ -1,45 +0,0 @@ -# Dependencies -node_modules/ -npm-debug.log* -yarn-debug.log* -yarn-error.log* -pnpm-debug.log* - -# Build output -dist/ -build/ -.cache/ - -# Environment files -.env -.env.local -.env.*.local - -# IDE and editor files -.vscode/ -.idea/ -*.swp -*.swo -*~ -.DS_Store - -# Git -.git -.gitignore - -# Testing -coverage/ -.nyc_output/ - -# Documentation -*.md -README.md -docs/ - -# Misc -.prettierrc -.prettierignore -.eslintrc* -# Keep build configuration files -# tsconfig*.json - NEEDED for build -# vite.config.ts - NEEDED for build diff --git a/frontend/.env.example b/frontend/.env.example deleted file mode 100644 index 2c37180d..00000000 --- a/frontend/.env.example +++ /dev/null @@ -1,22 +0,0 @@ -# API Configuration -VITE_API_BASE_URL=http://localhost:3001 -VITE_WS_URL=ws://localhost:3001 - -# Supabase Configuration -VITE_SUPABASE_URL=your-supabase-project-url -VITE_SUPABASE_ANON_KEY=your-supabase-anon-key - -# App Configuration -VITE_APP_NAME=ClaraVerse -VITE_APP_VERSION=1.0.0 - -# Cloudflare Turnstile CAPTCHA -# Get your site key from: https://dash.cloudflare.com/ -# For local testing, add localhost to domain allowlist -VITE_TURNSTILE_SITE_KEY=your-turnstile-site-key - -# Feature Flags -VITE_ENABLE_ANALYTICS=false - -# Error Reporting -VITE_DISCORD_ERROR_WEBHOOK= diff --git a/frontend/.gitignore b/frontend/.gitignore deleted file mode 100644 index a547bf36..00000000 --- a/frontend/.gitignore +++ /dev/null @@ -1,24 +0,0 @@ -# Logs -logs -*.log -npm-debug.log* -yarn-debug.log* -yarn-error.log* -pnpm-debug.log* -lerna-debug.log* - -node_modules -dist -dist-ssr -*.local - -# Editor directories and files -.vscode/* -!.vscode/extensions.json -.idea -.DS_Store -*.suo -*.ntvs* -*.njsproj -*.sln -*.sw? diff --git a/frontend/.prettierignore b/frontend/.prettierignore deleted file mode 100644 index 741fb784..00000000 --- a/frontend/.prettierignore +++ /dev/null @@ -1,11 +0,0 @@ -node_modules -dist -build -coverage -.next -.vite -*.min.js -*.min.css -package-lock.json -pnpm-lock.yaml -yarn.lock diff --git a/frontend/.prettierrc b/frontend/.prettierrc deleted file mode 100644 index 49fb29b5..00000000 --- a/frontend/.prettierrc +++ /dev/null @@ -1,10 +0,0 @@ -{ - "semi": true, - "trailingComma": "es5", - "singleQuote": true, - "printWidth": 100, - "tabWidth": 2, - "useTabs": false, - "arrowParens": "avoid", - "endOfLine": "lf" -} diff --git a/frontend/Dockerfile b/frontend/Dockerfile deleted file mode 100644 index 2c89ec9c..00000000 --- a/frontend/Dockerfile +++ /dev/null @@ -1,57 +0,0 @@ -# Stage 1: Builder -FROM node:20-alpine AS builder - -# Build arguments for environment variables -ARG VITE_API_BASE_URL=http://localhost:3001 -ARG VITE_WS_URL=ws://localhost:3001 -ARG VITE_SUPABASE_URL -ARG VITE_SUPABASE_ANON_KEY -ARG VITE_APP_NAME=ClaraVerse -ARG VITE_APP_VERSION=1.0.0 -ARG VITE_ENABLE_ANALYTICS=false -ARG VITE_TURNSTILE_SITE_KEY - -# Set environment variables for Vite build -ENV VITE_API_BASE_URL=$VITE_API_BASE_URL -ENV VITE_WS_URL=$VITE_WS_URL -ENV VITE_SUPABASE_URL=$VITE_SUPABASE_URL -ENV VITE_SUPABASE_ANON_KEY=$VITE_SUPABASE_ANON_KEY -ENV VITE_APP_NAME=$VITE_APP_NAME -ENV VITE_APP_VERSION=$VITE_APP_VERSION -ENV VITE_ENABLE_ANALYTICS=$VITE_ENABLE_ANALYTICS -ENV VITE_TURNSTILE_SITE_KEY=$VITE_TURNSTILE_SITE_KEY - -# Set working directory -WORKDIR /app - -# Copy package files first (for better layer caching) -COPY package.json package-lock.json ./ - -# Install dependencies -RUN npm ci --legacy-peer-deps - -# Copy source code -COPY . . - -# Build the application (skip type checking for Docker build) -# Type checking should be done in CI/CD or locally -RUN npx vite build - -# Stage 2: Production with Nginx -FROM nginx:alpine - -# Copy custom nginx configuration -COPY nginx.conf /etc/nginx/conf.d/default.conf - -# Copy built assets from builder stage -COPY --from=builder /app/dist /usr/share/nginx/html - -# Expose port 80 -EXPOSE 80 - -# Health check -HEALTHCHECK --interval=30s --timeout=3s --start-period=5s --retries=3 \ - CMD wget --no-verbose --tries=1 --spider http://localhost:80 || exit 1 - -# Start nginx -CMD ["nginx", "-g", "daemon off;"] diff --git a/frontend/Dockerfile.dev b/frontend/Dockerfile.dev deleted file mode 100644 index 997bd8e7..00000000 --- a/frontend/Dockerfile.dev +++ /dev/null @@ -1,20 +0,0 @@ -# Frontend Development Dockerfile with HMR -FROM node:20-alpine - -# Set working directory -WORKDIR /app - -# Install dependencies first (for better caching) -COPY package.json package-lock.json ./ -RUN npm ci --legacy-peer-deps - -# Expose Vite dev server port -EXPOSE 5173 - -# Health check for dev server -HEALTHCHECK --interval=30s --timeout=3s --start-period=30s --retries=3 \ - CMD wget --no-verbose --tries=1 --spider http://localhost:5173 || exit 1 - -# Start Vite dev server with host binding for Docker -CMD ["npm", "run", "dev", "--", "--host", "0.0.0.0"] - diff --git a/frontend/eslint.config.js b/frontend/eslint.config.js deleted file mode 100644 index 9c453a37..00000000 --- a/frontend/eslint.config.js +++ /dev/null @@ -1,33 +0,0 @@ -import js from '@eslint/js' -import globals from 'globals' -import reactHooks from 'eslint-plugin-react-hooks' -import reactRefresh from 'eslint-plugin-react-refresh' -import tseslint from 'typescript-eslint' -import prettier from 'eslint-plugin-prettier' -import prettierConfig from 'eslint-config-prettier' -import { defineConfig, globalIgnores } from 'eslint/config' - -export default defineConfig([ - globalIgnores(['dist', 'node_modules', 'build']), - { - files: ['**/*.{ts,tsx}'], - extends: [ - js.configs.recommended, - tseslint.configs.recommended, - reactHooks.configs['recommended-latest'], - reactRefresh.configs.vite, - prettierConfig, - ], - plugins: { - prettier, - }, - languageOptions: { - ecmaVersion: 2020, - globals: globals.browser, - }, - rules: { - 'prettier/prettier': 'error', - '@typescript-eslint/no-unused-vars': ['error', { argsIgnorePattern: '^_' }], - }, - }, -]) diff --git a/frontend/index.html b/frontend/index.html deleted file mode 100644 index f6ae18c2..00000000 --- a/frontend/index.html +++ /dev/null @@ -1,54 +0,0 @@ - - - - - - - - ClaraVerse - AI-Powered Chat Assistant - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
- - - diff --git a/frontend/nginx.conf b/frontend/nginx.conf deleted file mode 100644 index 8228318f..00000000 --- a/frontend/nginx.conf +++ /dev/null @@ -1,35 +0,0 @@ -server { - listen 80; - server_name localhost; - root /usr/share/nginx/html; - index index.html; - - # Gzip compression - gzip on; - gzip_vary on; - gzip_min_length 1024; - gzip_types text/plain text/css text/xml text/javascript application/x-javascript application/xml+rss application/json application/javascript; - - # Security headers - add_header X-Frame-Options "SAMEORIGIN" always; - add_header X-Content-Type-Options "nosniff" always; - add_header X-XSS-Protection "1; mode=block" always; - - # Cache static assets - location ~* \.(jpg|jpeg|png|gif|ico|css|js|svg|woff|woff2|ttf|eot)$ { - expires 1y; - add_header Cache-Control "public, immutable"; - } - - # SPA fallback - all routes serve index.html - location / { - try_files $uri $uri/ /index.html; - } - - # Health check endpoint - location /health { - access_log off; - return 200 "healthy\n"; - add_header Content-Type text/plain; - } -} diff --git a/frontend/package-lock.json b/frontend/package-lock.json deleted file mode 100644 index 009e4a2a..00000000 --- a/frontend/package-lock.json +++ /dev/null @@ -1,9433 +0,0 @@ -{ - "name": "frontend", - "version": "0.0.0", - "lockfileVersion": 3, - "requires": true, - "packages": { - "": { - "name": "frontend", - "version": "0.0.0", - "dependencies": { - "@dagrejs/dagre": "^1.1.8", - "@marsidev/react-turnstile": "^1.3.1", - "@radix-ui/react-slot": "^1.2.4", - "@supabase/supabase-js": "^2.81.1", - "@types/dagre": "^0.7.53", - "@xyflow/react": "^12.9.3", - "chart.js": "^4.5.1", - "class-variance-authority": "^0.7.1", - "clsx": "^2.1.1", - "dagre": "^0.8.5", - "dompurify": "^3.3.0", - "driver.js": "^1.4.0", - "framer-motion": "^12.23.24", - "html-to-image": "^1.11.13", - "idb": "^8.0.3", - "lucide-react": "^0.553.0", - "mermaid": "^11.12.1", - "pdfjs-dist": "^5.4.530", - "react": "^19.2.0", - "react-chartjs-2": "^5.3.1", - "react-dom": "^19.2.0", - "react-markdown": "^10.1.0", - "react-pdf": "^10.3.0", - "react-resizable-panels": "^3.0.6", - "react-router-dom": "^7.9.5", - "react-syntax-highlighter": "^16.1.0", - "react-virtuoso": "^4.14.1", - "react-zoom-pan-pinch": "^3.7.0", - "remark-gfm": "^4.0.1", - "tailwind-merge": "^3.4.0", - "zustand": "^5.0.8" - }, - "devDependencies": { - "@eslint/js": "^9.39.1", - "@types/dompurify": "^3.0.5", - "@types/node": "^24.10.0", - "@types/react": "^19.2.2", - "@types/react-dom": "^19.2.2", - "@types/react-syntax-highlighter": "^15.5.13", - "@vitejs/plugin-react": "^5.1.0", - "@vitest/coverage-v8": "^3.2.4", - "autoprefixer": "^10.4.22", - "eslint": "^9.39.1", - "eslint-config-prettier": "^10.1.8", - "eslint-plugin-prettier": "^5.5.4", - "eslint-plugin-react-hooks": "^5.2.0", - "eslint-plugin-react-refresh": "^0.4.24", - "globals": "^16.5.0", - "postcss": "^8.5.6", - "prettier": "^3.6.2", - "tailwindcss": "^3.4.17", - "typescript": "~5.9.3", - "typescript-eslint": "^8.46.3", - "vite": "^7.2.2", - "vitest": "^3.2.4" - } - }, - "node_modules/@alloc/quick-lru": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/@alloc/quick-lru/-/quick-lru-5.2.0.tgz", - "integrity": "sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/@ampproject/remapping": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.3.0.tgz", - "integrity": "sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "@jridgewell/gen-mapping": "^0.3.5", - "@jridgewell/trace-mapping": "^0.3.24" - }, - "engines": { - "node": ">=6.0.0" - } - }, - "node_modules/@antfu/install-pkg": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@antfu/install-pkg/-/install-pkg-1.1.0.tgz", - "integrity": "sha512-MGQsmw10ZyI+EJo45CdSER4zEb+p31LpDAFp2Z3gkSd1yqVZGi0Ebx++YTEMonJy4oChEMLsxZ64j8FH6sSqtQ==", - "license": "MIT", - "dependencies": { - "package-manager-detector": "^1.3.0", - "tinyexec": "^1.0.1" - }, - "funding": { - "url": "https://github.com/sponsors/antfu" - } - }, - "node_modules/@antfu/utils": { - "version": "9.3.0", - "resolved": "https://registry.npmjs.org/@antfu/utils/-/utils-9.3.0.tgz", - "integrity": "sha512-9hFT4RauhcUzqOE4f1+frMKLZrgNog5b06I7VmZQV1BkvwvqrbC8EBZf3L1eEL2AKb6rNKjER0sEvJiSP1FXEA==", - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/antfu" - } - }, - "node_modules/@babel/code-frame": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.27.1.tgz", - "integrity": "sha512-cjQ7ZlQ0Mv3b47hABuTevyTuYN4i+loJKGeV9flcCgIK37cCXRh+L1bd3iBHlynerhQ7BhCkn2BPbQUL+rGqFg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-validator-identifier": "^7.27.1", - "js-tokens": "^4.0.0", - "picocolors": "^1.1.1" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/compat-data": { - "version": "7.28.5", - "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.28.5.tgz", - "integrity": "sha512-6uFXyCayocRbqhZOB+6XcuZbkMNimwfVGFji8CTZnCzOHVGvDqzvitu1re2AU5LROliz7eQPhB8CpAMvnx9EjA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/core": { - "version": "7.28.5", - "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.28.5.tgz", - "integrity": "sha512-e7jT4DxYvIDLk1ZHmU/m/mB19rex9sv0c2ftBtjSBv+kVM/902eh0fINUzD7UwLLNR+jU585GxUJ8/EBfAM5fw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/code-frame": "^7.27.1", - "@babel/generator": "^7.28.5", - "@babel/helper-compilation-targets": "^7.27.2", - "@babel/helper-module-transforms": "^7.28.3", - "@babel/helpers": "^7.28.4", - "@babel/parser": "^7.28.5", - "@babel/template": "^7.27.2", - "@babel/traverse": "^7.28.5", - "@babel/types": "^7.28.5", - "@jridgewell/remapping": "^2.3.5", - "convert-source-map": "^2.0.0", - "debug": "^4.1.0", - "gensync": "^1.0.0-beta.2", - "json5": "^2.2.3", - "semver": "^6.3.1" - }, - "engines": { - "node": ">=6.9.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/babel" - } - }, - "node_modules/@babel/generator": { - "version": "7.28.5", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.28.5.tgz", - "integrity": "sha512-3EwLFhZ38J4VyIP6WNtt2kUdW9dokXA9Cr4IVIFHuCpZ3H8/YFOl5JjZHisrn1fATPBmKKqXzDFvh9fUwHz6CQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/parser": "^7.28.5", - "@babel/types": "^7.28.5", - "@jridgewell/gen-mapping": "^0.3.12", - "@jridgewell/trace-mapping": "^0.3.28", - "jsesc": "^3.0.2" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-compilation-targets": { - "version": "7.27.2", - "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.27.2.tgz", - "integrity": "sha512-2+1thGUUWWjLTYTHZWK1n8Yga0ijBz1XAhUXcKy81rd5g6yh7hGqMp45v7cadSbEHc9G3OTv45SyneRN3ps4DQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/compat-data": "^7.27.2", - "@babel/helper-validator-option": "^7.27.1", - "browserslist": "^4.24.0", - "lru-cache": "^5.1.1", - "semver": "^6.3.1" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-globals": { - "version": "7.28.0", - "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz", - "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-module-imports": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.27.1.tgz", - "integrity": "sha512-0gSFWUPNXNopqtIPQvlD5WgXYI5GY2kP2cCvoT8kczjbfcfuIljTbcWrulD1CIPIX2gt1wghbDy08yE1p+/r3w==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/traverse": "^7.27.1", - "@babel/types": "^7.27.1" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-module-transforms": { - "version": "7.28.3", - "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.3.tgz", - "integrity": "sha512-gytXUbs8k2sXS9PnQptz5o0QnpLL51SwASIORY6XaBKF88nsOT0Zw9szLqlSGQDP/4TljBAD5y98p2U1fqkdsw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-module-imports": "^7.27.1", - "@babel/helper-validator-identifier": "^7.27.1", - "@babel/traverse": "^7.28.3" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0" - } - }, - "node_modules/@babel/helper-plugin-utils": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.27.1.tgz", - "integrity": "sha512-1gn1Up5YXka3YYAHGKpbideQ5Yjf1tDa9qYcgysz+cNCXukyLl6DjPXhD3VRwSb8c0J9tA4b2+rHEZtc6R0tlw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-string-parser": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz", - "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-validator-identifier": { - "version": "7.28.5", - "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz", - "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-validator-option": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.27.1.tgz", - "integrity": "sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helpers": { - "version": "7.28.4", - "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.28.4.tgz", - "integrity": "sha512-HFN59MmQXGHVyYadKLVumYsA9dBFun/ldYxipEjzA4196jpLZd8UjEEBLkbEkvfYreDqJhZxYAWFPtrfhNpj4w==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/template": "^7.27.2", - "@babel/types": "^7.28.4" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/parser": { - "version": "7.28.5", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.28.5.tgz", - "integrity": "sha512-KKBU1VGYR7ORr3At5HAtUQ+TV3SzRCXmA/8OdDZiLDBIZxVyzXuztPjfLd3BV1PRAQGCMWWSHYhL0F8d5uHBDQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/types": "^7.28.5" - }, - "bin": { - "parser": "bin/babel-parser.js" - }, - "engines": { - "node": ">=6.0.0" - } - }, - "node_modules/@babel/plugin-transform-react-jsx-self": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-self/-/plugin-transform-react-jsx-self-7.27.1.tgz", - "integrity": "sha512-6UzkCs+ejGdZ5mFFC/OCUrv028ab2fp1znZmCZjAOBKiBK2jXD1O+BPSfX8X2qjJ75fZBMSnQn3Rq2mrBJK2mw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.27.1" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-react-jsx-source": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-source/-/plugin-transform-react-jsx-source-7.27.1.tgz", - "integrity": "sha512-zbwoTsBruTeKB9hSq73ha66iFeJHuaFkUbwvqElnygoNbj/jHRsSeokowZFN3CZ64IvEqcmmkVe89OPXc7ldAw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.27.1" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/runtime": { - "version": "7.28.4", - "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.28.4.tgz", - "integrity": "sha512-Q/N6JNWvIvPnLDvjlE1OUBLPQHH6l3CltCEsHIujp45zQUSSh8K+gHnaEX45yAT1nyngnINhvWtzN+Nb9D8RAQ==", - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/template": { - "version": "7.27.2", - "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.27.2.tgz", - "integrity": "sha512-LPDZ85aEJyYSd18/DkjNh4/y1ntkE5KwUHWTiqgRxruuZL2F1yuHligVHLvcHY2vMHXttKFpJn6LwfI7cw7ODw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/code-frame": "^7.27.1", - "@babel/parser": "^7.27.2", - "@babel/types": "^7.27.1" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/traverse": { - "version": "7.28.5", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.28.5.tgz", - "integrity": "sha512-TCCj4t55U90khlYkVV/0TfkJkAkUg3jZFA3Neb7unZT8CPok7iiRfaX0F+WnqWqt7OxhOn0uBKXCw4lbL8W0aQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/code-frame": "^7.27.1", - "@babel/generator": "^7.28.5", - "@babel/helper-globals": "^7.28.0", - "@babel/parser": "^7.28.5", - "@babel/template": "^7.27.2", - "@babel/types": "^7.28.5", - "debug": "^4.3.1" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/types": { - "version": "7.28.5", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.28.5.tgz", - "integrity": "sha512-qQ5m48eI/MFLQ5PxQj4PFaprjyCTLI37ElWMmNs0K8Lk3dVeOdNpB3ks8jc7yM5CDmVC73eMVk/trk3fgmrUpA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-string-parser": "^7.27.1", - "@babel/helper-validator-identifier": "^7.28.5" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@bcoe/v8-coverage": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/@bcoe/v8-coverage/-/v8-coverage-1.0.2.tgz", - "integrity": "sha512-6zABk/ECA/QYSCQ1NGiVwwbQerUCZ+TQbp64Q3AgmfNvurHH0j8TtXa1qbShXA6qqkpAj4V5W8pP6mLe1mcMqA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=18" - } - }, - "node_modules/@braintree/sanitize-url": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/@braintree/sanitize-url/-/sanitize-url-7.1.1.tgz", - "integrity": "sha512-i1L7noDNxtFyL5DmZafWy1wRVhGehQmzZaz1HiN5e7iylJMSZR7ekOV7NsIqa5qBldlLrsKv4HbgFUVlQrz8Mw==", - "license": "MIT" - }, - "node_modules/@chevrotain/cst-dts-gen": { - "version": "11.0.3", - "resolved": "https://registry.npmjs.org/@chevrotain/cst-dts-gen/-/cst-dts-gen-11.0.3.tgz", - "integrity": "sha512-BvIKpRLeS/8UbfxXxgC33xOumsacaeCKAjAeLyOn7Pcp95HiRbrpl14S+9vaZLolnbssPIUuiUd8IvgkRyt6NQ==", - "license": "Apache-2.0", - "dependencies": { - "@chevrotain/gast": "11.0.3", - "@chevrotain/types": "11.0.3", - "lodash-es": "4.17.21" - } - }, - "node_modules/@chevrotain/gast": { - "version": "11.0.3", - "resolved": "https://registry.npmjs.org/@chevrotain/gast/-/gast-11.0.3.tgz", - "integrity": "sha512-+qNfcoNk70PyS/uxmj3li5NiECO+2YKZZQMbmjTqRI3Qchu8Hig/Q9vgkHpI3alNjr7M+a2St5pw5w5F6NL5/Q==", - "license": "Apache-2.0", - "dependencies": { - "@chevrotain/types": "11.0.3", - "lodash-es": "4.17.21" - } - }, - "node_modules/@chevrotain/regexp-to-ast": { - "version": "11.0.3", - "resolved": "https://registry.npmjs.org/@chevrotain/regexp-to-ast/-/regexp-to-ast-11.0.3.tgz", - "integrity": "sha512-1fMHaBZxLFvWI067AVbGJav1eRY7N8DDvYCTwGBiE/ytKBgP8azTdgyrKyWZ9Mfh09eHWb5PgTSO8wi7U824RA==", - "license": "Apache-2.0" - }, - "node_modules/@chevrotain/types": { - "version": "11.0.3", - "resolved": "https://registry.npmjs.org/@chevrotain/types/-/types-11.0.3.tgz", - "integrity": "sha512-gsiM3G8b58kZC2HaWR50gu6Y1440cHiJ+i3JUvcp/35JchYejb2+5MVeJK0iKThYpAa/P2PYFV4hoi44HD+aHQ==", - "license": "Apache-2.0" - }, - "node_modules/@chevrotain/utils": { - "version": "11.0.3", - "resolved": "https://registry.npmjs.org/@chevrotain/utils/-/utils-11.0.3.tgz", - "integrity": "sha512-YslZMgtJUyuMbZ+aKvfF3x1f5liK4mWNxghFRv7jqRR9C3R3fAOGTTKvxXDa2Y1s9zSbcpuO0cAxDYsc9SrXoQ==", - "license": "Apache-2.0" - }, - "node_modules/@dagrejs/dagre": { - "version": "1.1.8", - "resolved": "https://registry.npmjs.org/@dagrejs/dagre/-/dagre-1.1.8.tgz", - "integrity": "sha512-5SEDlndt4W/LaVzPYJW+bSmSEZc9EzTf8rJ20WCKvjS5EAZAN0b+x0Yww7VMT4R3Wootkg+X9bUfUxazYw6Blw==", - "license": "MIT", - "dependencies": { - "@dagrejs/graphlib": "2.2.4" - } - }, - "node_modules/@dagrejs/graphlib": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/@dagrejs/graphlib/-/graphlib-2.2.4.tgz", - "integrity": "sha512-mepCf/e9+SKYy1d02/UkvSy6+6MoyXhVxP8lLDfA7BPE1X1d4dR0sZznmbM8/XVJ1GPM+Svnx7Xj6ZweByWUkw==", - "license": "MIT", - "engines": { - "node": ">17.0.0" - } - }, - "node_modules/@esbuild/aix-ppc64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.25.12.tgz", - "integrity": "sha512-Hhmwd6CInZ3dwpuGTF8fJG6yoWmsToE+vYgD4nytZVxcu1ulHpUQRAB1UJ8+N1Am3Mz4+xOByoQoSZf4D+CpkA==", - "cpu": [ - "ppc64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "aix" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/android-arm": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.25.12.tgz", - "integrity": "sha512-VJ+sKvNA/GE7Ccacc9Cha7bpS8nyzVv0jdVgwNDaR4gDMC/2TTRc33Ip8qrNYUcpkOHUT5OZ0bUcNNVZQ9RLlg==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/android-arm64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.25.12.tgz", - "integrity": "sha512-6AAmLG7zwD1Z159jCKPvAxZd4y/VTO0VkprYy+3N2FtJ8+BQWFXU+OxARIwA46c5tdD9SsKGZ/1ocqBS/gAKHg==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/android-x64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.25.12.tgz", - "integrity": "sha512-5jbb+2hhDHx5phYR2By8GTWEzn6I9UqR11Kwf22iKbNpYrsmRB18aX/9ivc5cabcUiAT/wM+YIZ6SG9QO6a8kg==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/darwin-arm64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.25.12.tgz", - "integrity": "sha512-N3zl+lxHCifgIlcMUP5016ESkeQjLj/959RxxNYIthIg+CQHInujFuXeWbWMgnTo4cp5XVHqFPmpyu9J65C1Yg==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/darwin-x64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.25.12.tgz", - "integrity": "sha512-HQ9ka4Kx21qHXwtlTUVbKJOAnmG1ipXhdWTmNXiPzPfWKpXqASVcWdnf2bnL73wgjNrFXAa3yYvBSd9pzfEIpA==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/freebsd-arm64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.12.tgz", - "integrity": "sha512-gA0Bx759+7Jve03K1S0vkOu5Lg/85dou3EseOGUes8flVOGxbhDDh/iZaoek11Y8mtyKPGF3vP8XhnkDEAmzeg==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/freebsd-x64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.25.12.tgz", - "integrity": "sha512-TGbO26Yw2xsHzxtbVFGEXBFH0FRAP7gtcPE7P5yP7wGy7cXK2oO7RyOhL5NLiqTlBh47XhmIUXuGciXEqYFfBQ==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-arm": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.25.12.tgz", - "integrity": "sha512-lPDGyC1JPDou8kGcywY0YILzWlhhnRjdof3UlcoqYmS9El818LLfJJc3PXXgZHrHCAKs/Z2SeZtDJr5MrkxtOw==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-arm64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.25.12.tgz", - "integrity": "sha512-8bwX7a8FghIgrupcxb4aUmYDLp8pX06rGh5HqDT7bB+8Rdells6mHvrFHHW2JAOPZUbnjUpKTLg6ECyzvas2AQ==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-ia32": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.25.12.tgz", - "integrity": "sha512-0y9KrdVnbMM2/vG8KfU0byhUN+EFCny9+8g202gYqSSVMonbsCfLjUO+rCci7pM0WBEtz+oK/PIwHkzxkyharA==", - "cpu": [ - "ia32" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-loong64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.25.12.tgz", - "integrity": "sha512-h///Lr5a9rib/v1GGqXVGzjL4TMvVTv+s1DPoxQdz7l/AYv6LDSxdIwzxkrPW438oUXiDtwM10o9PmwS/6Z0Ng==", - "cpu": [ - "loong64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-mips64el": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.25.12.tgz", - "integrity": "sha512-iyRrM1Pzy9GFMDLsXn1iHUm18nhKnNMWscjmp4+hpafcZjrr2WbT//d20xaGljXDBYHqRcl8HnxbX6uaA/eGVw==", - "cpu": [ - "mips64el" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-ppc64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.25.12.tgz", - "integrity": "sha512-9meM/lRXxMi5PSUqEXRCtVjEZBGwB7P/D4yT8UG/mwIdze2aV4Vo6U5gD3+RsoHXKkHCfSxZKzmDssVlRj1QQA==", - "cpu": [ - "ppc64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-riscv64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.25.12.tgz", - "integrity": "sha512-Zr7KR4hgKUpWAwb1f3o5ygT04MzqVrGEGXGLnj15YQDJErYu/BGg+wmFlIDOdJp0PmB0lLvxFIOXZgFRrdjR0w==", - "cpu": [ - "riscv64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-s390x": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.25.12.tgz", - "integrity": "sha512-MsKncOcgTNvdtiISc/jZs/Zf8d0cl/t3gYWX8J9ubBnVOwlk65UIEEvgBORTiljloIWnBzLs4qhzPkJcitIzIg==", - "cpu": [ - "s390x" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/linux-x64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.25.12.tgz", - "integrity": "sha512-uqZMTLr/zR/ed4jIGnwSLkaHmPjOjJvnm6TVVitAa08SLS9Z0VM8wIRx7gWbJB5/J54YuIMInDquWyYvQLZkgw==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/netbsd-arm64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.12.tgz", - "integrity": "sha512-xXwcTq4GhRM7J9A8Gv5boanHhRa/Q9KLVmcyXHCTaM4wKfIpWkdXiMog/KsnxzJ0A1+nD+zoecuzqPmCRyBGjg==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "netbsd" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/netbsd-x64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.25.12.tgz", - "integrity": "sha512-Ld5pTlzPy3YwGec4OuHh1aCVCRvOXdH8DgRjfDy/oumVovmuSzWfnSJg+VtakB9Cm0gxNO9BzWkj6mtO1FMXkQ==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "netbsd" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/openbsd-arm64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.12.tgz", - "integrity": "sha512-fF96T6KsBo/pkQI950FARU9apGNTSlZGsv1jZBAlcLL1MLjLNIWPBkj5NlSz8aAzYKg+eNqknrUJ24QBybeR5A==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "openbsd" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/openbsd-x64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.25.12.tgz", - "integrity": "sha512-MZyXUkZHjQxUvzK7rN8DJ3SRmrVrke8ZyRusHlP+kuwqTcfWLyqMOE3sScPPyeIXN/mDJIfGXvcMqCgYKekoQw==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "openbsd" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/openharmony-arm64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/openharmony-arm64/-/openharmony-arm64-0.25.12.tgz", - "integrity": "sha512-rm0YWsqUSRrjncSXGA7Zv78Nbnw4XL6/dzr20cyrQf7ZmRcsovpcRBdhD43Nuk3y7XIoW2OxMVvwuRvk9XdASg==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "openharmony" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/sunos-x64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.25.12.tgz", - "integrity": "sha512-3wGSCDyuTHQUzt0nV7bocDy72r2lI33QL3gkDNGkod22EsYl04sMf0qLb8luNKTOmgF/eDEDP5BFNwoBKH441w==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "sunos" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/win32-arm64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.25.12.tgz", - "integrity": "sha512-rMmLrur64A7+DKlnSuwqUdRKyd3UE7oPJZmnljqEptesKM8wx9J8gx5u0+9Pq0fQQW8vqeKebwNXdfOyP+8Bsg==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/win32-ia32": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.25.12.tgz", - "integrity": "sha512-HkqnmmBoCbCwxUKKNPBixiWDGCpQGVsrQfJoVGYLPT41XWF8lHuE5N6WhVia2n4o5QK5M4tYr21827fNhi4byQ==", - "cpu": [ - "ia32" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@esbuild/win32-x64": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.25.12.tgz", - "integrity": "sha512-alJC0uCZpTFrSL0CCDjcgleBXPnCrEAhTBILpeAp7M/OFgoqtAetfBzX0xM00MUsVVPpVjlPuMbREqnZCXaTnA==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=18" - } - }, - "node_modules/@eslint-community/eslint-utils": { - "version": "4.9.0", - "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.0.tgz", - "integrity": "sha512-ayVFHdtZ+hsq1t2Dy24wCmGXGe4q9Gu3smhLYALJrr473ZH27MsnSL+LKUlimp4BWJqMDMLmPpx/Q9R3OAlL4g==", - "dev": true, - "license": "MIT", - "dependencies": { - "eslint-visitor-keys": "^3.4.3" - }, - "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - }, - "peerDependencies": { - "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" - } - }, - "node_modules/@eslint-community/eslint-utils/node_modules/eslint-visitor-keys": { - "version": "3.4.3", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", - "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/@eslint-community/regexpp": { - "version": "4.12.2", - "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz", - "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^12.0.0 || ^14.0.0 || >=16.0.0" - } - }, - "node_modules/@eslint/config-array": { - "version": "0.21.1", - "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.21.1.tgz", - "integrity": "sha512-aw1gNayWpdI/jSYVgzN5pL0cfzU02GT3NBpeT/DXbx1/1x7ZKxFPd9bwrzygx/qiwIQiJ1sw/zD8qY/kRvlGHA==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "@eslint/object-schema": "^2.1.7", - "debug": "^4.3.1", - "minimatch": "^3.1.2" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - } - }, - "node_modules/@eslint/config-helpers": { - "version": "0.4.2", - "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.4.2.tgz", - "integrity": "sha512-gBrxN88gOIf3R7ja5K9slwNayVcZgK6SOUORm2uBzTeIEfeVaIhOpCtTox3P6R7o2jLFwLFTLnC7kU/RGcYEgw==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "@eslint/core": "^0.17.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - } - }, - "node_modules/@eslint/core": { - "version": "0.17.0", - "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.17.0.tgz", - "integrity": "sha512-yL/sLrpmtDaFEiUj1osRP4TI2MDz1AddJL+jZ7KSqvBuliN4xqYY54IfdN8qD8Toa6g1iloph1fxQNkjOxrrpQ==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "@types/json-schema": "^7.0.15" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - } - }, - "node_modules/@eslint/eslintrc": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.3.1.tgz", - "integrity": "sha512-gtF186CXhIl1p4pJNGZw8Yc6RlshoePRvE0X91oPGb3vZ8pM3qOS9W9NGPat9LziaBV7XrJWGylNQXkGcnM3IQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "ajv": "^6.12.4", - "debug": "^4.3.2", - "espree": "^10.0.1", - "globals": "^14.0.0", - "ignore": "^5.2.0", - "import-fresh": "^3.2.1", - "js-yaml": "^4.1.0", - "minimatch": "^3.1.2", - "strip-json-comments": "^3.1.1" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/@eslint/eslintrc/node_modules/globals": { - "version": "14.0.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-14.0.0.tgz", - "integrity": "sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/@eslint/js": { - "version": "9.39.1", - "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.39.1.tgz", - "integrity": "sha512-S26Stp4zCy88tH94QbBv3XCuzRQiZ9yXofEILmglYTh/Ug/a9/umqvgFtYBAo3Lp0nsI/5/qH1CCrbdK3AP1Tw==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://eslint.org/donate" - } - }, - "node_modules/@eslint/object-schema": { - "version": "2.1.7", - "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.7.tgz", - "integrity": "sha512-VtAOaymWVfZcmZbp6E2mympDIHvyjXs/12LqWYjVw6qjrfF+VK+fyG33kChz3nnK+SU5/NeHOqrTEHS8sXO3OA==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - } - }, - "node_modules/@eslint/plugin-kit": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.4.1.tgz", - "integrity": "sha512-43/qtrDUokr7LJqoF2c3+RInu/t4zfrpYdoSDfYyhg52rwLV6TnOvdG4fXm7IkSB3wErkcmJS9iEhjVtOSEjjA==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "@eslint/core": "^0.17.0", - "levn": "^0.4.1" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - } - }, - "node_modules/@humanfs/core": { - "version": "0.19.1", - "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.1.tgz", - "integrity": "sha512-5DyQ4+1JEUzejeK1JGICcideyfUbGixgS9jNgex5nqkW+cY7WZhxBigmieN5Qnw9ZosSNVC9KQKyb+GUaGyKUA==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": ">=18.18.0" - } - }, - "node_modules/@humanfs/node": { - "version": "0.16.7", - "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.7.tgz", - "integrity": "sha512-/zUx+yOsIrG4Y43Eh2peDeKCxlRt/gET6aHfaKpuq267qXdYDFViVHfMaLyygZOnl0kGWxFIgsBy8QFuTLUXEQ==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "@humanfs/core": "^0.19.1", - "@humanwhocodes/retry": "^0.4.0" - }, - "engines": { - "node": ">=18.18.0" - } - }, - "node_modules/@humanwhocodes/module-importer": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", - "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": ">=12.22" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/nzakas" - } - }, - "node_modules/@humanwhocodes/retry": { - "version": "0.4.3", - "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.4.3.tgz", - "integrity": "sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": ">=18.18" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/nzakas" - } - }, - "node_modules/@iconify/types": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/@iconify/types/-/types-2.0.0.tgz", - "integrity": "sha512-+wluvCrRhXrhyOmRDJ3q8mux9JkKy5SJ/v8ol2tu4FVjyYvtEzkc/3pK15ET6RKg4b4w4BmTk1+gsCUhf21Ykg==", - "license": "MIT" - }, - "node_modules/@iconify/utils": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/@iconify/utils/-/utils-3.0.2.tgz", - "integrity": "sha512-EfJS0rLfVuRuJRn4psJHtK2A9TqVnkxPpHY6lYHiB9+8eSuudsxbwMiavocG45ujOo6FJ+CIRlRnlOGinzkaGQ==", - "license": "MIT", - "dependencies": { - "@antfu/install-pkg": "^1.1.0", - "@antfu/utils": "^9.2.0", - "@iconify/types": "^2.0.0", - "debug": "^4.4.1", - "globals": "^15.15.0", - "kolorist": "^1.8.0", - "local-pkg": "^1.1.1", - "mlly": "^1.7.4" - } - }, - "node_modules/@iconify/utils/node_modules/globals": { - "version": "15.15.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-15.15.0.tgz", - "integrity": "sha512-7ACyT3wmyp3I61S4fG682L0VA2RGD9otkqGJIwNUMF1SWUombIIk+af1unuDYgMm082aHYwD+mzJvv9Iu8dsgg==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/@isaacs/cliui": { - "version": "8.0.2", - "resolved": "https://registry.npmjs.org/@isaacs/cliui/-/cliui-8.0.2.tgz", - "integrity": "sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==", - "dev": true, - "license": "ISC", - "dependencies": { - "string-width": "^5.1.2", - "string-width-cjs": "npm:string-width@^4.2.0", - "strip-ansi": "^7.0.1", - "strip-ansi-cjs": "npm:strip-ansi@^6.0.1", - "wrap-ansi": "^8.1.0", - "wrap-ansi-cjs": "npm:wrap-ansi@^7.0.0" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/@istanbuljs/schema": { - "version": "0.1.3", - "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz", - "integrity": "sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/@jridgewell/gen-mapping": { - "version": "0.3.13", - "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz", - "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@jridgewell/sourcemap-codec": "^1.5.0", - "@jridgewell/trace-mapping": "^0.3.24" - } - }, - "node_modules/@jridgewell/remapping": { - "version": "2.3.5", - "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz", - "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@jridgewell/gen-mapping": "^0.3.5", - "@jridgewell/trace-mapping": "^0.3.24" - } - }, - "node_modules/@jridgewell/resolve-uri": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", - "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6.0.0" - } - }, - "node_modules/@jridgewell/sourcemap-codec": { - "version": "1.5.5", - "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz", - "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==", - "dev": true, - "license": "MIT" - }, - "node_modules/@jridgewell/trace-mapping": { - "version": "0.3.31", - "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz", - "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@jridgewell/resolve-uri": "^3.1.0", - "@jridgewell/sourcemap-codec": "^1.4.14" - } - }, - "node_modules/@kurkle/color": { - "version": "0.3.4", - "resolved": "https://registry.npmjs.org/@kurkle/color/-/color-0.3.4.tgz", - "integrity": "sha512-M5UknZPHRu3DEDWoipU6sE8PdkZ6Z/S+v4dD+Ke8IaNlpdSQah50lz1KtcFBa2vsdOnwbbnxJwVM4wty6udA5w==", - "license": "MIT" - }, - "node_modules/@marsidev/react-turnstile": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/@marsidev/react-turnstile/-/react-turnstile-1.4.1.tgz", - "integrity": "sha512-1jE0IjvB8z+q1NFRs3149gXzXwIzXQWqQjn9fmAr13BiE3RYLWck5Me6flHYE90shW5L12Jkm6R1peS1OnA9oQ==", - "license": "MIT", - "peerDependencies": { - "react": "^17.0.2 || ^18.0.0 || ^19.0", - "react-dom": "^17.0.2 || ^18.0.0 || ^19.0" - } - }, - "node_modules/@mermaid-js/parser": { - "version": "0.6.3", - "resolved": "https://registry.npmjs.org/@mermaid-js/parser/-/parser-0.6.3.tgz", - "integrity": "sha512-lnjOhe7zyHjc+If7yT4zoedx2vo4sHaTmtkl1+or8BRTnCtDmcTpAjpzDSfCZrshM5bCoz0GyidzadJAH1xobA==", - "license": "MIT", - "dependencies": { - "langium": "3.3.1" - } - }, - "node_modules/@napi-rs/canvas": { - "version": "0.1.88", - "resolved": "https://registry.npmjs.org/@napi-rs/canvas/-/canvas-0.1.88.tgz", - "integrity": "sha512-/p08f93LEbsL5mDZFQ3DBxcPv/I4QG9EDYRRq1WNlCOXVfAHBTHMSVMwxlqG/AtnSfUr9+vgfN7MKiyDo0+Weg==", - "license": "MIT", - "optional": true, - "workspaces": [ - "e2e/*" - ], - "engines": { - "node": ">= 10" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/Brooooooklyn" - }, - "optionalDependencies": { - "@napi-rs/canvas-android-arm64": "0.1.88", - "@napi-rs/canvas-darwin-arm64": "0.1.88", - "@napi-rs/canvas-darwin-x64": "0.1.88", - "@napi-rs/canvas-linux-arm-gnueabihf": "0.1.88", - "@napi-rs/canvas-linux-arm64-gnu": "0.1.88", - "@napi-rs/canvas-linux-arm64-musl": "0.1.88", - "@napi-rs/canvas-linux-riscv64-gnu": "0.1.88", - "@napi-rs/canvas-linux-x64-gnu": "0.1.88", - "@napi-rs/canvas-linux-x64-musl": "0.1.88", - "@napi-rs/canvas-win32-arm64-msvc": "0.1.88", - "@napi-rs/canvas-win32-x64-msvc": "0.1.88" - } - }, - "node_modules/@napi-rs/canvas-android-arm64": { - "version": "0.1.88", - "resolved": "https://registry.npmjs.org/@napi-rs/canvas-android-arm64/-/canvas-android-arm64-0.1.88.tgz", - "integrity": "sha512-KEaClPnZuVxJ8smUWjV1wWFkByBO/D+vy4lN+Dm5DFH514oqwukxKGeck9xcKJhaWJGjfruGmYGiwRe//+/zQQ==", - "cpu": [ - "arm64" - ], - "license": "MIT", - "optional": true, - "os": [ - "android" - ], - "engines": { - "node": ">= 10" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/Brooooooklyn" - } - }, - "node_modules/@napi-rs/canvas-darwin-arm64": { - "version": "0.1.88", - "resolved": "https://registry.npmjs.org/@napi-rs/canvas-darwin-arm64/-/canvas-darwin-arm64-0.1.88.tgz", - "integrity": "sha512-Xgywz0dDxOKSgx3eZnK85WgGMmGrQEW7ZLA/E7raZdlEE+xXCozobgqz2ZvYigpB6DJFYkqnwHjqCOTSDGlFdg==", - "cpu": [ - "arm64" - ], - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">= 10" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/Brooooooklyn" - } - }, - "node_modules/@napi-rs/canvas-darwin-x64": { - "version": "0.1.88", - "resolved": "https://registry.npmjs.org/@napi-rs/canvas-darwin-x64/-/canvas-darwin-x64-0.1.88.tgz", - "integrity": "sha512-Yz4wSCIQOUgNucgk+8NFtQxQxZV5NO8VKRl9ePKE6XoNyNVC8JDqtvhh3b3TPqKK8W5p2EQpAr1rjjm0mfBxdg==", - "cpu": [ - "x64" - ], - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">= 10" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/Brooooooklyn" - } - }, - "node_modules/@napi-rs/canvas-linux-arm-gnueabihf": { - "version": "0.1.88", - "resolved": "https://registry.npmjs.org/@napi-rs/canvas-linux-arm-gnueabihf/-/canvas-linux-arm-gnueabihf-0.1.88.tgz", - "integrity": "sha512-9gQM2SlTo76hYhxHi2XxWTAqpTOb+JtxMPEIr+H5nAhHhyEtNmTSDRtz93SP7mGd2G3Ojf2oF5tP9OdgtgXyKg==", - "cpu": [ - "arm" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/Brooooooklyn" - } - }, - "node_modules/@napi-rs/canvas-linux-arm64-gnu": { - "version": "0.1.88", - "resolved": "https://registry.npmjs.org/@napi-rs/canvas-linux-arm64-gnu/-/canvas-linux-arm64-gnu-0.1.88.tgz", - "integrity": "sha512-7qgaOBMXuVRk9Fzztzr3BchQKXDxGbY+nwsovD3I/Sx81e+sX0ReEDYHTItNb0Je4NHbAl7D0MKyd4SvUc04sg==", - "cpu": [ - "arm64" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/Brooooooklyn" - } - }, - "node_modules/@napi-rs/canvas-linux-arm64-musl": { - "version": "0.1.88", - "resolved": "https://registry.npmjs.org/@napi-rs/canvas-linux-arm64-musl/-/canvas-linux-arm64-musl-0.1.88.tgz", - "integrity": "sha512-kYyNrUsHLkoGHBc77u4Unh067GrfiCUMbGHC2+OTxbeWfZkPt2o32UOQkhnSswKd9Fko/wSqqGkY956bIUzruA==", - "cpu": [ - "arm64" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/Brooooooklyn" - } - }, - "node_modules/@napi-rs/canvas-linux-riscv64-gnu": { - "version": "0.1.88", - "resolved": "https://registry.npmjs.org/@napi-rs/canvas-linux-riscv64-gnu/-/canvas-linux-riscv64-gnu-0.1.88.tgz", - "integrity": "sha512-HVuH7QgzB0yavYdNZDRyAsn/ejoXB0hn8twwFnOqUbCCdkV+REna7RXjSR7+PdfW0qMQ2YYWsLvVBT5iL/mGpw==", - "cpu": [ - "riscv64" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/Brooooooklyn" - } - }, - "node_modules/@napi-rs/canvas-linux-x64-gnu": { - "version": "0.1.88", - "resolved": "https://registry.npmjs.org/@napi-rs/canvas-linux-x64-gnu/-/canvas-linux-x64-gnu-0.1.88.tgz", - "integrity": "sha512-hvcvKIcPEQrvvJtJnwD35B3qk6umFJ8dFIr8bSymfrSMem0EQsfn1ztys8ETIFndTwdNWJKWluvxztA41ivsEw==", - "cpu": [ - "x64" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/Brooooooklyn" - } - }, - "node_modules/@napi-rs/canvas-linux-x64-musl": { - "version": "0.1.88", - "resolved": "https://registry.npmjs.org/@napi-rs/canvas-linux-x64-musl/-/canvas-linux-x64-musl-0.1.88.tgz", - "integrity": "sha512-eSMpGYY2xnZSQ6UxYJ6plDboxq4KeJ4zT5HaVkUnbObNN6DlbJe0Mclh3wifAmquXfrlgTZt6zhHsUgz++AK6g==", - "cpu": [ - "x64" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/Brooooooklyn" - } - }, - "node_modules/@napi-rs/canvas-win32-arm64-msvc": { - "version": "0.1.88", - "resolved": "https://registry.npmjs.org/@napi-rs/canvas-win32-arm64-msvc/-/canvas-win32-arm64-msvc-0.1.88.tgz", - "integrity": "sha512-qcIFfEgHrchyYqRrxsCeTQgpJZ/GqHiqPcU/Fvw/ARVlQeDX1VyFH+X+0gCR2tca6UJrq96vnW+5o7buCq+erA==", - "cpu": [ - "arm64" - ], - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">= 10" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/Brooooooklyn" - } - }, - "node_modules/@napi-rs/canvas-win32-x64-msvc": { - "version": "0.1.88", - "resolved": "https://registry.npmjs.org/@napi-rs/canvas-win32-x64-msvc/-/canvas-win32-x64-msvc-0.1.88.tgz", - "integrity": "sha512-ROVqbfS4QyZxYkqmaIBBpbz/BQvAR+05FXM5PAtTYVc0uyY8Y4BHJSMdGAaMf6TdIVRsQsiq+FG/dH9XhvWCFQ==", - "cpu": [ - "x64" - ], - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">= 10" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/Brooooooklyn" - } - }, - "node_modules/@nodelib/fs.scandir": { - "version": "2.1.5", - "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", - "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@nodelib/fs.stat": "2.0.5", - "run-parallel": "^1.1.9" - }, - "engines": { - "node": ">= 8" - } - }, - "node_modules/@nodelib/fs.stat": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", - "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 8" - } - }, - "node_modules/@nodelib/fs.walk": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", - "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@nodelib/fs.scandir": "2.1.5", - "fastq": "^1.6.0" - }, - "engines": { - "node": ">= 8" - } - }, - "node_modules/@pkgjs/parseargs": { - "version": "0.11.0", - "resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz", - "integrity": "sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==", - "dev": true, - "license": "MIT", - "optional": true, - "engines": { - "node": ">=14" - } - }, - "node_modules/@pkgr/core": { - "version": "0.2.9", - "resolved": "https://registry.npmjs.org/@pkgr/core/-/core-0.2.9.tgz", - "integrity": "sha512-QNqXyfVS2wm9hweSYD2O7F0G06uurj9kZ96TRQE5Y9hU7+tgdZwIkbAKc5Ocy1HxEY2kuDQa6cQ1WRs/O5LFKA==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^12.20.0 || ^14.18.0 || >=16.0.0" - }, - "funding": { - "url": "https://opencollective.com/pkgr" - } - }, - "node_modules/@radix-ui/react-compose-refs": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/@radix-ui/react-compose-refs/-/react-compose-refs-1.1.2.tgz", - "integrity": "sha512-z4eqJvfiNnFMHIIvXP3CY57y2WJs5g2v3X0zm9mEJkrkNv4rDxu+sg9Jh8EkXyeqBkB7SOcboo9dMVqhyrACIg==", - "license": "MIT", - "peerDependencies": { - "@types/react": "*", - "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" - }, - "peerDependenciesMeta": { - "@types/react": { - "optional": true - } - } - }, - "node_modules/@radix-ui/react-slot": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/@radix-ui/react-slot/-/react-slot-1.2.4.tgz", - "integrity": "sha512-Jl+bCv8HxKnlTLVrcDE8zTMJ09R9/ukw4qBs/oZClOfoQk/cOTbDn+NceXfV7j09YPVQUryJPHurafcSg6EVKA==", - "license": "MIT", - "dependencies": { - "@radix-ui/react-compose-refs": "1.1.2" - }, - "peerDependencies": { - "@types/react": "*", - "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" - }, - "peerDependenciesMeta": { - "@types/react": { - "optional": true - } - } - }, - "node_modules/@rolldown/pluginutils": { - "version": "1.0.0-beta.43", - "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-beta.43.tgz", - "integrity": "sha512-5Uxg7fQUCmfhax7FJke2+8B6cqgeUJUD9o2uXIKXhD+mG0mL6NObmVoi9wXEU1tY89mZKgAYA6fTbftx3q2ZPQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/@rollup/rollup-android-arm-eabi": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.53.2.tgz", - "integrity": "sha512-yDPzwsgiFO26RJA4nZo8I+xqzh7sJTZIWQOxn+/XOdPE31lAvLIYCKqjV+lNH/vxE2L2iH3plKxDCRK6i+CwhA==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ] - }, - "node_modules/@rollup/rollup-android-arm64": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.53.2.tgz", - "integrity": "sha512-k8FontTxIE7b0/OGKeSN5B6j25EuppBcWM33Z19JoVT7UTXFSo3D9CdU39wGTeb29NO3XxpMNauh09B+Ibw+9g==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ] - }, - "node_modules/@rollup/rollup-darwin-arm64": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.53.2.tgz", - "integrity": "sha512-A6s4gJpomNBtJ2yioj8bflM2oogDwzUiMl2yNJ2v9E7++sHrSrsQ29fOfn5DM/iCzpWcebNYEdXpaK4tr2RhfQ==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ] - }, - "node_modules/@rollup/rollup-darwin-x64": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.53.2.tgz", - "integrity": "sha512-e6XqVmXlHrBlG56obu9gDRPW3O3hLxpwHpLsBJvuI8qqnsrtSZ9ERoWUXtPOkY8c78WghyPHZdmPhHLWNdAGEw==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ] - }, - "node_modules/@rollup/rollup-freebsd-arm64": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.53.2.tgz", - "integrity": "sha512-v0E9lJW8VsrwPux5Qe5CwmH/CF/2mQs6xU1MF3nmUxmZUCHazCjLgYvToOk+YuuUqLQBio1qkkREhxhc656ViA==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ] - }, - "node_modules/@rollup/rollup-freebsd-x64": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.53.2.tgz", - "integrity": "sha512-ClAmAPx3ZCHtp6ysl4XEhWU69GUB1D+s7G9YjHGhIGCSrsg00nEGRRZHmINYxkdoJehde8VIsDC5t9C0gb6yqA==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ] - }, - "node_modules/@rollup/rollup-linux-arm-gnueabihf": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.53.2.tgz", - "integrity": "sha512-EPlb95nUsz6Dd9Qy13fI5kUPXNSljaG9FiJ4YUGU1O/Q77i5DYFW5KR8g1OzTcdZUqQQ1KdDqsTohdFVwCwjqg==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-arm-musleabihf": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.53.2.tgz", - "integrity": "sha512-BOmnVW+khAUX+YZvNfa0tGTEMVVEerOxN0pDk2E6N6DsEIa2Ctj48FOMfNDdrwinocKaC7YXUZ1pHlKpnkja/Q==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-arm64-gnu": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.53.2.tgz", - "integrity": "sha512-Xt2byDZ+6OVNuREgBXr4+CZDJtrVso5woFtpKdGPhpTPHcNG7D8YXeQzpNbFRxzTVqJf7kvPMCub/pcGUWgBjA==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-arm64-musl": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.53.2.tgz", - "integrity": "sha512-+LdZSldy/I9N8+klim/Y1HsKbJ3BbInHav5qE9Iy77dtHC/pibw1SR/fXlWyAk0ThnpRKoODwnAuSjqxFRDHUQ==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-loong64-gnu": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.53.2.tgz", - "integrity": "sha512-8ms8sjmyc1jWJS6WdNSA23rEfdjWB30LH8Wqj0Cqvv7qSHnvw6kgMMXRdop6hkmGPlyYBdRPkjJnj3KCUHV/uQ==", - "cpu": [ - "loong64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-ppc64-gnu": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.53.2.tgz", - "integrity": "sha512-3HRQLUQbpBDMmzoxPJYd3W6vrVHOo2cVW8RUo87Xz0JPJcBLBr5kZ1pGcQAhdZgX9VV7NbGNipah1omKKe23/g==", - "cpu": [ - "ppc64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-riscv64-gnu": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.53.2.tgz", - "integrity": "sha512-fMjKi+ojnmIvhk34gZP94vjogXNNUKMEYs+EDaB/5TG/wUkoeua7p7VCHnE6T2Tx+iaghAqQX8teQzcvrYpaQA==", - "cpu": [ - "riscv64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-riscv64-musl": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.53.2.tgz", - "integrity": "sha512-XuGFGU+VwUUV5kLvoAdi0Wz5Xbh2SrjIxCtZj6Wq8MDp4bflb/+ThZsVxokM7n0pcbkEr2h5/pzqzDYI7cCgLQ==", - "cpu": [ - "riscv64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-s390x-gnu": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.53.2.tgz", - "integrity": "sha512-w6yjZF0P+NGzWR3AXWX9zc0DNEGdtvykB03uhonSHMRa+oWA6novflo2WaJr6JZakG2ucsyb+rvhrKac6NIy+w==", - "cpu": [ - "s390x" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-x64-gnu": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.53.2.tgz", - "integrity": "sha512-yo8d6tdfdeBArzC7T/PnHd7OypfI9cbuZzPnzLJIyKYFhAQ8SvlkKtKBMbXDxe1h03Rcr7u++nFS7tqXz87Gtw==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-x64-musl": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.53.2.tgz", - "integrity": "sha512-ah59c1YkCxKExPP8O9PwOvs+XRLKwh/mV+3YdKqQ5AMQ0r4M4ZDuOrpWkUaqO7fzAHdINzV9tEVu8vNw48z0lA==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-openharmony-arm64": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.53.2.tgz", - "integrity": "sha512-4VEd19Wmhr+Zy7hbUsFZ6YXEiP48hE//KPLCSVNY5RMGX2/7HZ+QkN55a3atM1C/BZCGIgqN+xrVgtdak2S9+A==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "openharmony" - ] - }, - "node_modules/@rollup/rollup-win32-arm64-msvc": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.53.2.tgz", - "integrity": "sha512-IlbHFYc/pQCgew/d5fslcy1KEaYVCJ44G8pajugd8VoOEI8ODhtb/j8XMhLpwHCMB3yk2J07ctup10gpw2nyMA==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ] - }, - "node_modules/@rollup/rollup-win32-ia32-msvc": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.53.2.tgz", - "integrity": "sha512-lNlPEGgdUfSzdCWU176ku/dQRnA7W+Gp8d+cWv73jYrb8uT7HTVVxq62DUYxjbaByuf1Yk0RIIAbDzp+CnOTFg==", - "cpu": [ - "ia32" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ] - }, - "node_modules/@rollup/rollup-win32-x64-gnu": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.53.2.tgz", - "integrity": "sha512-S6YojNVrHybQis2lYov1sd+uj7K0Q05NxHcGktuMMdIQ2VixGwAfbJ23NnlvvVV1bdpR2m5MsNBViHJKcA4ADw==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ] - }, - "node_modules/@rollup/rollup-win32-x64-msvc": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.53.2.tgz", - "integrity": "sha512-k+/Rkcyx//P6fetPoLMb8pBeqJBNGx81uuf7iljX9++yNBVRDQgD04L+SVXmXmh5ZP4/WOp4mWF0kmi06PW2tA==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ] - }, - "node_modules/@supabase/auth-js": { - "version": "2.81.1", - "resolved": "https://registry.npmjs.org/@supabase/auth-js/-/auth-js-2.81.1.tgz", - "integrity": "sha512-K20GgiSm9XeRLypxYHa5UCnybWc2K0ok0HLbqCej/wRxDpJxToXNOwKt0l7nO8xI1CyQ+GrNfU6bcRzvdbeopQ==", - "license": "MIT", - "dependencies": { - "tslib": "2.8.1" - }, - "engines": { - "node": ">=20.0.0" - } - }, - "node_modules/@supabase/functions-js": { - "version": "2.81.1", - "resolved": "https://registry.npmjs.org/@supabase/functions-js/-/functions-js-2.81.1.tgz", - "integrity": "sha512-sYgSO3mlgL0NvBFS3oRfCK4OgKGQwuOWJLzfPyWg0k8MSxSFSDeN/JtrDJD5GQrxskP6c58+vUzruBJQY78AqQ==", - "license": "MIT", - "dependencies": { - "tslib": "2.8.1" - }, - "engines": { - "node": ">=20.0.0" - } - }, - "node_modules/@supabase/postgrest-js": { - "version": "2.81.1", - "resolved": "https://registry.npmjs.org/@supabase/postgrest-js/-/postgrest-js-2.81.1.tgz", - "integrity": "sha512-DePpUTAPXJyBurQ4IH2e42DWoA+/Qmr5mbgY4B6ZcxVc/ZUKfTVK31BYIFBATMApWraFc8Q/Sg+yxtfJ3E0wSg==", - "license": "MIT", - "dependencies": { - "tslib": "2.8.1" - }, - "engines": { - "node": ">=20.0.0" - } - }, - "node_modules/@supabase/realtime-js": { - "version": "2.81.1", - "resolved": "https://registry.npmjs.org/@supabase/realtime-js/-/realtime-js-2.81.1.tgz", - "integrity": "sha512-ViQ+Kxm8BuUP/TcYmH9tViqYKGSD1LBjdqx2p5J+47RES6c+0QHedM0PPAjthMdAHWyb2LGATE9PD2++2rO/tw==", - "license": "MIT", - "dependencies": { - "@types/phoenix": "^1.6.6", - "@types/ws": "^8.18.1", - "tslib": "2.8.1", - "ws": "^8.18.2" - }, - "engines": { - "node": ">=20.0.0" - } - }, - "node_modules/@supabase/storage-js": { - "version": "2.81.1", - "resolved": "https://registry.npmjs.org/@supabase/storage-js/-/storage-js-2.81.1.tgz", - "integrity": "sha512-UNmYtjnZnhouqnbEMC1D5YJot7y0rIaZx7FG2Fv8S3hhNjcGVvO+h9We/tggi273BFkiahQPS/uRsapo1cSapw==", - "license": "MIT", - "dependencies": { - "tslib": "2.8.1" - }, - "engines": { - "node": ">=20.0.0" - } - }, - "node_modules/@supabase/supabase-js": { - "version": "2.81.1", - "resolved": "https://registry.npmjs.org/@supabase/supabase-js/-/supabase-js-2.81.1.tgz", - "integrity": "sha512-KSdY7xb2L0DlLmlYzIOghdw/na4gsMcqJ8u4sD6tOQJr+x3hLujU9s4R8N3ob84/1bkvpvlU5PYKa1ae+OICnw==", - "license": "MIT", - "dependencies": { - "@supabase/auth-js": "2.81.1", - "@supabase/functions-js": "2.81.1", - "@supabase/postgrest-js": "2.81.1", - "@supabase/realtime-js": "2.81.1", - "@supabase/storage-js": "2.81.1" - }, - "engines": { - "node": ">=20.0.0" - } - }, - "node_modules/@types/babel__core": { - "version": "7.20.5", - "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.20.5.tgz", - "integrity": "sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/parser": "^7.20.7", - "@babel/types": "^7.20.7", - "@types/babel__generator": "*", - "@types/babel__template": "*", - "@types/babel__traverse": "*" - } - }, - "node_modules/@types/babel__generator": { - "version": "7.27.0", - "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.27.0.tgz", - "integrity": "sha512-ufFd2Xi92OAVPYsy+P4n7/U7e68fex0+Ee8gSG9KX7eo084CWiQ4sdxktvdl0bOPupXtVJPY19zk6EwWqUQ8lg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/types": "^7.0.0" - } - }, - "node_modules/@types/babel__template": { - "version": "7.4.4", - "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.4.tgz", - "integrity": "sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/parser": "^7.1.0", - "@babel/types": "^7.0.0" - } - }, - "node_modules/@types/babel__traverse": { - "version": "7.28.0", - "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.28.0.tgz", - "integrity": "sha512-8PvcXf70gTDZBgt9ptxJ8elBeBjcLOAcOtoO/mPJjtji1+CdGbHgm77om1GrsPxsiE+uXIpNSK64UYaIwQXd4Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/types": "^7.28.2" - } - }, - "node_modules/@types/chai": { - "version": "5.2.3", - "resolved": "https://registry.npmjs.org/@types/chai/-/chai-5.2.3.tgz", - "integrity": "sha512-Mw558oeA9fFbv65/y4mHtXDs9bPnFMZAL/jxdPFUpOHHIXX91mcgEHbS5Lahr+pwZFR8A7GQleRWeI6cGFC2UA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/deep-eql": "*", - "assertion-error": "^2.0.1" - } - }, - "node_modules/@types/d3": { - "version": "7.4.3", - "resolved": "https://registry.npmjs.org/@types/d3/-/d3-7.4.3.tgz", - "integrity": "sha512-lZXZ9ckh5R8uiFVt8ogUNf+pIrK4EsWrx2Np75WvF/eTpJ0FMHNhjXk8CKEx/+gpHbNQyJWehbFaTvqmHWB3ww==", - "license": "MIT", - "dependencies": { - "@types/d3-array": "*", - "@types/d3-axis": "*", - "@types/d3-brush": "*", - "@types/d3-chord": "*", - "@types/d3-color": "*", - "@types/d3-contour": "*", - "@types/d3-delaunay": "*", - "@types/d3-dispatch": "*", - "@types/d3-drag": "*", - "@types/d3-dsv": "*", - "@types/d3-ease": "*", - "@types/d3-fetch": "*", - "@types/d3-force": "*", - "@types/d3-format": "*", - "@types/d3-geo": "*", - "@types/d3-hierarchy": "*", - "@types/d3-interpolate": "*", - "@types/d3-path": "*", - "@types/d3-polygon": "*", - "@types/d3-quadtree": "*", - "@types/d3-random": "*", - "@types/d3-scale": "*", - "@types/d3-scale-chromatic": "*", - "@types/d3-selection": "*", - "@types/d3-shape": "*", - "@types/d3-time": "*", - "@types/d3-time-format": "*", - "@types/d3-timer": "*", - "@types/d3-transition": "*", - "@types/d3-zoom": "*" - } - }, - "node_modules/@types/d3-array": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/@types/d3-array/-/d3-array-3.2.2.tgz", - "integrity": "sha512-hOLWVbm7uRza0BYXpIIW5pxfrKe0W+D5lrFiAEYR+pb6w3N2SwSMaJbXdUfSEv+dT4MfHBLtn5js0LAWaO6otw==", - "license": "MIT" - }, - "node_modules/@types/d3-axis": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/@types/d3-axis/-/d3-axis-3.0.6.tgz", - "integrity": "sha512-pYeijfZuBd87T0hGn0FO1vQ/cgLk6E1ALJjfkC0oJ8cbwkZl3TpgS8bVBLZN+2jjGgg38epgxb2zmoGtSfvgMw==", - "license": "MIT", - "dependencies": { - "@types/d3-selection": "*" - } - }, - "node_modules/@types/d3-brush": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/@types/d3-brush/-/d3-brush-3.0.6.tgz", - "integrity": "sha512-nH60IZNNxEcrh6L1ZSMNA28rj27ut/2ZmI3r96Zd+1jrZD++zD3LsMIjWlvg4AYrHn/Pqz4CF3veCxGjtbqt7A==", - "license": "MIT", - "dependencies": { - "@types/d3-selection": "*" - } - }, - "node_modules/@types/d3-chord": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/@types/d3-chord/-/d3-chord-3.0.6.tgz", - "integrity": "sha512-LFYWWd8nwfwEmTZG9PfQxd17HbNPksHBiJHaKuY1XeqscXacsS2tyoo6OdRsjf+NQYeB6XrNL3a25E3gH69lcg==", - "license": "MIT" - }, - "node_modules/@types/d3-color": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/@types/d3-color/-/d3-color-3.1.3.tgz", - "integrity": "sha512-iO90scth9WAbmgv7ogoq57O9YpKmFBbmoEoCHDB2xMBY0+/KVrqAaCDyCE16dUspeOvIxFFRI+0sEtqDqy2b4A==", - "license": "MIT" - }, - "node_modules/@types/d3-contour": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/@types/d3-contour/-/d3-contour-3.0.6.tgz", - "integrity": "sha512-BjzLgXGnCWjUSYGfH1cpdo41/hgdWETu4YxpezoztawmqsvCeep+8QGfiY6YbDvfgHz/DkjeIkkZVJavB4a3rg==", - "license": "MIT", - "dependencies": { - "@types/d3-array": "*", - "@types/geojson": "*" - } - }, - "node_modules/@types/d3-delaunay": { - "version": "6.0.4", - "resolved": "https://registry.npmjs.org/@types/d3-delaunay/-/d3-delaunay-6.0.4.tgz", - "integrity": "sha512-ZMaSKu4THYCU6sV64Lhg6qjf1orxBthaC161plr5KuPHo3CNm8DTHiLw/5Eq2b6TsNP0W0iJrUOFscY6Q450Hw==", - "license": "MIT" - }, - "node_modules/@types/d3-dispatch": { - "version": "3.0.7", - "resolved": "https://registry.npmjs.org/@types/d3-dispatch/-/d3-dispatch-3.0.7.tgz", - "integrity": "sha512-5o9OIAdKkhN1QItV2oqaE5KMIiXAvDWBDPrD85e58Qlz1c1kI/J0NcqbEG88CoTwJrYe7ntUCVfeUl2UJKbWgA==", - "license": "MIT" - }, - "node_modules/@types/d3-drag": { - "version": "3.0.7", - "resolved": "https://registry.npmjs.org/@types/d3-drag/-/d3-drag-3.0.7.tgz", - "integrity": "sha512-HE3jVKlzU9AaMazNufooRJ5ZpWmLIoc90A37WU2JMmeq28w1FQqCZswHZ3xR+SuxYftzHq6WU6KJHvqxKzTxxQ==", - "license": "MIT", - "dependencies": { - "@types/d3-selection": "*" - } - }, - "node_modules/@types/d3-dsv": { - "version": "3.0.7", - "resolved": "https://registry.npmjs.org/@types/d3-dsv/-/d3-dsv-3.0.7.tgz", - "integrity": "sha512-n6QBF9/+XASqcKK6waudgL0pf/S5XHPPI8APyMLLUHd8NqouBGLsU8MgtO7NINGtPBtk9Kko/W4ea0oAspwh9g==", - "license": "MIT" - }, - "node_modules/@types/d3-ease": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/@types/d3-ease/-/d3-ease-3.0.2.tgz", - "integrity": "sha512-NcV1JjO5oDzoK26oMzbILE6HW7uVXOHLQvHshBUW4UMdZGfiY6v5BeQwh9a9tCzv+CeefZQHJt5SRgK154RtiA==", - "license": "MIT" - }, - "node_modules/@types/d3-fetch": { - "version": "3.0.7", - "resolved": "https://registry.npmjs.org/@types/d3-fetch/-/d3-fetch-3.0.7.tgz", - "integrity": "sha512-fTAfNmxSb9SOWNB9IoG5c8Hg6R+AzUHDRlsXsDZsNp6sxAEOP0tkP3gKkNSO/qmHPoBFTxNrjDprVHDQDvo5aA==", - "license": "MIT", - "dependencies": { - "@types/d3-dsv": "*" - } - }, - "node_modules/@types/d3-force": { - "version": "3.0.10", - "resolved": "https://registry.npmjs.org/@types/d3-force/-/d3-force-3.0.10.tgz", - "integrity": "sha512-ZYeSaCF3p73RdOKcjj+swRlZfnYpK1EbaDiYICEEp5Q6sUiqFaFQ9qgoshp5CzIyyb/yD09kD9o2zEltCexlgw==", - "license": "MIT" - }, - "node_modules/@types/d3-format": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/@types/d3-format/-/d3-format-3.0.4.tgz", - "integrity": "sha512-fALi2aI6shfg7vM5KiR1wNJnZ7r6UuggVqtDA+xiEdPZQwy/trcQaHnwShLuLdta2rTymCNpxYTiMZX/e09F4g==", - "license": "MIT" - }, - "node_modules/@types/d3-geo": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/@types/d3-geo/-/d3-geo-3.1.0.tgz", - "integrity": "sha512-856sckF0oP/diXtS4jNsiQw/UuK5fQG8l/a9VVLeSouf1/PPbBE1i1W852zVwKwYCBkFJJB7nCFTbk6UMEXBOQ==", - "license": "MIT", - "dependencies": { - "@types/geojson": "*" - } - }, - "node_modules/@types/d3-hierarchy": { - "version": "3.1.7", - "resolved": "https://registry.npmjs.org/@types/d3-hierarchy/-/d3-hierarchy-3.1.7.tgz", - "integrity": "sha512-tJFtNoYBtRtkNysX1Xq4sxtjK8YgoWUNpIiUee0/jHGRwqvzYxkq0hGVbbOGSz+JgFxxRu4K8nb3YpG3CMARtg==", - "license": "MIT" - }, - "node_modules/@types/d3-interpolate": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/@types/d3-interpolate/-/d3-interpolate-3.0.4.tgz", - "integrity": "sha512-mgLPETlrpVV1YRJIglr4Ez47g7Yxjl1lj7YKsiMCb27VJH9W8NVM6Bb9d8kkpG/uAQS5AmbA48q2IAolKKo1MA==", - "license": "MIT", - "dependencies": { - "@types/d3-color": "*" - } - }, - "node_modules/@types/d3-path": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/@types/d3-path/-/d3-path-3.1.1.tgz", - "integrity": "sha512-VMZBYyQvbGmWyWVea0EHs/BwLgxc+MKi1zLDCONksozI4YJMcTt8ZEuIR4Sb1MMTE8MMW49v0IwI5+b7RmfWlg==", - "license": "MIT" - }, - "node_modules/@types/d3-polygon": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/@types/d3-polygon/-/d3-polygon-3.0.2.tgz", - "integrity": "sha512-ZuWOtMaHCkN9xoeEMr1ubW2nGWsp4nIql+OPQRstu4ypeZ+zk3YKqQT0CXVe/PYqrKpZAi+J9mTs05TKwjXSRA==", - "license": "MIT" - }, - "node_modules/@types/d3-quadtree": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/@types/d3-quadtree/-/d3-quadtree-3.0.6.tgz", - "integrity": "sha512-oUzyO1/Zm6rsxKRHA1vH0NEDG58HrT5icx/azi9MF1TWdtttWl0UIUsjEQBBh+SIkrpd21ZjEv7ptxWys1ncsg==", - "license": "MIT" - }, - "node_modules/@types/d3-random": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/@types/d3-random/-/d3-random-3.0.3.tgz", - "integrity": "sha512-Imagg1vJ3y76Y2ea0871wpabqp613+8/r0mCLEBfdtqC7xMSfj9idOnmBYyMoULfHePJyxMAw3nWhJxzc+LFwQ==", - "license": "MIT" - }, - "node_modules/@types/d3-scale": { - "version": "4.0.9", - "resolved": "https://registry.npmjs.org/@types/d3-scale/-/d3-scale-4.0.9.tgz", - "integrity": "sha512-dLmtwB8zkAeO/juAMfnV+sItKjlsw2lKdZVVy6LRr0cBmegxSABiLEpGVmSJJ8O08i4+sGR6qQtb6WtuwJdvVw==", - "license": "MIT", - "dependencies": { - "@types/d3-time": "*" - } - }, - "node_modules/@types/d3-scale-chromatic": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/@types/d3-scale-chromatic/-/d3-scale-chromatic-3.1.0.tgz", - "integrity": "sha512-iWMJgwkK7yTRmWqRB5plb1kadXyQ5Sj8V/zYlFGMUBbIPKQScw+Dku9cAAMgJG+z5GYDoMjWGLVOvjghDEFnKQ==", - "license": "MIT" - }, - "node_modules/@types/d3-selection": { - "version": "3.0.11", - "resolved": "https://registry.npmjs.org/@types/d3-selection/-/d3-selection-3.0.11.tgz", - "integrity": "sha512-bhAXu23DJWsrI45xafYpkQ4NtcKMwWnAC/vKrd2l+nxMFuvOT3XMYTIj2opv8vq8AO5Yh7Qac/nSeP/3zjTK0w==", - "license": "MIT" - }, - "node_modules/@types/d3-shape": { - "version": "3.1.7", - "resolved": "https://registry.npmjs.org/@types/d3-shape/-/d3-shape-3.1.7.tgz", - "integrity": "sha512-VLvUQ33C+3J+8p+Daf+nYSOsjB4GXp19/S/aGo60m9h1v6XaxjiT82lKVWJCfzhtuZ3yD7i/TPeC/fuKLLOSmg==", - "license": "MIT", - "dependencies": { - "@types/d3-path": "*" - } - }, - "node_modules/@types/d3-time": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/@types/d3-time/-/d3-time-3.0.4.tgz", - "integrity": "sha512-yuzZug1nkAAaBlBBikKZTgzCeA+k1uy4ZFwWANOfKw5z5LRhV0gNA7gNkKm7HoK+HRN0wX3EkxGk0fpbWhmB7g==", - "license": "MIT" - }, - "node_modules/@types/d3-time-format": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/@types/d3-time-format/-/d3-time-format-4.0.3.tgz", - "integrity": "sha512-5xg9rC+wWL8kdDj153qZcsJ0FWiFt0J5RB6LYUNZjwSnesfblqrI/bJ1wBdJ8OQfncgbJG5+2F+qfqnqyzYxyg==", - "license": "MIT" - }, - "node_modules/@types/d3-timer": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/@types/d3-timer/-/d3-timer-3.0.2.tgz", - "integrity": "sha512-Ps3T8E8dZDam6fUyNiMkekK3XUsaUEik+idO9/YjPtfj2qruF8tFBXS7XhtE4iIXBLxhmLjP3SXpLhVf21I9Lw==", - "license": "MIT" - }, - "node_modules/@types/d3-transition": { - "version": "3.0.9", - "resolved": "https://registry.npmjs.org/@types/d3-transition/-/d3-transition-3.0.9.tgz", - "integrity": "sha512-uZS5shfxzO3rGlu0cC3bjmMFKsXv+SmZZcgp0KD22ts4uGXp5EVYGzu/0YdwZeKmddhcAccYtREJKkPfXkZuCg==", - "license": "MIT", - "dependencies": { - "@types/d3-selection": "*" - } - }, - "node_modules/@types/d3-zoom": { - "version": "3.0.8", - "resolved": "https://registry.npmjs.org/@types/d3-zoom/-/d3-zoom-3.0.8.tgz", - "integrity": "sha512-iqMC4/YlFCSlO8+2Ii1GGGliCAY4XdeG748w5vQUbevlbDu0zSjH/+jojorQVBK/se0j6DUFNPBGSqD3YWYnDw==", - "license": "MIT", - "dependencies": { - "@types/d3-interpolate": "*", - "@types/d3-selection": "*" - } - }, - "node_modules/@types/dagre": { - "version": "0.7.53", - "resolved": "https://registry.npmjs.org/@types/dagre/-/dagre-0.7.53.tgz", - "integrity": "sha512-f4gkWqzPZvYmKhOsDnhq/R8mO4UMcKdxZo+i5SCkOU1wvGeHJeUXGIHeE9pnwGyPMDof1Vx5ZQo4nxpeg2TTVQ==", - "license": "MIT" - }, - "node_modules/@types/debug": { - "version": "4.1.12", - "resolved": "https://registry.npmjs.org/@types/debug/-/debug-4.1.12.tgz", - "integrity": "sha512-vIChWdVG3LG1SMxEvI/AK+FWJthlrqlTu7fbrlywTkkaONwk/UAGaULXRlf8vkzFBLVm0zkMdCquhL5aOjhXPQ==", - "license": "MIT", - "dependencies": { - "@types/ms": "*" - } - }, - "node_modules/@types/deep-eql": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/@types/deep-eql/-/deep-eql-4.0.2.tgz", - "integrity": "sha512-c9h9dVVMigMPc4bwTvC5dxqtqJZwQPePsWjPlpSOnojbor6pGqdk541lfA7AqFQr5pB1BRdq0juY9db81BwyFw==", - "dev": true, - "license": "MIT" - }, - "node_modules/@types/dompurify": { - "version": "3.0.5", - "resolved": "https://registry.npmjs.org/@types/dompurify/-/dompurify-3.0.5.tgz", - "integrity": "sha512-1Wg0g3BtQF7sSb27fJQAKck1HECM6zV1EB66j8JH9i3LCjYabJa0FSdiSgsD5K/RbrsR0SiraKacLB+T8ZVYAg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/trusted-types": "*" - } - }, - "node_modules/@types/estree": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz", - "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==", - "license": "MIT" - }, - "node_modules/@types/estree-jsx": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/@types/estree-jsx/-/estree-jsx-1.0.5.tgz", - "integrity": "sha512-52CcUVNFyfb1A2ALocQw/Dd1BQFNmSdkuC3BkZ6iqhdMfQz7JWOFRuJFloOzjk+6WijU56m9oKXFAXc7o3Towg==", - "license": "MIT", - "dependencies": { - "@types/estree": "*" - } - }, - "node_modules/@types/geojson": { - "version": "7946.0.16", - "resolved": "https://registry.npmjs.org/@types/geojson/-/geojson-7946.0.16.tgz", - "integrity": "sha512-6C8nqWur3j98U6+lXDfTUWIfgvZU+EumvpHKcYjujKH7woYyLj2sUmff0tRhrqM7BohUw7Pz3ZB1jj2gW9Fvmg==", - "license": "MIT" - }, - "node_modules/@types/hast": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/@types/hast/-/hast-3.0.4.tgz", - "integrity": "sha512-WPs+bbQw5aCj+x6laNGWLH3wviHtoCv/P3+otBhbOhJgG8qtpdAMlTCxLtsTWA7LH1Oh/bFCHsBn0TPS5m30EQ==", - "license": "MIT", - "dependencies": { - "@types/unist": "*" - } - }, - "node_modules/@types/json-schema": { - "version": "7.0.15", - "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", - "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", - "dev": true, - "license": "MIT" - }, - "node_modules/@types/mdast": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/@types/mdast/-/mdast-4.0.4.tgz", - "integrity": "sha512-kGaNbPh1k7AFzgpud/gMdvIm5xuECykRR+JnWKQno9TAXVa6WIVCGTPvYGekIDL4uwCZQSYbUxNBSb1aUo79oA==", - "license": "MIT", - "dependencies": { - "@types/unist": "*" - } - }, - "node_modules/@types/ms": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/@types/ms/-/ms-2.1.0.tgz", - "integrity": "sha512-GsCCIZDE/p3i96vtEqx+7dBUGXrc7zeSK3wwPHIaRThS+9OhWIXRqzs4d6k1SVU8g91DrNRWxWUGhp5KXQb2VA==", - "license": "MIT" - }, - "node_modules/@types/node": { - "version": "24.10.0", - "resolved": "https://registry.npmjs.org/@types/node/-/node-24.10.0.tgz", - "integrity": "sha512-qzQZRBqkFsYyaSWXuEHc2WR9c0a0CXwiE5FWUvn7ZM+vdy1uZLfCunD38UzhuB7YN/J11ndbDBcTmOdxJo9Q7A==", - "license": "MIT", - "dependencies": { - "undici-types": "~7.16.0" - } - }, - "node_modules/@types/phoenix": { - "version": "1.6.6", - "resolved": "https://registry.npmjs.org/@types/phoenix/-/phoenix-1.6.6.tgz", - "integrity": "sha512-PIzZZlEppgrpoT2QgbnDU+MMzuR6BbCjllj0bM70lWoejMeNJAxCchxnv7J3XFkI8MpygtRpzXrIlmWUBclP5A==", - "license": "MIT" - }, - "node_modules/@types/prismjs": { - "version": "1.26.5", - "resolved": "https://registry.npmjs.org/@types/prismjs/-/prismjs-1.26.5.tgz", - "integrity": "sha512-AUZTa7hQ2KY5L7AmtSiqxlhWxb4ina0yd8hNbl4TWuqnv/pFP0nDMb3YrfSBf4hJVGLh2YEIBfKaBW/9UEl6IQ==", - "license": "MIT" - }, - "node_modules/@types/react": { - "version": "19.2.2", - "resolved": "https://registry.npmjs.org/@types/react/-/react-19.2.2.tgz", - "integrity": "sha512-6mDvHUFSjyT2B2yeNx2nUgMxh9LtOWvkhIU3uePn2I2oyNymUAX1NIsdgviM4CH+JSrp2D2hsMvJOkxY+0wNRA==", - "license": "MIT", - "dependencies": { - "csstype": "^3.0.2" - } - }, - "node_modules/@types/react-dom": { - "version": "19.2.2", - "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-19.2.2.tgz", - "integrity": "sha512-9KQPoO6mZCi7jcIStSnlOWn2nEF3mNmyr3rIAsGnAbQKYbRLyqmeSc39EVgtxXVia+LMT8j3knZLAZAh+xLmrw==", - "dev": true, - "license": "MIT", - "peerDependencies": { - "@types/react": "^19.2.0" - } - }, - "node_modules/@types/react-syntax-highlighter": { - "version": "15.5.13", - "resolved": "https://registry.npmjs.org/@types/react-syntax-highlighter/-/react-syntax-highlighter-15.5.13.tgz", - "integrity": "sha512-uLGJ87j6Sz8UaBAooU0T6lWJ0dBmjZgN1PZTrj05TNql2/XpC6+4HhMT5syIdFUUt+FASfCeLLv4kBygNU+8qA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/react": "*" - } - }, - "node_modules/@types/trusted-types": { - "version": "2.0.7", - "resolved": "https://registry.npmjs.org/@types/trusted-types/-/trusted-types-2.0.7.tgz", - "integrity": "sha512-ScaPdn1dQczgbl0QFTeTOmVHFULt394XJgOQNoyVhZ6r2vLnMLJfBPd53SB52T/3G36VI1/g2MZaX0cwDuXsfw==", - "devOptional": true, - "license": "MIT" - }, - "node_modules/@types/unist": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/@types/unist/-/unist-3.0.3.tgz", - "integrity": "sha512-ko/gIFJRv177XgZsZcBwnqJN5x/Gien8qNOn0D5bQU/zAzVf9Zt3BlcUiLqhV9y4ARk0GbT3tnUiPNgnTXzc/Q==", - "license": "MIT" - }, - "node_modules/@types/ws": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.18.1.tgz", - "integrity": "sha512-ThVF6DCVhA8kUGy+aazFQ4kXQ7E1Ty7A3ypFOe0IcJV8O/M511G99AW24irKrW56Wt44yG9+ij8FaqoBGkuBXg==", - "license": "MIT", - "dependencies": { - "@types/node": "*" - } - }, - "node_modules/@typescript-eslint/eslint-plugin": { - "version": "8.46.4", - "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.46.4.tgz", - "integrity": "sha512-R48VhmTJqplNyDxCyqqVkFSZIx1qX6PzwqgcXn1olLrzxcSBDlOsbtcnQuQhNtnNiJ4Xe5gREI1foajYaYU2Vg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@eslint-community/regexpp": "^4.10.0", - "@typescript-eslint/scope-manager": "8.46.4", - "@typescript-eslint/type-utils": "8.46.4", - "@typescript-eslint/utils": "8.46.4", - "@typescript-eslint/visitor-keys": "8.46.4", - "graphemer": "^1.4.0", - "ignore": "^7.0.0", - "natural-compare": "^1.4.0", - "ts-api-utils": "^2.1.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "@typescript-eslint/parser": "^8.46.4", - "eslint": "^8.57.0 || ^9.0.0", - "typescript": ">=4.8.4 <6.0.0" - } - }, - "node_modules/@typescript-eslint/eslint-plugin/node_modules/ignore": { - "version": "7.0.5", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-7.0.5.tgz", - "integrity": "sha512-Hs59xBNfUIunMFgWAbGX5cq6893IbWg4KnrjbYwX3tx0ztorVgTDA6B2sxf8ejHJ4wz8BqGUMYlnzNBer5NvGg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 4" - } - }, - "node_modules/@typescript-eslint/parser": { - "version": "8.46.4", - "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.46.4.tgz", - "integrity": "sha512-tK3GPFWbirvNgsNKto+UmB/cRtn6TZfyw0D6IKrW55n6Vbs7KJoZtI//kpTKzE/DUmmnAFD8/Ca46s7Obs92/w==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/scope-manager": "8.46.4", - "@typescript-eslint/types": "8.46.4", - "@typescript-eslint/typescript-estree": "8.46.4", - "@typescript-eslint/visitor-keys": "8.46.4", - "debug": "^4.3.4" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0", - "typescript": ">=4.8.4 <6.0.0" - } - }, - "node_modules/@typescript-eslint/project-service": { - "version": "8.46.4", - "resolved": "https://registry.npmjs.org/@typescript-eslint/project-service/-/project-service-8.46.4.tgz", - "integrity": "sha512-nPiRSKuvtTN+no/2N1kt2tUh/HoFzeEgOm9fQ6XQk4/ApGqjx0zFIIaLJ6wooR1HIoozvj2j6vTi/1fgAz7UYQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/tsconfig-utils": "^8.46.4", - "@typescript-eslint/types": "^8.46.4", - "debug": "^4.3.4" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "typescript": ">=4.8.4 <6.0.0" - } - }, - "node_modules/@typescript-eslint/scope-manager": { - "version": "8.46.4", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.46.4.tgz", - "integrity": "sha512-tMDbLGXb1wC+McN1M6QeDx7P7c0UWO5z9CXqp7J8E+xGcJuUuevWKxuG8j41FoweS3+L41SkyKKkia16jpX7CA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/types": "8.46.4", - "@typescript-eslint/visitor-keys": "8.46.4" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/@typescript-eslint/tsconfig-utils": { - "version": "8.46.4", - "resolved": "https://registry.npmjs.org/@typescript-eslint/tsconfig-utils/-/tsconfig-utils-8.46.4.tgz", - "integrity": "sha512-+/XqaZPIAk6Cjg7NWgSGe27X4zMGqrFqZ8atJsX3CWxH/jACqWnrWI68h7nHQld0y+k9eTTjb9r+KU4twLoo9A==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "typescript": ">=4.8.4 <6.0.0" - } - }, - "node_modules/@typescript-eslint/type-utils": { - "version": "8.46.4", - "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.46.4.tgz", - "integrity": "sha512-V4QC8h3fdT5Wro6vANk6eojqfbv5bpwHuMsBcJUJkqs2z5XnYhJzyz9Y02eUmF9u3PgXEUiOt4w4KHR3P+z0PQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/types": "8.46.4", - "@typescript-eslint/typescript-estree": "8.46.4", - "@typescript-eslint/utils": "8.46.4", - "debug": "^4.3.4", - "ts-api-utils": "^2.1.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0", - "typescript": ">=4.8.4 <6.0.0" - } - }, - "node_modules/@typescript-eslint/types": { - "version": "8.46.4", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.46.4.tgz", - "integrity": "sha512-USjyxm3gQEePdUwJBFjjGNG18xY9A2grDVGuk7/9AkjIF1L+ZrVnwR5VAU5JXtUnBL/Nwt3H31KlRDaksnM7/w==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/@typescript-eslint/typescript-estree": { - "version": "8.46.4", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.46.4.tgz", - "integrity": "sha512-7oV2qEOr1d4NWNmpXLR35LvCfOkTNymY9oyW+lUHkmCno7aOmIf/hMaydnJBUTBMRCOGZh8YjkFOc8dadEoNGA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/project-service": "8.46.4", - "@typescript-eslint/tsconfig-utils": "8.46.4", - "@typescript-eslint/types": "8.46.4", - "@typescript-eslint/visitor-keys": "8.46.4", - "debug": "^4.3.4", - "fast-glob": "^3.3.2", - "is-glob": "^4.0.3", - "minimatch": "^9.0.4", - "semver": "^7.6.0", - "ts-api-utils": "^2.1.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "typescript": ">=4.8.4 <6.0.0" - } - }, - "node_modules/@typescript-eslint/typescript-estree/node_modules/brace-expansion": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.2.tgz", - "integrity": "sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "balanced-match": "^1.0.0" - } - }, - "node_modules/@typescript-eslint/typescript-estree/node_modules/minimatch": { - "version": "9.0.5", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", - "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", - "dev": true, - "license": "ISC", - "dependencies": { - "brace-expansion": "^2.0.1" - }, - "engines": { - "node": ">=16 || 14 >=14.17" - }, - "funding": { - "url": "https://github.com/sponsors/isaacs" - } - }, - "node_modules/@typescript-eslint/typescript-estree/node_modules/semver": { - "version": "7.7.3", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", - "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", - "dev": true, - "license": "ISC", - "bin": { - "semver": "bin/semver.js" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/@typescript-eslint/utils": { - "version": "8.46.4", - "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.46.4.tgz", - "integrity": "sha512-AbSv11fklGXV6T28dp2Me04Uw90R2iJ30g2bgLz529Koehrmkbs1r7paFqr1vPCZi7hHwYxYtxfyQMRC8QaVSg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@eslint-community/eslint-utils": "^4.7.0", - "@typescript-eslint/scope-manager": "8.46.4", - "@typescript-eslint/types": "8.46.4", - "@typescript-eslint/typescript-estree": "8.46.4" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0", - "typescript": ">=4.8.4 <6.0.0" - } - }, - "node_modules/@typescript-eslint/visitor-keys": { - "version": "8.46.4", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.46.4.tgz", - "integrity": "sha512-/++5CYLQqsO9HFGLI7APrxBJYo+5OCMpViuhV8q5/Qa3o5mMrF//eQHks+PXcsAVaLdn817fMuS7zqoXNNZGaw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/types": "8.46.4", - "eslint-visitor-keys": "^4.2.1" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/@ungap/structured-clone": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.3.0.tgz", - "integrity": "sha512-WmoN8qaIAo7WTYWbAZuG8PYEhn5fkz7dZrqTBZ7dtt//lL2Gwms1IcnQ5yHqjDfX8Ft5j4YzDM23f87zBfDe9g==", - "license": "ISC" - }, - "node_modules/@vitejs/plugin-react": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-5.1.0.tgz", - "integrity": "sha512-4LuWrg7EKWgQaMJfnN+wcmbAW+VSsCmqGohftWjuct47bv8uE4n/nPpq4XjJPsxgq00GGG5J8dvBczp8uxScew==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/core": "^7.28.4", - "@babel/plugin-transform-react-jsx-self": "^7.27.1", - "@babel/plugin-transform-react-jsx-source": "^7.27.1", - "@rolldown/pluginutils": "1.0.0-beta.43", - "@types/babel__core": "^7.20.5", - "react-refresh": "^0.18.0" - }, - "engines": { - "node": "^20.19.0 || >=22.12.0" - }, - "peerDependencies": { - "vite": "^4.2.0 || ^5.0.0 || ^6.0.0 || ^7.0.0" - } - }, - "node_modules/@vitest/coverage-v8": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/@vitest/coverage-v8/-/coverage-v8-3.2.4.tgz", - "integrity": "sha512-EyF9SXU6kS5Ku/U82E259WSnvg6c8KTjppUncuNdm5QHpe17mwREHnjDzozC8x9MZ0xfBUFSaLkRv4TMA75ALQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@ampproject/remapping": "^2.3.0", - "@bcoe/v8-coverage": "^1.0.2", - "ast-v8-to-istanbul": "^0.3.3", - "debug": "^4.4.1", - "istanbul-lib-coverage": "^3.2.2", - "istanbul-lib-report": "^3.0.1", - "istanbul-lib-source-maps": "^5.0.6", - "istanbul-reports": "^3.1.7", - "magic-string": "^0.30.17", - "magicast": "^0.3.5", - "std-env": "^3.9.0", - "test-exclude": "^7.0.1", - "tinyrainbow": "^2.0.0" - }, - "funding": { - "url": "https://opencollective.com/vitest" - }, - "peerDependencies": { - "@vitest/browser": "3.2.4", - "vitest": "3.2.4" - }, - "peerDependenciesMeta": { - "@vitest/browser": { - "optional": true - } - } - }, - "node_modules/@vitest/expect": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/@vitest/expect/-/expect-3.2.4.tgz", - "integrity": "sha512-Io0yyORnB6sikFlt8QW5K7slY4OjqNX9jmJQ02QDda8lyM6B5oNgVWoSoKPac8/kgnCUzuHQKrSLtu/uOqqrig==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/chai": "^5.2.2", - "@vitest/spy": "3.2.4", - "@vitest/utils": "3.2.4", - "chai": "^5.2.0", - "tinyrainbow": "^2.0.0" - }, - "funding": { - "url": "https://opencollective.com/vitest" - } - }, - "node_modules/@vitest/mocker": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/@vitest/mocker/-/mocker-3.2.4.tgz", - "integrity": "sha512-46ryTE9RZO/rfDd7pEqFl7etuyzekzEhUbTW3BvmeO/BcCMEgq59BKhek3dXDWgAj4oMK6OZi+vRr1wPW6qjEQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@vitest/spy": "3.2.4", - "estree-walker": "^3.0.3", - "magic-string": "^0.30.17" - }, - "funding": { - "url": "https://opencollective.com/vitest" - }, - "peerDependencies": { - "msw": "^2.4.9", - "vite": "^5.0.0 || ^6.0.0 || ^7.0.0-0" - }, - "peerDependenciesMeta": { - "msw": { - "optional": true - }, - "vite": { - "optional": true - } - } - }, - "node_modules/@vitest/pretty-format": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/@vitest/pretty-format/-/pretty-format-3.2.4.tgz", - "integrity": "sha512-IVNZik8IVRJRTr9fxlitMKeJeXFFFN0JaB9PHPGQ8NKQbGpfjlTx9zO4RefN8gp7eqjNy8nyK3NZmBzOPeIxtA==", - "dev": true, - "license": "MIT", - "dependencies": { - "tinyrainbow": "^2.0.0" - }, - "funding": { - "url": "https://opencollective.com/vitest" - } - }, - "node_modules/@vitest/runner": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/@vitest/runner/-/runner-3.2.4.tgz", - "integrity": "sha512-oukfKT9Mk41LreEW09vt45f8wx7DordoWUZMYdY/cyAk7w5TWkTRCNZYF7sX7n2wB7jyGAl74OxgwhPgKaqDMQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@vitest/utils": "3.2.4", - "pathe": "^2.0.3", - "strip-literal": "^3.0.0" - }, - "funding": { - "url": "https://opencollective.com/vitest" - } - }, - "node_modules/@vitest/snapshot": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/@vitest/snapshot/-/snapshot-3.2.4.tgz", - "integrity": "sha512-dEYtS7qQP2CjU27QBC5oUOxLE/v5eLkGqPE0ZKEIDGMs4vKWe7IjgLOeauHsR0D5YuuycGRO5oSRXnwnmA78fQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@vitest/pretty-format": "3.2.4", - "magic-string": "^0.30.17", - "pathe": "^2.0.3" - }, - "funding": { - "url": "https://opencollective.com/vitest" - } - }, - "node_modules/@vitest/spy": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/@vitest/spy/-/spy-3.2.4.tgz", - "integrity": "sha512-vAfasCOe6AIK70iP5UD11Ac4siNUNJ9i/9PZ3NKx07sG6sUxeag1LWdNrMWeKKYBLlzuK+Gn65Yd5nyL6ds+nw==", - "dev": true, - "license": "MIT", - "dependencies": { - "tinyspy": "^4.0.3" - }, - "funding": { - "url": "https://opencollective.com/vitest" - } - }, - "node_modules/@vitest/utils": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/@vitest/utils/-/utils-3.2.4.tgz", - "integrity": "sha512-fB2V0JFrQSMsCo9HiSq3Ezpdv4iYaXRG1Sx8edX3MwxfyNn83mKiGzOcH+Fkxt4MHxr3y42fQi1oeAInqgX2QA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@vitest/pretty-format": "3.2.4", - "loupe": "^3.1.4", - "tinyrainbow": "^2.0.0" - }, - "funding": { - "url": "https://opencollective.com/vitest" - } - }, - "node_modules/@xyflow/react": { - "version": "12.9.3", - "resolved": "https://registry.npmjs.org/@xyflow/react/-/react-12.9.3.tgz", - "integrity": "sha512-PSWoJ8vHiEqSIkLIkge+0eiHWiw4C6dyFDA03VKWJkqbU4A13VlDIVwKqf/Znuysn2GQw/zA61zpHE4rGgax7Q==", - "license": "MIT", - "dependencies": { - "@xyflow/system": "0.0.73", - "classcat": "^5.0.3", - "zustand": "^4.4.0" - }, - "peerDependencies": { - "react": ">=17", - "react-dom": ">=17" - } - }, - "node_modules/@xyflow/react/node_modules/zustand": { - "version": "4.5.7", - "resolved": "https://registry.npmjs.org/zustand/-/zustand-4.5.7.tgz", - "integrity": "sha512-CHOUy7mu3lbD6o6LJLfllpjkzhHXSBlX8B9+qPddUsIfeF5S/UZ5q0kmCsnRqT1UHFQZchNFDDzMbQsuesHWlw==", - "license": "MIT", - "dependencies": { - "use-sync-external-store": "^1.2.2" - }, - "engines": { - "node": ">=12.7.0" - }, - "peerDependencies": { - "@types/react": ">=16.8", - "immer": ">=9.0.6", - "react": ">=16.8" - }, - "peerDependenciesMeta": { - "@types/react": { - "optional": true - }, - "immer": { - "optional": true - }, - "react": { - "optional": true - } - } - }, - "node_modules/@xyflow/system": { - "version": "0.0.73", - "resolved": "https://registry.npmjs.org/@xyflow/system/-/system-0.0.73.tgz", - "integrity": "sha512-C2ymH2V4mYDkdVSiRx0D7R0s3dvfXiupVBcko6tXP5K4tVdSBMo22/e3V9yRNdn+2HQFv44RFKzwOyCcUUDAVQ==", - "license": "MIT", - "dependencies": { - "@types/d3-drag": "^3.0.7", - "@types/d3-interpolate": "^3.0.4", - "@types/d3-selection": "^3.0.10", - "@types/d3-transition": "^3.0.8", - "@types/d3-zoom": "^3.0.8", - "d3-drag": "^3.0.0", - "d3-interpolate": "^3.0.1", - "d3-selection": "^3.0.0", - "d3-zoom": "^3.0.0" - } - }, - "node_modules/acorn": { - "version": "8.15.0", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz", - "integrity": "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==", - "license": "MIT", - "bin": { - "acorn": "bin/acorn" - }, - "engines": { - "node": ">=0.4.0" - } - }, - "node_modules/acorn-jsx": { - "version": "5.3.2", - "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", - "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", - "dev": true, - "license": "MIT", - "peerDependencies": { - "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" - } - }, - "node_modules/ajv": { - "version": "6.12.6", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", - "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", - "dev": true, - "license": "MIT", - "dependencies": { - "fast-deep-equal": "^3.1.1", - "fast-json-stable-stringify": "^2.0.0", - "json-schema-traverse": "^0.4.1", - "uri-js": "^4.2.2" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/epoberezkin" - } - }, - "node_modules/ansi-regex": { - "version": "6.2.2", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.2.2.tgz", - "integrity": "sha512-Bq3SmSpyFHaWjPk8If9yc6svM8c56dB5BAtW4Qbw5jHTwwXXcTLoRMkpDJp6VL0XzlWaCHTXrkFURMYmD0sLqg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/chalk/ansi-regex?sponsor=1" - } - }, - "node_modules/ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", - "dev": true, - "license": "MIT", - "dependencies": { - "color-convert": "^2.0.1" - }, - "engines": { - "node": ">=8" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, - "node_modules/any-promise": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/any-promise/-/any-promise-1.3.0.tgz", - "integrity": "sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==", - "dev": true, - "license": "MIT" - }, - "node_modules/anymatch": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", - "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", - "dev": true, - "license": "ISC", - "dependencies": { - "normalize-path": "^3.0.0", - "picomatch": "^2.0.4" - }, - "engines": { - "node": ">= 8" - } - }, - "node_modules/arg": { - "version": "5.0.2", - "resolved": "https://registry.npmjs.org/arg/-/arg-5.0.2.tgz", - "integrity": "sha512-PYjyFOLKQ9y57JvQ6QLo8dAgNqswh8M1RMJYdQduT6xbWSgK36P/Z/v+p888pM69jMMfS8Xd8F6I1kQ/I9HUGg==", - "dev": true, - "license": "MIT" - }, - "node_modules/argparse": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", - "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", - "dev": true, - "license": "Python-2.0" - }, - "node_modules/assertion-error": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-2.0.1.tgz", - "integrity": "sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=12" - } - }, - "node_modules/ast-v8-to-istanbul": { - "version": "0.3.10", - "resolved": "https://registry.npmjs.org/ast-v8-to-istanbul/-/ast-v8-to-istanbul-0.3.10.tgz", - "integrity": "sha512-p4K7vMz2ZSk3wN8l5o3y2bJAoZXT3VuJI5OLTATY/01CYWumWvwkUw0SqDBnNq6IiTO3qDa1eSQDibAV8g7XOQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@jridgewell/trace-mapping": "^0.3.31", - "estree-walker": "^3.0.3", - "js-tokens": "^9.0.1" - } - }, - "node_modules/ast-v8-to-istanbul/node_modules/js-tokens": { - "version": "9.0.1", - "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-9.0.1.tgz", - "integrity": "sha512-mxa9E9ITFOt0ban3j6L5MpjwegGz6lBQmM1IJkWeBZGcMxto50+eWdjC/52xDbS2vy0k7vIMK0Fe2wfL9OQSpQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/autoprefixer": { - "version": "10.4.22", - "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-10.4.22.tgz", - "integrity": "sha512-ARe0v/t9gO28Bznv6GgqARmVqcWOV3mfgUPn9becPHMiD3o9BwlRgaeccZnwTpZ7Zwqrm+c1sUSsMxIzQzc8Xg==", - "dev": true, - "funding": [ - { - "type": "opencollective", - "url": "https://opencollective.com/postcss/" - }, - { - "type": "tidelift", - "url": "https://tidelift.com/funding/github/npm/autoprefixer" - }, - { - "type": "github", - "url": "https://github.com/sponsors/ai" - } - ], - "license": "MIT", - "dependencies": { - "browserslist": "^4.27.0", - "caniuse-lite": "^1.0.30001754", - "fraction.js": "^5.3.4", - "normalize-range": "^0.1.2", - "picocolors": "^1.1.1", - "postcss-value-parser": "^4.2.0" - }, - "bin": { - "autoprefixer": "bin/autoprefixer" - }, - "engines": { - "node": "^10 || ^12 || >=14" - }, - "peerDependencies": { - "postcss": "^8.1.0" - } - }, - "node_modules/bail": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/bail/-/bail-2.0.2.tgz", - "integrity": "sha512-0xO6mYd7JB2YesxDKplafRpsiOzPt9V02ddPCLbY1xYGPOX24NTyN50qnUxgCPcSoYMhKpAuBTjQoRZCAkUDRw==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/balanced-match": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", - "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", - "dev": true, - "license": "MIT" - }, - "node_modules/baseline-browser-mapping": { - "version": "2.8.25", - "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.8.25.tgz", - "integrity": "sha512-2NovHVesVF5TXefsGX1yzx1xgr7+m9JQenvz6FQY3qd+YXkKkYiv+vTCc7OriP9mcDZpTC5mAOYN4ocd29+erA==", - "dev": true, - "license": "Apache-2.0", - "bin": { - "baseline-browser-mapping": "dist/cli.js" - } - }, - "node_modules/binary-extensions": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", - "integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/brace-expansion": { - "version": "1.1.12", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", - "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", - "dev": true, - "license": "MIT", - "dependencies": { - "balanced-match": "^1.0.0", - "concat-map": "0.0.1" - } - }, - "node_modules/braces": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", - "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", - "dev": true, - "license": "MIT", - "dependencies": { - "fill-range": "^7.1.1" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/browserslist": { - "version": "4.28.0", - "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.0.tgz", - "integrity": "sha512-tbydkR/CxfMwelN0vwdP/pLkDwyAASZ+VfWm4EOwlB6SWhx1sYnWLqo8N5j0rAzPfzfRaxt0mM/4wPU/Su84RQ==", - "dev": true, - "funding": [ - { - "type": "opencollective", - "url": "https://opencollective.com/browserslist" - }, - { - "type": "tidelift", - "url": "https://tidelift.com/funding/github/npm/browserslist" - }, - { - "type": "github", - "url": "https://github.com/sponsors/ai" - } - ], - "license": "MIT", - "dependencies": { - "baseline-browser-mapping": "^2.8.25", - "caniuse-lite": "^1.0.30001754", - "electron-to-chromium": "^1.5.249", - "node-releases": "^2.0.27", - "update-browserslist-db": "^1.1.4" - }, - "bin": { - "browserslist": "cli.js" - }, - "engines": { - "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" - } - }, - "node_modules/cac": { - "version": "6.7.14", - "resolved": "https://registry.npmjs.org/cac/-/cac-6.7.14.tgz", - "integrity": "sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/callsites": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", - "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/camelcase-css": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/camelcase-css/-/camelcase-css-2.0.1.tgz", - "integrity": "sha512-QOSvevhslijgYwRx6Rv7zKdMF8lbRmx+uQGx2+vDc+KI/eBnsy9kit5aj23AgGu3pa4t9AgwbnXWqS+iOY+2aA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 6" - } - }, - "node_modules/caniuse-lite": { - "version": "1.0.30001754", - "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001754.tgz", - "integrity": "sha512-x6OeBXueoAceOmotzx3PO4Zpt4rzpeIFsSr6AAePTZxSkXiYDUmpypEl7e2+8NCd9bD7bXjqyef8CJYPC1jfxg==", - "dev": true, - "funding": [ - { - "type": "opencollective", - "url": "https://opencollective.com/browserslist" - }, - { - "type": "tidelift", - "url": "https://tidelift.com/funding/github/npm/caniuse-lite" - }, - { - "type": "github", - "url": "https://github.com/sponsors/ai" - } - ], - "license": "CC-BY-4.0" - }, - "node_modules/ccount": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/ccount/-/ccount-2.0.1.tgz", - "integrity": "sha512-eyrF0jiFpY+3drT6383f1qhkbGsLSifNAjA61IUjZjmLCWjItY6LB9ft9YhoDgwfmclB2zhu51Lc7+95b8NRAg==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/chai": { - "version": "5.3.3", - "resolved": "https://registry.npmjs.org/chai/-/chai-5.3.3.tgz", - "integrity": "sha512-4zNhdJD/iOjSH0A05ea+Ke6MU5mmpQcbQsSOkgdaUMJ9zTlDTD/GYlwohmIE2u0gaxHYiVHEn1Fw9mZ/ktJWgw==", - "dev": true, - "license": "MIT", - "dependencies": { - "assertion-error": "^2.0.1", - "check-error": "^2.1.1", - "deep-eql": "^5.0.1", - "loupe": "^3.1.0", - "pathval": "^2.0.0" - }, - "engines": { - "node": ">=18" - } - }, - "node_modules/chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", - "dev": true, - "license": "MIT", - "dependencies": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/chalk?sponsor=1" - } - }, - "node_modules/character-entities": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/character-entities/-/character-entities-2.0.2.tgz", - "integrity": "sha512-shx7oQ0Awen/BRIdkjkvz54PnEEI/EjwXDSIZp86/KKdbafHh1Df/RYGBhn4hbe2+uKC9FnT5UCEdyPz3ai9hQ==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/character-entities-html4": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/character-entities-html4/-/character-entities-html4-2.1.0.tgz", - "integrity": "sha512-1v7fgQRj6hnSwFpq1Eu0ynr/CDEw0rXo2B61qXrLNdHZmPKgb7fqS1a2JwF0rISo9q77jDI8VMEHoApn8qDoZA==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/character-entities-legacy": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/character-entities-legacy/-/character-entities-legacy-3.0.0.tgz", - "integrity": "sha512-RpPp0asT/6ufRm//AJVwpViZbGM/MkjQFxJccQRHmISF/22NBtsHqAWmL+/pmkPWoIUJdWyeVleTl1wydHATVQ==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/character-reference-invalid": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/character-reference-invalid/-/character-reference-invalid-2.0.1.tgz", - "integrity": "sha512-iBZ4F4wRbyORVsu0jPV7gXkOsGYjGHPmAyv+HiHG8gi5PtC9KI2j1+v8/tlibRvjoWX027ypmG/n0HtO5t7unw==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/chart.js": { - "version": "4.5.1", - "resolved": "https://registry.npmjs.org/chart.js/-/chart.js-4.5.1.tgz", - "integrity": "sha512-GIjfiT9dbmHRiYi6Nl2yFCq7kkwdkp1W/lp2J99rX0yo9tgJGn3lKQATztIjb5tVtevcBtIdICNWqlq5+E8/Pw==", - "license": "MIT", - "dependencies": { - "@kurkle/color": "^0.3.0" - }, - "engines": { - "pnpm": ">=8" - } - }, - "node_modules/check-error": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/check-error/-/check-error-2.1.3.tgz", - "integrity": "sha512-PAJdDJusoxnwm1VwW07VWwUN1sl7smmC3OKggvndJFadxxDRyFJBX/ggnu/KE4kQAB7a3Dp8f/YXC1FlUprWmA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 16" - } - }, - "node_modules/chevrotain": { - "version": "11.0.3", - "resolved": "https://registry.npmjs.org/chevrotain/-/chevrotain-11.0.3.tgz", - "integrity": "sha512-ci2iJH6LeIkvP9eJW6gpueU8cnZhv85ELY8w8WiFtNjMHA5ad6pQLaJo9mEly/9qUyCpvqX8/POVUTf18/HFdw==", - "license": "Apache-2.0", - "dependencies": { - "@chevrotain/cst-dts-gen": "11.0.3", - "@chevrotain/gast": "11.0.3", - "@chevrotain/regexp-to-ast": "11.0.3", - "@chevrotain/types": "11.0.3", - "@chevrotain/utils": "11.0.3", - "lodash-es": "4.17.21" - } - }, - "node_modules/chevrotain-allstar": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/chevrotain-allstar/-/chevrotain-allstar-0.3.1.tgz", - "integrity": "sha512-b7g+y9A0v4mxCW1qUhf3BSVPg+/NvGErk/dOkrDaHA0nQIQGAtrOjlX//9OQtRlSCy+x9rfB5N8yC71lH1nvMw==", - "license": "MIT", - "dependencies": { - "lodash-es": "^4.17.21" - }, - "peerDependencies": { - "chevrotain": "^11.0.0" - } - }, - "node_modules/chokidar": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", - "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", - "dev": true, - "license": "MIT", - "dependencies": { - "anymatch": "~3.1.2", - "braces": "~3.0.2", - "glob-parent": "~5.1.2", - "is-binary-path": "~2.1.0", - "is-glob": "~4.0.1", - "normalize-path": "~3.0.0", - "readdirp": "~3.6.0" - }, - "engines": { - "node": ">= 8.10.0" - }, - "funding": { - "url": "https://paulmillr.com/funding/" - }, - "optionalDependencies": { - "fsevents": "~2.3.2" - } - }, - "node_modules/chokidar/node_modules/glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "dev": true, - "license": "ISC", - "dependencies": { - "is-glob": "^4.0.1" - }, - "engines": { - "node": ">= 6" - } - }, - "node_modules/class-variance-authority": { - "version": "0.7.1", - "resolved": "https://registry.npmjs.org/class-variance-authority/-/class-variance-authority-0.7.1.tgz", - "integrity": "sha512-Ka+9Trutv7G8M6WT6SeiRWz792K5qEqIGEGzXKhAE6xOWAY6pPH8U+9IY3oCMv6kqTmLsv7Xh/2w2RigkePMsg==", - "license": "Apache-2.0", - "dependencies": { - "clsx": "^2.1.1" - }, - "funding": { - "url": "https://polar.sh/cva" - } - }, - "node_modules/classcat": { - "version": "5.0.5", - "resolved": "https://registry.npmjs.org/classcat/-/classcat-5.0.5.tgz", - "integrity": "sha512-JhZUT7JFcQy/EzW605k/ktHtncoo9vnyW/2GspNYwFlN1C/WmjuV/xtS04e9SOkL2sTdw0VAZ2UGCcQ9lR6p6w==", - "license": "MIT" - }, - "node_modules/clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==", - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "color-name": "~1.1.4" - }, - "engines": { - "node": ">=7.0.0" - } - }, - "node_modules/color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true, - "license": "MIT" - }, - "node_modules/comma-separated-tokens": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/comma-separated-tokens/-/comma-separated-tokens-2.0.3.tgz", - "integrity": "sha512-Fu4hJdvzeylCfQPp9SGWidpzrMs7tTrlu6Vb8XGaRGck8QSNZJJp538Wrb60Lax4fPwR64ViY468OIUTbRlGZg==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/commander": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/commander/-/commander-4.1.1.tgz", - "integrity": "sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 6" - } - }, - "node_modules/concat-map": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", - "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", - "dev": true, - "license": "MIT" - }, - "node_modules/confbox": { - "version": "0.2.2", - "resolved": "https://registry.npmjs.org/confbox/-/confbox-0.2.2.tgz", - "integrity": "sha512-1NB+BKqhtNipMsov4xI/NnhCKp9XG9NamYp5PVm9klAT0fsrNPjaFICsCFhNhwZJKNh7zB/3q8qXz0E9oaMNtQ==", - "license": "MIT" - }, - "node_modules/convert-source-map": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", - "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", - "dev": true, - "license": "MIT" - }, - "node_modules/cookie": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-1.1.1.tgz", - "integrity": "sha512-ei8Aos7ja0weRpFzJnEA9UHJ/7XQmqglbRwnf2ATjcB9Wq874VKH9kfjjirM6UhU2/E5fFYadylyhFldcqSidQ==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/cose-base": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/cose-base/-/cose-base-1.0.3.tgz", - "integrity": "sha512-s9whTXInMSgAp/NVXVNuVxVKzGH2qck3aQlVHxDCdAEPgtMKwc4Wq6/QKhgdEdgbLSi9rBTAcPoRa6JpiG4ksg==", - "license": "MIT", - "dependencies": { - "layout-base": "^1.0.0" - } - }, - "node_modules/cross-spawn": { - "version": "7.0.6", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", - "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", - "dev": true, - "license": "MIT", - "dependencies": { - "path-key": "^3.1.0", - "shebang-command": "^2.0.0", - "which": "^2.0.1" - }, - "engines": { - "node": ">= 8" - } - }, - "node_modules/cssesc": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz", - "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==", - "dev": true, - "license": "MIT", - "bin": { - "cssesc": "bin/cssesc" - }, - "engines": { - "node": ">=4" - } - }, - "node_modules/csstype": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz", - "integrity": "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==", - "license": "MIT" - }, - "node_modules/cytoscape": { - "version": "3.33.1", - "resolved": "https://registry.npmjs.org/cytoscape/-/cytoscape-3.33.1.tgz", - "integrity": "sha512-iJc4TwyANnOGR1OmWhsS9ayRS3s+XQ185FmuHObThD+5AeJCakAAbWv8KimMTt08xCCLNgneQwFp+JRJOr9qGQ==", - "license": "MIT", - "engines": { - "node": ">=0.10" - } - }, - "node_modules/cytoscape-cose-bilkent": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/cytoscape-cose-bilkent/-/cytoscape-cose-bilkent-4.1.0.tgz", - "integrity": "sha512-wgQlVIUJF13Quxiv5e1gstZ08rnZj2XaLHGoFMYXz7SkNfCDOOteKBE6SYRfA9WxxI/iBc3ajfDoc6hb/MRAHQ==", - "license": "MIT", - "dependencies": { - "cose-base": "^1.0.0" - }, - "peerDependencies": { - "cytoscape": "^3.2.0" - } - }, - "node_modules/cytoscape-fcose": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/cytoscape-fcose/-/cytoscape-fcose-2.2.0.tgz", - "integrity": "sha512-ki1/VuRIHFCzxWNrsshHYPs6L7TvLu3DL+TyIGEsRcvVERmxokbf5Gdk7mFxZnTdiGtnA4cfSmjZJMviqSuZrQ==", - "license": "MIT", - "dependencies": { - "cose-base": "^2.2.0" - }, - "peerDependencies": { - "cytoscape": "^3.2.0" - } - }, - "node_modules/cytoscape-fcose/node_modules/cose-base": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/cose-base/-/cose-base-2.2.0.tgz", - "integrity": "sha512-AzlgcsCbUMymkADOJtQm3wO9S3ltPfYOFD5033keQn9NJzIbtnZj+UdBJe7DYml/8TdbtHJW3j58SOnKhWY/5g==", - "license": "MIT", - "dependencies": { - "layout-base": "^2.0.0" - } - }, - "node_modules/cytoscape-fcose/node_modules/layout-base": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/layout-base/-/layout-base-2.0.1.tgz", - "integrity": "sha512-dp3s92+uNI1hWIpPGH3jK2kxE2lMjdXdr+DH8ynZHpd6PUlH6x6cbuXnoMmiNumznqaNO31xu9e79F0uuZ0JFg==", - "license": "MIT" - }, - "node_modules/d3": { - "version": "7.9.0", - "resolved": "https://registry.npmjs.org/d3/-/d3-7.9.0.tgz", - "integrity": "sha512-e1U46jVP+w7Iut8Jt8ri1YsPOvFpg46k+K8TpCb0P+zjCkjkPnV7WzfDJzMHy1LnA+wj5pLT1wjO901gLXeEhA==", - "license": "ISC", - "dependencies": { - "d3-array": "3", - "d3-axis": "3", - "d3-brush": "3", - "d3-chord": "3", - "d3-color": "3", - "d3-contour": "4", - "d3-delaunay": "6", - "d3-dispatch": "3", - "d3-drag": "3", - "d3-dsv": "3", - "d3-ease": "3", - "d3-fetch": "3", - "d3-force": "3", - "d3-format": "3", - "d3-geo": "3", - "d3-hierarchy": "3", - "d3-interpolate": "3", - "d3-path": "3", - "d3-polygon": "3", - "d3-quadtree": "3", - "d3-random": "3", - "d3-scale": "4", - "d3-scale-chromatic": "3", - "d3-selection": "3", - "d3-shape": "3", - "d3-time": "3", - "d3-time-format": "4", - "d3-timer": "3", - "d3-transition": "3", - "d3-zoom": "3" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-array": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/d3-array/-/d3-array-3.2.4.tgz", - "integrity": "sha512-tdQAmyA18i4J7wprpYq8ClcxZy3SC31QMeByyCFyRt7BVHdREQZ5lpzoe5mFEYZUWe+oq8HBvk9JjpibyEV4Jg==", - "license": "ISC", - "dependencies": { - "internmap": "1 - 2" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-axis": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/d3-axis/-/d3-axis-3.0.0.tgz", - "integrity": "sha512-IH5tgjV4jE/GhHkRV0HiVYPDtvfjHQlQfJHs0usq7M30XcSBvOotpmH1IgkcXsO/5gEQZD43B//fc7SRT5S+xw==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-brush": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/d3-brush/-/d3-brush-3.0.0.tgz", - "integrity": "sha512-ALnjWlVYkXsVIGlOsuWH1+3udkYFI48Ljihfnh8FZPF2QS9o+PzGLBslO0PjzVoHLZ2KCVgAM8NVkXPJB2aNnQ==", - "license": "ISC", - "dependencies": { - "d3-dispatch": "1 - 3", - "d3-drag": "2 - 3", - "d3-interpolate": "1 - 3", - "d3-selection": "3", - "d3-transition": "3" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-chord": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/d3-chord/-/d3-chord-3.0.1.tgz", - "integrity": "sha512-VE5S6TNa+j8msksl7HwjxMHDM2yNK3XCkusIlpX5kwauBfXuyLAtNg9jCp/iHH61tgI4sb6R/EIMWCqEIdjT/g==", - "license": "ISC", - "dependencies": { - "d3-path": "1 - 3" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-color": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/d3-color/-/d3-color-3.1.0.tgz", - "integrity": "sha512-zg/chbXyeBtMQ1LbD/WSoW2DpC3I0mpmPdW+ynRTj/x2DAWYrIY7qeZIHidozwV24m4iavr15lNwIwLxRmOxhA==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-contour": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/d3-contour/-/d3-contour-4.0.2.tgz", - "integrity": "sha512-4EzFTRIikzs47RGmdxbeUvLWtGedDUNkTcmzoeyg4sP/dvCexO47AaQL7VKy/gul85TOxw+IBgA8US2xwbToNA==", - "license": "ISC", - "dependencies": { - "d3-array": "^3.2.0" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-delaunay": { - "version": "6.0.4", - "resolved": "https://registry.npmjs.org/d3-delaunay/-/d3-delaunay-6.0.4.tgz", - "integrity": "sha512-mdjtIZ1XLAM8bm/hx3WwjfHt6Sggek7qH043O8KEjDXN40xi3vx/6pYSVTwLjEgiXQTbvaouWKynLBiUZ6SK6A==", - "license": "ISC", - "dependencies": { - "delaunator": "5" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-dispatch": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/d3-dispatch/-/d3-dispatch-3.0.1.tgz", - "integrity": "sha512-rzUyPU/S7rwUflMyLc1ETDeBj0NRuHKKAcvukozwhshr6g6c5d8zh4c2gQjY2bZ0dXeGLWc1PF174P2tVvKhfg==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-drag": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/d3-drag/-/d3-drag-3.0.0.tgz", - "integrity": "sha512-pWbUJLdETVA8lQNJecMxoXfH6x+mO2UQo8rSmZ+QqxcbyA3hfeprFgIT//HW2nlHChWeIIMwS2Fq+gEARkhTkg==", - "license": "ISC", - "dependencies": { - "d3-dispatch": "1 - 3", - "d3-selection": "3" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-dsv": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/d3-dsv/-/d3-dsv-3.0.1.tgz", - "integrity": "sha512-UG6OvdI5afDIFP9w4G0mNq50dSOsXHJaRE8arAS5o9ApWnIElp8GZw1Dun8vP8OyHOZ/QJUKUJwxiiCCnUwm+Q==", - "license": "ISC", - "dependencies": { - "commander": "7", - "iconv-lite": "0.6", - "rw": "1" - }, - "bin": { - "csv2json": "bin/dsv2json.js", - "csv2tsv": "bin/dsv2dsv.js", - "dsv2dsv": "bin/dsv2dsv.js", - "dsv2json": "bin/dsv2json.js", - "json2csv": "bin/json2dsv.js", - "json2dsv": "bin/json2dsv.js", - "json2tsv": "bin/json2dsv.js", - "tsv2csv": "bin/dsv2dsv.js", - "tsv2json": "bin/dsv2json.js" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-dsv/node_modules/commander": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/commander/-/commander-7.2.0.tgz", - "integrity": "sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==", - "license": "MIT", - "engines": { - "node": ">= 10" - } - }, - "node_modules/d3-ease": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/d3-ease/-/d3-ease-3.0.1.tgz", - "integrity": "sha512-wR/XK3D3XcLIZwpbvQwQ5fK+8Ykds1ip7A2Txe0yxncXSdq1L9skcG7blcedkOX+ZcgxGAmLX1FrRGbADwzi0w==", - "license": "BSD-3-Clause", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-fetch": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/d3-fetch/-/d3-fetch-3.0.1.tgz", - "integrity": "sha512-kpkQIM20n3oLVBKGg6oHrUchHM3xODkTzjMoj7aWQFq5QEM+R6E4WkzT5+tojDY7yjez8KgCBRoj4aEr99Fdqw==", - "license": "ISC", - "dependencies": { - "d3-dsv": "1 - 3" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-force": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/d3-force/-/d3-force-3.0.0.tgz", - "integrity": "sha512-zxV/SsA+U4yte8051P4ECydjD/S+qeYtnaIyAs9tgHCqfguma/aAQDjo85A9Z6EKhBirHRJHXIgJUlffT4wdLg==", - "license": "ISC", - "dependencies": { - "d3-dispatch": "1 - 3", - "d3-quadtree": "1 - 3", - "d3-timer": "1 - 3" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-format": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/d3-format/-/d3-format-3.1.0.tgz", - "integrity": "sha512-YyUI6AEuY/Wpt8KWLgZHsIU86atmikuoOmCfommt0LYHiQSPjvX2AcFc38PX0CBpr2RCyZhjex+NS/LPOv6YqA==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-geo": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/d3-geo/-/d3-geo-3.1.1.tgz", - "integrity": "sha512-637ln3gXKXOwhalDzinUgY83KzNWZRKbYubaG+fGVuc/dxO64RRljtCTnf5ecMyE1RIdtqpkVcq0IbtU2S8j2Q==", - "license": "ISC", - "dependencies": { - "d3-array": "2.5.0 - 3" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-hierarchy": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/d3-hierarchy/-/d3-hierarchy-3.1.2.tgz", - "integrity": "sha512-FX/9frcub54beBdugHjDCdikxThEqjnR93Qt7PvQTOHxyiNCAlvMrHhclk3cD5VeAaq9fxmfRp+CnWw9rEMBuA==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-interpolate": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/d3-interpolate/-/d3-interpolate-3.0.1.tgz", - "integrity": "sha512-3bYs1rOD33uo8aqJfKP3JWPAibgw8Zm2+L9vBKEHJ2Rg+viTR7o5Mmv5mZcieN+FRYaAOWX5SJATX6k1PWz72g==", - "license": "ISC", - "dependencies": { - "d3-color": "1 - 3" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-path": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/d3-path/-/d3-path-3.1.0.tgz", - "integrity": "sha512-p3KP5HCf/bvjBSSKuXid6Zqijx7wIfNW+J/maPs+iwR35at5JCbLUT0LzF1cnjbCHWhqzQTIN2Jpe8pRebIEFQ==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-polygon": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/d3-polygon/-/d3-polygon-3.0.1.tgz", - "integrity": "sha512-3vbA7vXYwfe1SYhED++fPUQlWSYTTGmFmQiany/gdbiWgU/iEyQzyymwL9SkJjFFuCS4902BSzewVGsHHmHtXg==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-quadtree": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/d3-quadtree/-/d3-quadtree-3.0.1.tgz", - "integrity": "sha512-04xDrxQTDTCFwP5H6hRhsRcb9xxv2RzkcsygFzmkSIOJy3PeRJP7sNk3VRIbKXcog561P9oU0/rVH6vDROAgUw==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-random": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/d3-random/-/d3-random-3.0.1.tgz", - "integrity": "sha512-FXMe9GfxTxqd5D6jFsQ+DJ8BJS4E/fT5mqqdjovykEB2oFbTMDVdg1MGFxfQW+FBOGoB++k8swBrgwSHT1cUXQ==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-sankey": { - "version": "0.12.3", - "resolved": "https://registry.npmjs.org/d3-sankey/-/d3-sankey-0.12.3.tgz", - "integrity": "sha512-nQhsBRmM19Ax5xEIPLMY9ZmJ/cDvd1BG3UVvt5h3WRxKg5zGRbvnteTyWAbzeSvlh3tW7ZEmq4VwR5mB3tutmQ==", - "license": "BSD-3-Clause", - "dependencies": { - "d3-array": "1 - 2", - "d3-shape": "^1.2.0" - } - }, - "node_modules/d3-sankey/node_modules/d3-array": { - "version": "2.12.1", - "resolved": "https://registry.npmjs.org/d3-array/-/d3-array-2.12.1.tgz", - "integrity": "sha512-B0ErZK/66mHtEsR1TkPEEkwdy+WDesimkM5gpZr5Dsg54BiTA5RXtYW5qTLIAcekaS9xfZrzBLF/OAkB3Qn1YQ==", - "license": "BSD-3-Clause", - "dependencies": { - "internmap": "^1.0.0" - } - }, - "node_modules/d3-sankey/node_modules/d3-path": { - "version": "1.0.9", - "resolved": "https://registry.npmjs.org/d3-path/-/d3-path-1.0.9.tgz", - "integrity": "sha512-VLaYcn81dtHVTjEHd8B+pbe9yHWpXKZUC87PzoFmsFrJqgFwDe/qxfp5MlfsfM1V5E/iVt0MmEbWQ7FVIXh/bg==", - "license": "BSD-3-Clause" - }, - "node_modules/d3-sankey/node_modules/d3-shape": { - "version": "1.3.7", - "resolved": "https://registry.npmjs.org/d3-shape/-/d3-shape-1.3.7.tgz", - "integrity": "sha512-EUkvKjqPFUAZyOlhY5gzCxCeI0Aep04LwIRpsZ/mLFelJiUfnK56jo5JMDSE7yyP2kLSb6LtF+S5chMk7uqPqw==", - "license": "BSD-3-Clause", - "dependencies": { - "d3-path": "1" - } - }, - "node_modules/d3-sankey/node_modules/internmap": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/internmap/-/internmap-1.0.1.tgz", - "integrity": "sha512-lDB5YccMydFBtasVtxnZ3MRBHuaoE8GKsppq+EchKL2U4nK/DmEpPHNH8MZe5HkMtpSiTSOZwfN0tzYjO/lJEw==", - "license": "ISC" - }, - "node_modules/d3-scale": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/d3-scale/-/d3-scale-4.0.2.tgz", - "integrity": "sha512-GZW464g1SH7ag3Y7hXjf8RoUuAFIqklOAq3MRl4OaWabTFJY9PN/E1YklhXLh+OQ3fM9yS2nOkCoS+WLZ6kvxQ==", - "license": "ISC", - "dependencies": { - "d3-array": "2.10.0 - 3", - "d3-format": "1 - 3", - "d3-interpolate": "1.2.0 - 3", - "d3-time": "2.1.1 - 3", - "d3-time-format": "2 - 4" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-scale-chromatic": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/d3-scale-chromatic/-/d3-scale-chromatic-3.1.0.tgz", - "integrity": "sha512-A3s5PWiZ9YCXFye1o246KoscMWqf8BsD9eRiJ3He7C9OBaxKhAd5TFCdEx/7VbKtxxTsu//1mMJFrEt572cEyQ==", - "license": "ISC", - "dependencies": { - "d3-color": "1 - 3", - "d3-interpolate": "1 - 3" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-selection": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/d3-selection/-/d3-selection-3.0.0.tgz", - "integrity": "sha512-fmTRWbNMmsmWq6xJV8D19U/gw/bwrHfNXxrIN+HfZgnzqTHp9jOmKMhsTUjXOJnZOdZY9Q28y4yebKzqDKlxlQ==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-shape": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/d3-shape/-/d3-shape-3.2.0.tgz", - "integrity": "sha512-SaLBuwGm3MOViRq2ABk3eLoxwZELpH6zhl3FbAoJ7Vm1gofKx6El1Ib5z23NUEhF9AsGl7y+dzLe5Cw2AArGTA==", - "license": "ISC", - "dependencies": { - "d3-path": "^3.1.0" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-time": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/d3-time/-/d3-time-3.1.0.tgz", - "integrity": "sha512-VqKjzBLejbSMT4IgbmVgDjpkYrNWUYJnbCGo874u7MMKIWsILRX+OpX/gTk8MqjpT1A/c6HY2dCA77ZN0lkQ2Q==", - "license": "ISC", - "dependencies": { - "d3-array": "2 - 3" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-time-format": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/d3-time-format/-/d3-time-format-4.1.0.tgz", - "integrity": "sha512-dJxPBlzC7NugB2PDLwo9Q8JiTR3M3e4/XANkreKSUxF8vvXKqm1Yfq4Q5dl8budlunRVlUUaDUgFt7eA8D6NLg==", - "license": "ISC", - "dependencies": { - "d3-time": "1 - 3" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-timer": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/d3-timer/-/d3-timer-3.0.1.tgz", - "integrity": "sha512-ndfJ/JxxMd3nw31uyKoY2naivF+r29V+Lc0svZxe1JvvIRmi8hUsrMvdOwgS1o6uBHmiz91geQ0ylPP0aj1VUA==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-transition": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/d3-transition/-/d3-transition-3.0.1.tgz", - "integrity": "sha512-ApKvfjsSR6tg06xrL434C0WydLr7JewBB3V+/39RMHsaXTOG0zmt/OAXeng5M5LBm0ojmxJrpomQVZ1aPvBL4w==", - "license": "ISC", - "dependencies": { - "d3-color": "1 - 3", - "d3-dispatch": "1 - 3", - "d3-ease": "1 - 3", - "d3-interpolate": "1 - 3", - "d3-timer": "1 - 3" - }, - "engines": { - "node": ">=12" - }, - "peerDependencies": { - "d3-selection": "2 - 3" - } - }, - "node_modules/d3-zoom": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/d3-zoom/-/d3-zoom-3.0.0.tgz", - "integrity": "sha512-b8AmV3kfQaqWAuacbPuNbL6vahnOJflOhexLzMMNLga62+/nh0JzvJ0aO/5a5MVgUFGS7Hu1P9P03o3fJkDCyw==", - "license": "ISC", - "dependencies": { - "d3-dispatch": "1 - 3", - "d3-drag": "2 - 3", - "d3-interpolate": "1 - 3", - "d3-selection": "2 - 3", - "d3-transition": "2 - 3" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/dagre": { - "version": "0.8.5", - "resolved": "https://registry.npmjs.org/dagre/-/dagre-0.8.5.tgz", - "integrity": "sha512-/aTqmnRta7x7MCCpExk7HQL2O4owCT2h8NT//9I1OQ9vt29Pa0BzSAkR5lwFUcQ7491yVi/3CXU9jQ5o0Mn2Sw==", - "license": "MIT", - "dependencies": { - "graphlib": "^2.1.8", - "lodash": "^4.17.15" - } - }, - "node_modules/dagre-d3-es": { - "version": "7.0.13", - "resolved": "https://registry.npmjs.org/dagre-d3-es/-/dagre-d3-es-7.0.13.tgz", - "integrity": "sha512-efEhnxpSuwpYOKRm/L5KbqoZmNNukHa/Flty4Wp62JRvgH2ojwVgPgdYyr4twpieZnyRDdIH7PY2mopX26+j2Q==", - "license": "MIT", - "dependencies": { - "d3": "^7.9.0", - "lodash-es": "^4.17.21" - } - }, - "node_modules/dayjs": { - "version": "1.11.19", - "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.11.19.tgz", - "integrity": "sha512-t5EcLVS6QPBNqM2z8fakk/NKel+Xzshgt8FFKAn+qwlD1pzZWxh0nVCrvFK7ZDb6XucZeF9z8C7CBWTRIVApAw==", - "license": "MIT" - }, - "node_modules/debug": { - "version": "4.4.3", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", - "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", - "license": "MIT", - "dependencies": { - "ms": "^2.1.3" - }, - "engines": { - "node": ">=6.0" - }, - "peerDependenciesMeta": { - "supports-color": { - "optional": true - } - } - }, - "node_modules/decode-named-character-reference": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/decode-named-character-reference/-/decode-named-character-reference-1.2.0.tgz", - "integrity": "sha512-c6fcElNV6ShtZXmsgNgFFV5tVX2PaV4g+MOAkb8eXHvn6sryJBrZa9r0zV6+dtTyoCKxtDy5tyQ5ZwQuidtd+Q==", - "license": "MIT", - "dependencies": { - "character-entities": "^2.0.0" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/deep-eql": { - "version": "5.0.2", - "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-5.0.2.tgz", - "integrity": "sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/deep-is": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", - "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/delaunator": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/delaunator/-/delaunator-5.0.1.tgz", - "integrity": "sha512-8nvh+XBe96aCESrGOqMp/84b13H9cdKbG5P2ejQCh4d4sK9RL4371qou9drQjMhvnPmhWl5hnmqbEE0fXr9Xnw==", - "license": "ISC", - "dependencies": { - "robust-predicates": "^3.0.2" - } - }, - "node_modules/dequal": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", - "integrity": "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==", - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/detect-libc": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz", - "integrity": "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==", - "dev": true, - "license": "Apache-2.0", - "optional": true, - "peer": true, - "engines": { - "node": ">=8" - } - }, - "node_modules/devlop": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/devlop/-/devlop-1.1.0.tgz", - "integrity": "sha512-RWmIqhcFf1lRYBvNmr7qTNuyCt/7/ns2jbpp1+PalgE/rDQcBT0fioSMUpJ93irlUhC5hrg4cYqe6U+0ImW0rA==", - "license": "MIT", - "dependencies": { - "dequal": "^2.0.0" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/didyoumean": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/didyoumean/-/didyoumean-1.2.2.tgz", - "integrity": "sha512-gxtyfqMg7GKyhQmb056K7M3xszy/myH8w+B4RT+QXBQsvAOdc3XymqDDPHx1BgPgsdAA5SIifona89YtRATDzw==", - "dev": true, - "license": "Apache-2.0" - }, - "node_modules/dlv": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/dlv/-/dlv-1.1.3.tgz", - "integrity": "sha512-+HlytyjlPKnIG8XuRG8WvmBP8xs8P71y+SKKS6ZXWoEgLuePxtDoUEiH7WkdePWrQ5JBpE6aoVqfZfJUQkjXwA==", - "dev": true, - "license": "MIT" - }, - "node_modules/dompurify": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/dompurify/-/dompurify-3.3.0.tgz", - "integrity": "sha512-r+f6MYR1gGN1eJv0TVQbhA7if/U7P87cdPl3HN5rikqaBSBxLiCb/b9O+2eG0cxz0ghyU+mU1QkbsOwERMYlWQ==", - "license": "(MPL-2.0 OR Apache-2.0)", - "optionalDependencies": { - "@types/trusted-types": "^2.0.7" - } - }, - "node_modules/driver.js": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/driver.js/-/driver.js-1.4.0.tgz", - "integrity": "sha512-Gm64jm6PmcU+si21sQhBrTAM1JvUrR0QhNmjkprNLxohOBzul9+pNHXgQaT9lW84gwg9GMLB3NZGuGolsz5uew==", - "license": "MIT" - }, - "node_modules/eastasianwidth": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", - "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==", - "dev": true, - "license": "MIT" - }, - "node_modules/electron-to-chromium": { - "version": "1.5.250", - "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.250.tgz", - "integrity": "sha512-/5UMj9IiGDMOFBnN4i7/Ry5onJrAGSbOGo3s9FEKmwobGq6xw832ccET0CE3CkkMBZ8GJSlUIesZofpyurqDXw==", - "dev": true, - "license": "ISC" - }, - "node_modules/emoji-regex": { - "version": "9.2.2", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", - "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==", - "dev": true, - "license": "MIT" - }, - "node_modules/es-module-lexer": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.7.0.tgz", - "integrity": "sha512-jEQoCwk8hyb2AZziIOLhDqpm5+2ww5uIE6lkO/6jcOCusfk6LhMHpXXfBLXTZ7Ydyt0j4VoUQv6uGNYbdW+kBA==", - "dev": true, - "license": "MIT" - }, - "node_modules/esbuild": { - "version": "0.25.12", - "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.25.12.tgz", - "integrity": "sha512-bbPBYYrtZbkt6Os6FiTLCTFxvq4tt3JKall1vRwshA3fdVztsLAatFaZobhkBC8/BrPetoa0oksYoKXoG4ryJg==", - "dev": true, - "hasInstallScript": true, - "license": "MIT", - "bin": { - "esbuild": "bin/esbuild" - }, - "engines": { - "node": ">=18" - }, - "optionalDependencies": { - "@esbuild/aix-ppc64": "0.25.12", - "@esbuild/android-arm": "0.25.12", - "@esbuild/android-arm64": "0.25.12", - "@esbuild/android-x64": "0.25.12", - "@esbuild/darwin-arm64": "0.25.12", - "@esbuild/darwin-x64": "0.25.12", - "@esbuild/freebsd-arm64": "0.25.12", - "@esbuild/freebsd-x64": "0.25.12", - "@esbuild/linux-arm": "0.25.12", - "@esbuild/linux-arm64": "0.25.12", - "@esbuild/linux-ia32": "0.25.12", - "@esbuild/linux-loong64": "0.25.12", - "@esbuild/linux-mips64el": "0.25.12", - "@esbuild/linux-ppc64": "0.25.12", - "@esbuild/linux-riscv64": "0.25.12", - "@esbuild/linux-s390x": "0.25.12", - "@esbuild/linux-x64": "0.25.12", - "@esbuild/netbsd-arm64": "0.25.12", - "@esbuild/netbsd-x64": "0.25.12", - "@esbuild/openbsd-arm64": "0.25.12", - "@esbuild/openbsd-x64": "0.25.12", - "@esbuild/openharmony-arm64": "0.25.12", - "@esbuild/sunos-x64": "0.25.12", - "@esbuild/win32-arm64": "0.25.12", - "@esbuild/win32-ia32": "0.25.12", - "@esbuild/win32-x64": "0.25.12" - } - }, - "node_modules/escalade": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", - "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/escape-string-regexp": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", - "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/eslint": { - "version": "9.39.1", - "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.39.1.tgz", - "integrity": "sha512-BhHmn2yNOFA9H9JmmIVKJmd288g9hrVRDkdoIgRCRuSySRUHH7r/DI6aAXW9T1WwUuY3DFgrcaqB+deURBLR5g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@eslint-community/eslint-utils": "^4.8.0", - "@eslint-community/regexpp": "^4.12.1", - "@eslint/config-array": "^0.21.1", - "@eslint/config-helpers": "^0.4.2", - "@eslint/core": "^0.17.0", - "@eslint/eslintrc": "^3.3.1", - "@eslint/js": "9.39.1", - "@eslint/plugin-kit": "^0.4.1", - "@humanfs/node": "^0.16.6", - "@humanwhocodes/module-importer": "^1.0.1", - "@humanwhocodes/retry": "^0.4.2", - "@types/estree": "^1.0.6", - "ajv": "^6.12.4", - "chalk": "^4.0.0", - "cross-spawn": "^7.0.6", - "debug": "^4.3.2", - "escape-string-regexp": "^4.0.0", - "eslint-scope": "^8.4.0", - "eslint-visitor-keys": "^4.2.1", - "espree": "^10.4.0", - "esquery": "^1.5.0", - "esutils": "^2.0.2", - "fast-deep-equal": "^3.1.3", - "file-entry-cache": "^8.0.0", - "find-up": "^5.0.0", - "glob-parent": "^6.0.2", - "ignore": "^5.2.0", - "imurmurhash": "^0.1.4", - "is-glob": "^4.0.0", - "json-stable-stringify-without-jsonify": "^1.0.1", - "lodash.merge": "^4.6.2", - "minimatch": "^3.1.2", - "natural-compare": "^1.4.0", - "optionator": "^0.9.3" - }, - "bin": { - "eslint": "bin/eslint.js" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://eslint.org/donate" - }, - "peerDependencies": { - "jiti": "*" - }, - "peerDependenciesMeta": { - "jiti": { - "optional": true - } - } - }, - "node_modules/eslint-config-prettier": { - "version": "10.1.8", - "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-10.1.8.tgz", - "integrity": "sha512-82GZUjRS0p/jganf6q1rEO25VSoHH0hKPCTrgillPjdI/3bgBhAE1QzHrHTizjpRvy6pGAvKjDJtk2pF9NDq8w==", - "dev": true, - "license": "MIT", - "bin": { - "eslint-config-prettier": "bin/cli.js" - }, - "funding": { - "url": "https://opencollective.com/eslint-config-prettier" - }, - "peerDependencies": { - "eslint": ">=7.0.0" - } - }, - "node_modules/eslint-plugin-prettier": { - "version": "5.5.4", - "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-5.5.4.tgz", - "integrity": "sha512-swNtI95SToIz05YINMA6Ox5R057IMAmWZ26GqPxusAp1TZzj+IdY9tXNWWD3vkF/wEqydCONcwjTFpxybBqZsg==", - "dev": true, - "license": "MIT", - "dependencies": { - "prettier-linter-helpers": "^1.0.0", - "synckit": "^0.11.7" - }, - "engines": { - "node": "^14.18.0 || >=16.0.0" - }, - "funding": { - "url": "https://opencollective.com/eslint-plugin-prettier" - }, - "peerDependencies": { - "@types/eslint": ">=8.0.0", - "eslint": ">=8.0.0", - "eslint-config-prettier": ">= 7.0.0 <10.0.0 || >=10.1.0", - "prettier": ">=3.0.0" - }, - "peerDependenciesMeta": { - "@types/eslint": { - "optional": true - }, - "eslint-config-prettier": { - "optional": true - } - } - }, - "node_modules/eslint-plugin-react-hooks": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-5.2.0.tgz", - "integrity": "sha512-+f15FfK64YQwZdJNELETdn5ibXEUQmW1DZL6KXhNnc2heoy/sg9VJJeT7n8TlMWouzWqSWavFkIhHyIbIAEapg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=10" - }, - "peerDependencies": { - "eslint": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 || ^9.0.0" - } - }, - "node_modules/eslint-plugin-react-refresh": { - "version": "0.4.24", - "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.4.24.tgz", - "integrity": "sha512-nLHIW7TEq3aLrEYWpVaJ1dRgFR+wLDPN8e8FpYAql/bMV2oBEfC37K0gLEGgv9fy66juNShSMV8OkTqzltcG/w==", - "dev": true, - "license": "MIT", - "peerDependencies": { - "eslint": ">=8.40" - } - }, - "node_modules/eslint-scope": { - "version": "8.4.0", - "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.4.0.tgz", - "integrity": "sha512-sNXOfKCn74rt8RICKMvJS7XKV/Xk9kA7DyJr8mJik3S7Cwgy3qlkkmyS2uQB3jiJg6VNdZd/pDBJu0nvG2NlTg==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "esrecurse": "^4.3.0", - "estraverse": "^5.2.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/eslint-visitor-keys": { - "version": "4.2.1", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.1.tgz", - "integrity": "sha512-Uhdk5sfqcee/9H/rCOJikYz67o0a2Tw2hGRPOG2Y1R2dg7brRe1uG0yaNQDHu+TO/uQPF/5eCapvYSmHUjt7JQ==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/espree": { - "version": "10.4.0", - "resolved": "https://registry.npmjs.org/espree/-/espree-10.4.0.tgz", - "integrity": "sha512-j6PAQ2uUr79PZhBjP5C5fhl8e39FmRnOjsD5lGnWrFU8i2G776tBK7+nP8KuQUTTyAZUwfQqXAgrVH5MbH9CYQ==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "acorn": "^8.15.0", - "acorn-jsx": "^5.3.2", - "eslint-visitor-keys": "^4.2.1" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/esquery": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz", - "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==", - "dev": true, - "license": "BSD-3-Clause", - "dependencies": { - "estraverse": "^5.1.0" - }, - "engines": { - "node": ">=0.10" - } - }, - "node_modules/esrecurse": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", - "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "estraverse": "^5.2.0" - }, - "engines": { - "node": ">=4.0" - } - }, - "node_modules/estraverse": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", - "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", - "dev": true, - "license": "BSD-2-Clause", - "engines": { - "node": ">=4.0" - } - }, - "node_modules/estree-util-is-identifier-name": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/estree-util-is-identifier-name/-/estree-util-is-identifier-name-3.0.0.tgz", - "integrity": "sha512-hFtqIDZTIUZ9BXLb8y4pYGyk6+wekIivNVTcmvk8NoOh+VeRn5y6cEHzbURrWbfp1fIqdVipilzj+lfaadNZmg==", - "license": "MIT", - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/estree-walker": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-3.0.3.tgz", - "integrity": "sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/estree": "^1.0.0" - } - }, - "node_modules/esutils": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", - "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", - "dev": true, - "license": "BSD-2-Clause", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/expect-type": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/expect-type/-/expect-type-1.3.0.tgz", - "integrity": "sha512-knvyeauYhqjOYvQ66MznSMs83wmHrCycNEN6Ao+2AeYEfxUIkuiVxdEa1qlGEPK+We3n0THiDciYSsCcgW/DoA==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": ">=12.0.0" - } - }, - "node_modules/exsolve": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/exsolve/-/exsolve-1.0.8.tgz", - "integrity": "sha512-LmDxfWXwcTArk8fUEnOfSZpHOJ6zOMUJKOtFLFqJLoKJetuQG874Uc7/Kki7zFLzYybmZhp1M7+98pfMqeX8yA==", - "license": "MIT" - }, - "node_modules/extend": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", - "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", - "license": "MIT" - }, - "node_modules/fast-deep-equal": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", - "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", - "dev": true, - "license": "MIT" - }, - "node_modules/fast-diff": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/fast-diff/-/fast-diff-1.3.0.tgz", - "integrity": "sha512-VxPP4NqbUjj6MaAOafWeUn2cXWLcCtljklUtZf0Ind4XQ+QPtmA0b18zZy0jIQx+ExRVCR/ZQpBmik5lXshNsw==", - "dev": true, - "license": "Apache-2.0" - }, - "node_modules/fast-glob": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz", - "integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@nodelib/fs.stat": "^2.0.2", - "@nodelib/fs.walk": "^1.2.3", - "glob-parent": "^5.1.2", - "merge2": "^1.3.0", - "micromatch": "^4.0.8" - }, - "engines": { - "node": ">=8.6.0" - } - }, - "node_modules/fast-glob/node_modules/glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "dev": true, - "license": "ISC", - "dependencies": { - "is-glob": "^4.0.1" - }, - "engines": { - "node": ">= 6" - } - }, - "node_modules/fast-json-stable-stringify": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", - "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", - "dev": true, - "license": "MIT" - }, - "node_modules/fast-levenshtein": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", - "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", - "dev": true, - "license": "MIT" - }, - "node_modules/fastq": { - "version": "1.19.1", - "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.19.1.tgz", - "integrity": "sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ==", - "dev": true, - "license": "ISC", - "dependencies": { - "reusify": "^1.0.4" - } - }, - "node_modules/fault": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/fault/-/fault-1.0.4.tgz", - "integrity": "sha512-CJ0HCB5tL5fYTEA7ToAq5+kTwd++Borf1/bifxd9iT70QcXr4MRrO3Llf8Ifs70q+SJcGHFtnIE/Nw6giCtECA==", - "license": "MIT", - "dependencies": { - "format": "^0.2.0" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/file-entry-cache": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz", - "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "flat-cache": "^4.0.0" - }, - "engines": { - "node": ">=16.0.0" - } - }, - "node_modules/fill-range": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", - "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", - "dev": true, - "license": "MIT", - "dependencies": { - "to-regex-range": "^5.0.1" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/find-up": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", - "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", - "dev": true, - "license": "MIT", - "dependencies": { - "locate-path": "^6.0.0", - "path-exists": "^4.0.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/flat-cache": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz", - "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==", - "dev": true, - "license": "MIT", - "dependencies": { - "flatted": "^3.2.9", - "keyv": "^4.5.4" - }, - "engines": { - "node": ">=16" - } - }, - "node_modules/flatted": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.3.tgz", - "integrity": "sha512-GX+ysw4PBCz0PzosHDepZGANEuFCMLrnRTiEy9McGjmkCQYwRq4A/X786G/fjM/+OjsWSU1ZrY5qyARZmO/uwg==", - "dev": true, - "license": "ISC" - }, - "node_modules/foreground-child": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.3.1.tgz", - "integrity": "sha512-gIXjKqtFuWEgzFRJA9WCQeSJLZDjgJUOMCMzxtvFq/37KojM1BFGufqsCy0r4qSQmYLsZYMeyRqzIWOMup03sw==", - "dev": true, - "license": "ISC", - "dependencies": { - "cross-spawn": "^7.0.6", - "signal-exit": "^4.0.1" - }, - "engines": { - "node": ">=14" - }, - "funding": { - "url": "https://github.com/sponsors/isaacs" - } - }, - "node_modules/format": { - "version": "0.2.2", - "resolved": "https://registry.npmjs.org/format/-/format-0.2.2.tgz", - "integrity": "sha512-wzsgA6WOq+09wrU1tsJ09udeR/YZRaeArL9e1wPbFg3GG2yDnC2ldKpxs4xunpFF9DgqCqOIra3bc1HWrJ37Ww==", - "engines": { - "node": ">=0.4.x" - } - }, - "node_modules/fraction.js": { - "version": "5.3.4", - "resolved": "https://registry.npmjs.org/fraction.js/-/fraction.js-5.3.4.tgz", - "integrity": "sha512-1X1NTtiJphryn/uLQz3whtY6jK3fTqoE3ohKs0tT+Ujr1W59oopxmoEh7Lu5p6vBaPbgoM0bzveAW4Qi5RyWDQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": "*" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/rawify" - } - }, - "node_modules/framer-motion": { - "version": "12.23.24", - "resolved": "https://registry.npmjs.org/framer-motion/-/framer-motion-12.23.24.tgz", - "integrity": "sha512-HMi5HRoRCTou+3fb3h9oTLyJGBxHfW+HnNE25tAXOvVx/IvwMHK0cx7IR4a2ZU6sh3IX1Z+4ts32PcYBOqka8w==", - "license": "MIT", - "dependencies": { - "motion-dom": "^12.23.23", - "motion-utils": "^12.23.6", - "tslib": "^2.4.0" - }, - "peerDependencies": { - "@emotion/is-prop-valid": "*", - "react": "^18.0.0 || ^19.0.0", - "react-dom": "^18.0.0 || ^19.0.0" - }, - "peerDependenciesMeta": { - "@emotion/is-prop-valid": { - "optional": true - }, - "react": { - "optional": true - }, - "react-dom": { - "optional": true - } - } - }, - "node_modules/fsevents": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", - "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", - "dev": true, - "hasInstallScript": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": "^8.16.0 || ^10.6.0 || >=11.0.0" - } - }, - "node_modules/function-bind": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", - "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", - "dev": true, - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/gensync": { - "version": "1.0.0-beta.2", - "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", - "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/glob": { - "version": "10.5.0", - "resolved": "https://registry.npmjs.org/glob/-/glob-10.5.0.tgz", - "integrity": "sha512-DfXN8DfhJ7NH3Oe7cFmu3NCu1wKbkReJ8TorzSAFbSKrlNaQSKfIzqYqVY8zlbs2NLBbWpRiU52GX2PbaBVNkg==", - "dev": true, - "license": "ISC", - "dependencies": { - "foreground-child": "^3.1.0", - "jackspeak": "^3.1.2", - "minimatch": "^9.0.4", - "minipass": "^7.1.2", - "package-json-from-dist": "^1.0.0", - "path-scurry": "^1.11.1" - }, - "bin": { - "glob": "dist/esm/bin.mjs" - }, - "funding": { - "url": "https://github.com/sponsors/isaacs" - } - }, - "node_modules/glob-parent": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", - "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", - "dev": true, - "license": "ISC", - "dependencies": { - "is-glob": "^4.0.3" - }, - "engines": { - "node": ">=10.13.0" - } - }, - "node_modules/glob/node_modules/brace-expansion": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.2.tgz", - "integrity": "sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "balanced-match": "^1.0.0" - } - }, - "node_modules/glob/node_modules/minimatch": { - "version": "9.0.5", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", - "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", - "dev": true, - "license": "ISC", - "dependencies": { - "brace-expansion": "^2.0.1" - }, - "engines": { - "node": ">=16 || 14 >=14.17" - }, - "funding": { - "url": "https://github.com/sponsors/isaacs" - } - }, - "node_modules/globals": { - "version": "16.5.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-16.5.0.tgz", - "integrity": "sha512-c/c15i26VrJ4IRt5Z89DnIzCGDn9EcebibhAOjw5ibqEHsE1wLUgkPn9RDmNcUKyU87GeaL633nyJ+pplFR2ZQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/graphemer": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", - "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", - "dev": true, - "license": "MIT" - }, - "node_modules/graphlib": { - "version": "2.1.8", - "resolved": "https://registry.npmjs.org/graphlib/-/graphlib-2.1.8.tgz", - "integrity": "sha512-jcLLfkpoVGmH7/InMC/1hIvOPSUh38oJtGhvrOFGzioE1DZ+0YW16RgmOJhHiuWTvGiJQ9Z1Ik43JvkRPRvE+A==", - "license": "MIT", - "dependencies": { - "lodash": "^4.17.15" - } - }, - "node_modules/hachure-fill": { - "version": "0.5.2", - "resolved": "https://registry.npmjs.org/hachure-fill/-/hachure-fill-0.5.2.tgz", - "integrity": "sha512-3GKBOn+m2LX9iq+JC1064cSFprJY4jL1jCXTcpnfER5HYE2l/4EfWSGzkPa/ZDBmYI0ZOEj5VHV/eKnPGkHuOg==", - "license": "MIT" - }, - "node_modules/has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/hasown": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", - "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "function-bind": "^1.1.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/hast-util-parse-selector": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/hast-util-parse-selector/-/hast-util-parse-selector-4.0.0.tgz", - "integrity": "sha512-wkQCkSYoOGCRKERFWcxMVMOcYE2K1AaNLU8DXS9arxnLOUEWbOXKXiJUNzEpqZ3JOKpnha3jkFrumEjVliDe7A==", - "license": "MIT", - "dependencies": { - "@types/hast": "^3.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/hast-util-to-jsx-runtime": { - "version": "2.3.6", - "resolved": "https://registry.npmjs.org/hast-util-to-jsx-runtime/-/hast-util-to-jsx-runtime-2.3.6.tgz", - "integrity": "sha512-zl6s8LwNyo1P9uw+XJGvZtdFF1GdAkOg8ujOw+4Pyb76874fLps4ueHXDhXWdk6YHQ6OgUtinliG7RsYvCbbBg==", - "license": "MIT", - "dependencies": { - "@types/estree": "^1.0.0", - "@types/hast": "^3.0.0", - "@types/unist": "^3.0.0", - "comma-separated-tokens": "^2.0.0", - "devlop": "^1.0.0", - "estree-util-is-identifier-name": "^3.0.0", - "hast-util-whitespace": "^3.0.0", - "mdast-util-mdx-expression": "^2.0.0", - "mdast-util-mdx-jsx": "^3.0.0", - "mdast-util-mdxjs-esm": "^2.0.0", - "property-information": "^7.0.0", - "space-separated-tokens": "^2.0.0", - "style-to-js": "^1.0.0", - "unist-util-position": "^5.0.0", - "vfile-message": "^4.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/hast-util-whitespace": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/hast-util-whitespace/-/hast-util-whitespace-3.0.0.tgz", - "integrity": "sha512-88JUN06ipLwsnv+dVn+OIYOvAuvBMy/Qoi6O7mQHxdPXpjy+Cd6xRkWwux7DKO+4sYILtLBRIKgsdpS2gQc7qw==", - "license": "MIT", - "dependencies": { - "@types/hast": "^3.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/hastscript": { - "version": "9.0.1", - "resolved": "https://registry.npmjs.org/hastscript/-/hastscript-9.0.1.tgz", - "integrity": "sha512-g7df9rMFX/SPi34tyGCyUBREQoKkapwdY/T04Qn9TDWfHhAYt4/I0gMVirzK5wEzeUqIjEB+LXC/ypb7Aqno5w==", - "license": "MIT", - "dependencies": { - "@types/hast": "^3.0.0", - "comma-separated-tokens": "^2.0.0", - "hast-util-parse-selector": "^4.0.0", - "property-information": "^7.0.0", - "space-separated-tokens": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/highlight.js": { - "version": "10.7.3", - "resolved": "https://registry.npmjs.org/highlight.js/-/highlight.js-10.7.3.tgz", - "integrity": "sha512-tzcUFauisWKNHaRkN4Wjl/ZA07gENAjFl3J/c480dprkGTg5EQstgaNFqBfUqCq54kZRIEcreTsAgF/m2quD7A==", - "license": "BSD-3-Clause", - "engines": { - "node": "*" - } - }, - "node_modules/highlightjs-vue": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/highlightjs-vue/-/highlightjs-vue-1.0.0.tgz", - "integrity": "sha512-PDEfEF102G23vHmPhLyPboFCD+BkMGu+GuJe2d9/eH4FsCwvgBpnc9n0pGE+ffKdph38s6foEZiEjdgHdzp+IA==", - "license": "CC0-1.0" - }, - "node_modules/html-escaper": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", - "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", - "dev": true, - "license": "MIT" - }, - "node_modules/html-to-image": { - "version": "1.11.13", - "resolved": "https://registry.npmjs.org/html-to-image/-/html-to-image-1.11.13.tgz", - "integrity": "sha512-cuOPoI7WApyhBElTTb9oqsawRvZ0rHhaHwghRLlTuffoD1B2aDemlCruLeZrUIIdvG7gs9xeELEPm6PhuASqrg==", - "license": "MIT" - }, - "node_modules/html-url-attributes": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/html-url-attributes/-/html-url-attributes-3.0.1.tgz", - "integrity": "sha512-ol6UPyBWqsrO6EJySPz2O7ZSr856WDrEzM5zMqp+FJJLGMW35cLYmmZnl0vztAZxRUoNZJFTCohfjuIJ8I4QBQ==", - "license": "MIT", - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/iconv-lite": { - "version": "0.6.3", - "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", - "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", - "license": "MIT", - "dependencies": { - "safer-buffer": ">= 2.1.2 < 3.0.0" - }, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/idb": { - "version": "8.0.3", - "resolved": "https://registry.npmjs.org/idb/-/idb-8.0.3.tgz", - "integrity": "sha512-LtwtVyVYO5BqRvcsKuB2iUMnHwPVByPCXFXOpuU96IZPPoPN6xjOGxZQ74pgSVVLQWtUOYgyeL4GE98BY5D3wg==", - "license": "ISC" - }, - "node_modules/ignore": { - "version": "5.3.2", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", - "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 4" - } - }, - "node_modules/import-fresh": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz", - "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "parent-module": "^1.0.0", - "resolve-from": "^4.0.0" - }, - "engines": { - "node": ">=6" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/imurmurhash": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", - "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.8.19" - } - }, - "node_modules/inline-style-parser": { - "version": "0.2.6", - "resolved": "https://registry.npmjs.org/inline-style-parser/-/inline-style-parser-0.2.6.tgz", - "integrity": "sha512-gtGXVaBdl5mAes3rPcMedEBm12ibjt1kDMFfheul1wUAOVEJW60voNdMVzVkfLN06O7ZaD/rxhfKgtlgtTbMjg==", - "license": "MIT" - }, - "node_modules/internmap": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/internmap/-/internmap-2.0.3.tgz", - "integrity": "sha512-5Hh7Y1wQbvY5ooGgPbDaL5iYLAPzMTUrjMulskHLH6wnv/A+1q5rgEaiuqEjB+oxGXIVZs1FF+R/KPN3ZSQYYg==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/is-alphabetical": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/is-alphabetical/-/is-alphabetical-2.0.1.tgz", - "integrity": "sha512-FWyyY60MeTNyeSRpkM2Iry0G9hpr7/9kD40mD/cGQEuilcZYS4okz8SN2Q6rLCJ8gbCt6fN+rC+6tMGS99LaxQ==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/is-alphanumerical": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/is-alphanumerical/-/is-alphanumerical-2.0.1.tgz", - "integrity": "sha512-hmbYhX/9MUMF5uh7tOXyK/n0ZvWpad5caBA17GsC6vyuCqaWliRG5K1qS9inmUhEMaOBIW7/whAnSwveW/LtZw==", - "license": "MIT", - "dependencies": { - "is-alphabetical": "^2.0.0", - "is-decimal": "^2.0.0" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/is-binary-path": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", - "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", - "dev": true, - "license": "MIT", - "dependencies": { - "binary-extensions": "^2.0.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/is-core-module": { - "version": "2.16.1", - "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", - "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==", - "dev": true, - "license": "MIT", - "dependencies": { - "hasown": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-decimal": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/is-decimal/-/is-decimal-2.0.1.tgz", - "integrity": "sha512-AAB9hiomQs5DXWcRB1rqsxGUstbRroFOPPVAomNk/3XHR5JyEZChOyTWe2oayKnsSsr/kcGqF+z6yuH6HHpN0A==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/is-extglob": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", - "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/is-fullwidth-code-point": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/is-glob": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", - "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-extglob": "^2.1.1" - }, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/is-hexadecimal": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/is-hexadecimal/-/is-hexadecimal-2.0.1.tgz", - "integrity": "sha512-DgZQp241c8oO6cA1SbTEWiXeoxV42vlcJxgH+B3hi1AiqqKruZR3ZGF8In3fj4+/y/7rHvlOZLZtgJ/4ttYGZg==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/is-number": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.12.0" - } - }, - "node_modules/is-plain-obj": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-4.1.0.tgz", - "integrity": "sha512-+Pgi+vMuUNkJyExiMBt5IlFoMyKnr5zhJ4Uspz58WOhBF5QoIZkFyNHIbBAtHwzVAgk5RtndVNsDRN61/mmDqg==", - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/isexe": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", - "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", - "dev": true, - "license": "ISC" - }, - "node_modules/istanbul-lib-coverage": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.2.2.tgz", - "integrity": "sha512-O8dpsF+r0WV/8MNRKfnmrtCWhuKjxrq2w+jpzBL5UZKTi2LeVWnWOmWRxFlesJONmc+wLAGvKQZEOanko0LFTg==", - "dev": true, - "license": "BSD-3-Clause", - "engines": { - "node": ">=8" - } - }, - "node_modules/istanbul-lib-report": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.1.tgz", - "integrity": "sha512-GCfE1mtsHGOELCU8e/Z7YWzpmybrx/+dSTfLrvY8qRmaY6zXTKWn6WQIjaAFw069icm6GVMNkgu0NzI4iPZUNw==", - "dev": true, - "license": "BSD-3-Clause", - "dependencies": { - "istanbul-lib-coverage": "^3.0.0", - "make-dir": "^4.0.0", - "supports-color": "^7.1.0" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/istanbul-lib-source-maps": { - "version": "5.0.6", - "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-5.0.6.tgz", - "integrity": "sha512-yg2d+Em4KizZC5niWhQaIomgf5WlL4vOOjZ5xGCmF8SnPE/mDWWXgvRExdcpCgh9lLRRa1/fSYp2ymmbJ1pI+A==", - "dev": true, - "license": "BSD-3-Clause", - "dependencies": { - "@jridgewell/trace-mapping": "^0.3.23", - "debug": "^4.1.1", - "istanbul-lib-coverage": "^3.0.0" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/istanbul-reports": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.2.0.tgz", - "integrity": "sha512-HGYWWS/ehqTV3xN10i23tkPkpH46MLCIMFNCaaKNavAXTF1RkqxawEPtnjnGZ6XKSInBKkiOA5BKS+aZiY3AvA==", - "dev": true, - "license": "BSD-3-Clause", - "dependencies": { - "html-escaper": "^2.0.0", - "istanbul-lib-report": "^3.0.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/jackspeak": { - "version": "3.4.3", - "resolved": "https://registry.npmjs.org/jackspeak/-/jackspeak-3.4.3.tgz", - "integrity": "sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==", - "dev": true, - "license": "BlueOak-1.0.0", - "dependencies": { - "@isaacs/cliui": "^8.0.2" - }, - "funding": { - "url": "https://github.com/sponsors/isaacs" - }, - "optionalDependencies": { - "@pkgjs/parseargs": "^0.11.0" - } - }, - "node_modules/jiti": { - "version": "2.6.1", - "resolved": "https://registry.npmjs.org/jiti/-/jiti-2.6.1.tgz", - "integrity": "sha512-ekilCSN1jwRvIbgeg/57YFh8qQDNbwDb9xT/qu2DAHbFFZUicIl4ygVaAvzveMhMVr3LnpSKTNnwt8PoOfmKhQ==", - "dev": true, - "license": "MIT", - "optional": true, - "peer": true, - "bin": { - "jiti": "lib/jiti-cli.mjs" - } - }, - "node_modules/js-tokens": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", - "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", - "license": "MIT" - }, - "node_modules/js-yaml": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.1.tgz", - "integrity": "sha512-qQKT4zQxXl8lLwBtHMWwaTcGfFOZviOJet3Oy/xmGk2gZH677CJM9EvtfdSkgWcATZhj/55JZ0rmy3myCT5lsA==", - "dev": true, - "license": "MIT", - "dependencies": { - "argparse": "^2.0.1" - }, - "bin": { - "js-yaml": "bin/js-yaml.js" - } - }, - "node_modules/jsesc": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz", - "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==", - "dev": true, - "license": "MIT", - "bin": { - "jsesc": "bin/jsesc" - }, - "engines": { - "node": ">=6" - } - }, - "node_modules/json-buffer": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", - "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/json-schema-traverse": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", - "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", - "dev": true, - "license": "MIT" - }, - "node_modules/json-stable-stringify-without-jsonify": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", - "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", - "dev": true, - "license": "MIT" - }, - "node_modules/json5": { - "version": "2.2.3", - "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", - "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", - "dev": true, - "license": "MIT", - "bin": { - "json5": "lib/cli.js" - }, - "engines": { - "node": ">=6" - } - }, - "node_modules/katex": { - "version": "0.16.25", - "resolved": "https://registry.npmjs.org/katex/-/katex-0.16.25.tgz", - "integrity": "sha512-woHRUZ/iF23GBP1dkDQMh1QBad9dmr8/PAwNA54VrSOVYgI12MAcE14TqnDdQOdzyEonGzMepYnqBMYdsoAr8Q==", - "funding": [ - "https://opencollective.com/katex", - "https://github.com/sponsors/katex" - ], - "license": "MIT", - "dependencies": { - "commander": "^8.3.0" - }, - "bin": { - "katex": "cli.js" - } - }, - "node_modules/katex/node_modules/commander": { - "version": "8.3.0", - "resolved": "https://registry.npmjs.org/commander/-/commander-8.3.0.tgz", - "integrity": "sha512-OkTL9umf+He2DZkUq8f8J9of7yL6RJKI24dVITBmNfZBmri9zYZQrKkuXiKhyfPSu8tUhnVBB1iKXevvnlR4Ww==", - "license": "MIT", - "engines": { - "node": ">= 12" - } - }, - "node_modules/keyv": { - "version": "4.5.4", - "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", - "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", - "dev": true, - "license": "MIT", - "dependencies": { - "json-buffer": "3.0.1" - } - }, - "node_modules/khroma": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/khroma/-/khroma-2.1.0.tgz", - "integrity": "sha512-Ls993zuzfayK269Svk9hzpeGUKob/sIgZzyHYdjQoAdQetRKpOLj+k/QQQ/6Qi0Yz65mlROrfd+Ev+1+7dz9Kw==" - }, - "node_modules/kolorist": { - "version": "1.8.0", - "resolved": "https://registry.npmjs.org/kolorist/-/kolorist-1.8.0.tgz", - "integrity": "sha512-Y+60/zizpJ3HRH8DCss+q95yr6145JXZo46OTpFvDZWLfRCE4qChOyk1b26nMaNpfHHgxagk9dXT5OP0Tfe+dQ==", - "license": "MIT" - }, - "node_modules/langium": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/langium/-/langium-3.3.1.tgz", - "integrity": "sha512-QJv/h939gDpvT+9SiLVlY7tZC3xB2qK57v0J04Sh9wpMb6MP1q8gB21L3WIo8T5P1MSMg3Ep14L7KkDCFG3y4w==", - "license": "MIT", - "dependencies": { - "chevrotain": "~11.0.3", - "chevrotain-allstar": "~0.3.0", - "vscode-languageserver": "~9.0.1", - "vscode-languageserver-textdocument": "~1.0.11", - "vscode-uri": "~3.0.8" - }, - "engines": { - "node": ">=16.0.0" - } - }, - "node_modules/layout-base": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/layout-base/-/layout-base-1.0.2.tgz", - "integrity": "sha512-8h2oVEZNktL4BH2JCOI90iD1yXwL6iNW7KcCKT2QZgQJR2vbqDsldCTPRU9NifTCqHZci57XvQQ15YTu+sTYPg==", - "license": "MIT" - }, - "node_modules/levn": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", - "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "prelude-ls": "^1.2.1", - "type-check": "~0.4.0" - }, - "engines": { - "node": ">= 0.8.0" - } - }, - "node_modules/lightningcss": { - "version": "1.30.2", - "resolved": "https://registry.npmjs.org/lightningcss/-/lightningcss-1.30.2.tgz", - "integrity": "sha512-utfs7Pr5uJyyvDETitgsaqSyjCb2qNRAtuqUeWIAKztsOYdcACf2KtARYXg2pSvhkt+9NfoaNY7fxjl6nuMjIQ==", - "dev": true, - "license": "MPL-2.0", - "optional": true, - "peer": true, - "dependencies": { - "detect-libc": "^2.0.3" - }, - "engines": { - "node": ">= 12.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - }, - "optionalDependencies": { - "lightningcss-android-arm64": "1.30.2", - "lightningcss-darwin-arm64": "1.30.2", - "lightningcss-darwin-x64": "1.30.2", - "lightningcss-freebsd-x64": "1.30.2", - "lightningcss-linux-arm-gnueabihf": "1.30.2", - "lightningcss-linux-arm64-gnu": "1.30.2", - "lightningcss-linux-arm64-musl": "1.30.2", - "lightningcss-linux-x64-gnu": "1.30.2", - "lightningcss-linux-x64-musl": "1.30.2", - "lightningcss-win32-arm64-msvc": "1.30.2", - "lightningcss-win32-x64-msvc": "1.30.2" - } - }, - "node_modules/lightningcss-android-arm64": { - "version": "1.30.2", - "resolved": "https://registry.npmjs.org/lightningcss-android-arm64/-/lightningcss-android-arm64-1.30.2.tgz", - "integrity": "sha512-BH9sEdOCahSgmkVhBLeU7Hc9DWeZ1Eb6wNS6Da8igvUwAe0sqROHddIlvU06q3WyXVEOYDZ6ykBZQnjTbmo4+A==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MPL-2.0", - "optional": true, - "os": [ - "android" - ], - "peer": true, - "engines": { - "node": ">= 12.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/lightningcss-darwin-arm64": { - "version": "1.30.2", - "resolved": "https://registry.npmjs.org/lightningcss-darwin-arm64/-/lightningcss-darwin-arm64-1.30.2.tgz", - "integrity": "sha512-ylTcDJBN3Hp21TdhRT5zBOIi73P6/W0qwvlFEk22fkdXchtNTOU4Qc37SkzV+EKYxLouZ6M4LG9NfZ1qkhhBWA==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MPL-2.0", - "optional": true, - "os": [ - "darwin" - ], - "peer": true, - "engines": { - "node": ">= 12.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/lightningcss-darwin-x64": { - "version": "1.30.2", - "resolved": "https://registry.npmjs.org/lightningcss-darwin-x64/-/lightningcss-darwin-x64-1.30.2.tgz", - "integrity": "sha512-oBZgKchomuDYxr7ilwLcyms6BCyLn0z8J0+ZZmfpjwg9fRVZIR5/GMXd7r9RH94iDhld3UmSjBM6nXWM2TfZTQ==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MPL-2.0", - "optional": true, - "os": [ - "darwin" - ], - "peer": true, - "engines": { - "node": ">= 12.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/lightningcss-freebsd-x64": { - "version": "1.30.2", - "resolved": "https://registry.npmjs.org/lightningcss-freebsd-x64/-/lightningcss-freebsd-x64-1.30.2.tgz", - "integrity": "sha512-c2bH6xTrf4BDpK8MoGG4Bd6zAMZDAXS569UxCAGcA7IKbHNMlhGQ89eRmvpIUGfKWNVdbhSbkQaWhEoMGmGslA==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MPL-2.0", - "optional": true, - "os": [ - "freebsd" - ], - "peer": true, - "engines": { - "node": ">= 12.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/lightningcss-linux-arm-gnueabihf": { - "version": "1.30.2", - "resolved": "https://registry.npmjs.org/lightningcss-linux-arm-gnueabihf/-/lightningcss-linux-arm-gnueabihf-1.30.2.tgz", - "integrity": "sha512-eVdpxh4wYcm0PofJIZVuYuLiqBIakQ9uFZmipf6LF/HRj5Bgm0eb3qL/mr1smyXIS1twwOxNWndd8z0E374hiA==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MPL-2.0", - "optional": true, - "os": [ - "linux" - ], - "peer": true, - "engines": { - "node": ">= 12.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/lightningcss-linux-arm64-gnu": { - "version": "1.30.2", - "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-gnu/-/lightningcss-linux-arm64-gnu-1.30.2.tgz", - "integrity": "sha512-UK65WJAbwIJbiBFXpxrbTNArtfuznvxAJw4Q2ZGlU8kPeDIWEX1dg3rn2veBVUylA2Ezg89ktszWbaQnxD/e3A==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MPL-2.0", - "optional": true, - "os": [ - "linux" - ], - "peer": true, - "engines": { - "node": ">= 12.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/lightningcss-linux-arm64-musl": { - "version": "1.30.2", - "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-musl/-/lightningcss-linux-arm64-musl-1.30.2.tgz", - "integrity": "sha512-5Vh9dGeblpTxWHpOx8iauV02popZDsCYMPIgiuw97OJ5uaDsL86cnqSFs5LZkG3ghHoX5isLgWzMs+eD1YzrnA==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MPL-2.0", - "optional": true, - "os": [ - "linux" - ], - "peer": true, - "engines": { - "node": ">= 12.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/lightningcss-linux-x64-gnu": { - "version": "1.30.2", - "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-gnu/-/lightningcss-linux-x64-gnu-1.30.2.tgz", - "integrity": "sha512-Cfd46gdmj1vQ+lR6VRTTadNHu6ALuw2pKR9lYq4FnhvgBc4zWY1EtZcAc6EffShbb1MFrIPfLDXD6Xprbnni4w==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MPL-2.0", - "optional": true, - "os": [ - "linux" - ], - "peer": true, - "engines": { - "node": ">= 12.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/lightningcss-linux-x64-musl": { - "version": "1.30.2", - "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-musl/-/lightningcss-linux-x64-musl-1.30.2.tgz", - "integrity": "sha512-XJaLUUFXb6/QG2lGIW6aIk6jKdtjtcffUT0NKvIqhSBY3hh9Ch+1LCeH80dR9q9LBjG3ewbDjnumefsLsP6aiA==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MPL-2.0", - "optional": true, - "os": [ - "linux" - ], - "peer": true, - "engines": { - "node": ">= 12.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/lightningcss-win32-arm64-msvc": { - "version": "1.30.2", - "resolved": "https://registry.npmjs.org/lightningcss-win32-arm64-msvc/-/lightningcss-win32-arm64-msvc-1.30.2.tgz", - "integrity": "sha512-FZn+vaj7zLv//D/192WFFVA0RgHawIcHqLX9xuWiQt7P0PtdFEVaxgF9rjM/IRYHQXNnk61/H/gb2Ei+kUQ4xQ==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MPL-2.0", - "optional": true, - "os": [ - "win32" - ], - "peer": true, - "engines": { - "node": ">= 12.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/lightningcss-win32-x64-msvc": { - "version": "1.30.2", - "resolved": "https://registry.npmjs.org/lightningcss-win32-x64-msvc/-/lightningcss-win32-x64-msvc-1.30.2.tgz", - "integrity": "sha512-5g1yc73p+iAkid5phb4oVFMB45417DkRevRbt/El/gKXJk4jid+vPFF/AXbxn05Aky8PapwzZrdJShv5C0avjw==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MPL-2.0", - "optional": true, - "os": [ - "win32" - ], - "peer": true, - "engines": { - "node": ">= 12.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/lilconfig": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/lilconfig/-/lilconfig-3.1.3.tgz", - "integrity": "sha512-/vlFKAoH5Cgt3Ie+JLhRbwOsCQePABiU3tJ1egGvyQ+33R/vcwM2Zl2QR/LzjsBeItPt3oSVXapn+m4nQDvpzw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=14" - }, - "funding": { - "url": "https://github.com/sponsors/antonk52" - } - }, - "node_modules/lines-and-columns": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", - "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", - "dev": true, - "license": "MIT" - }, - "node_modules/local-pkg": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/local-pkg/-/local-pkg-1.1.2.tgz", - "integrity": "sha512-arhlxbFRmoQHl33a0Zkle/YWlmNwoyt6QNZEIJcqNbdrsix5Lvc4HyyI3EnwxTYlZYc32EbYrQ8SzEZ7dqgg9A==", - "license": "MIT", - "dependencies": { - "mlly": "^1.7.4", - "pkg-types": "^2.3.0", - "quansync": "^0.2.11" - }, - "engines": { - "node": ">=14" - }, - "funding": { - "url": "https://github.com/sponsors/antfu" - } - }, - "node_modules/locate-path": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", - "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", - "dev": true, - "license": "MIT", - "dependencies": { - "p-locate": "^5.0.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/lodash": { - "version": "4.17.21", - "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", - "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", - "license": "MIT" - }, - "node_modules/lodash-es": { - "version": "4.17.21", - "resolved": "https://registry.npmjs.org/lodash-es/-/lodash-es-4.17.21.tgz", - "integrity": "sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw==", - "license": "MIT" - }, - "node_modules/lodash.merge": { - "version": "4.6.2", - "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", - "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/longest-streak": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/longest-streak/-/longest-streak-3.1.0.tgz", - "integrity": "sha512-9Ri+o0JYgehTaVBBDoMqIl8GXtbWg711O3srftcHhZ0dqnETqLaoIK0x17fUw9rFSlK/0NlsKe0Ahhyl5pXE2g==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/loose-envify": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", - "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", - "license": "MIT", - "dependencies": { - "js-tokens": "^3.0.0 || ^4.0.0" - }, - "bin": { - "loose-envify": "cli.js" - } - }, - "node_modules/loupe": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/loupe/-/loupe-3.2.1.tgz", - "integrity": "sha512-CdzqowRJCeLU72bHvWqwRBBlLcMEtIvGrlvef74kMnV2AolS9Y8xUv1I0U/MNAWMhBlKIoyuEgoJ0t/bbwHbLQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/lowlight": { - "version": "1.20.0", - "resolved": "https://registry.npmjs.org/lowlight/-/lowlight-1.20.0.tgz", - "integrity": "sha512-8Ktj+prEb1RoCPkEOrPMYUN/nCggB7qAWe3a7OpMjWQkh3l2RD5wKRQ+o8Q8YuI9RG/xs95waaI/E6ym/7NsTw==", - "license": "MIT", - "dependencies": { - "fault": "^1.0.0", - "highlight.js": "~10.7.0" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/lru-cache": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", - "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", - "dev": true, - "license": "ISC", - "dependencies": { - "yallist": "^3.0.2" - } - }, - "node_modules/lucide-react": { - "version": "0.553.0", - "resolved": "https://registry.npmjs.org/lucide-react/-/lucide-react-0.553.0.tgz", - "integrity": "sha512-BRgX5zrWmNy/lkVAe0dXBgd7XQdZ3HTf+Hwe3c9WK6dqgnj9h+hxV+MDncM88xDWlCq27+TKvHGE70ViODNILw==", - "license": "ISC", - "peerDependencies": { - "react": "^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0" - } - }, - "node_modules/magic-string": { - "version": "0.30.21", - "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.21.tgz", - "integrity": "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@jridgewell/sourcemap-codec": "^1.5.5" - } - }, - "node_modules/magicast": { - "version": "0.3.5", - "resolved": "https://registry.npmjs.org/magicast/-/magicast-0.3.5.tgz", - "integrity": "sha512-L0WhttDl+2BOsybvEOLK7fW3UA0OQ0IQ2d6Zl2x/a6vVRs3bAY0ECOSHHeL5jD+SbOpOCUEi0y1DgHEn9Qn1AQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/parser": "^7.25.4", - "@babel/types": "^7.25.4", - "source-map-js": "^1.2.0" - } - }, - "node_modules/make-cancellable-promise": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/make-cancellable-promise/-/make-cancellable-promise-2.0.0.tgz", - "integrity": "sha512-3SEQqTpV9oqVsIWqAcmDuaNeo7yBO3tqPtqGRcKkEo0lrzD3wqbKG9mkxO65KoOgXqj+zH2phJ2LiAsdzlogSw==", - "license": "MIT", - "funding": { - "url": "https://github.com/wojtekmaj/make-cancellable-promise?sponsor=1" - } - }, - "node_modules/make-dir": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-4.0.0.tgz", - "integrity": "sha512-hXdUTZYIVOt1Ex//jAQi+wTZZpUpwBj/0QsOzqegb3rGMMeJiSEu5xLHnYfBrRV4RH2+OCSOO95Is/7x1WJ4bw==", - "dev": true, - "license": "MIT", - "dependencies": { - "semver": "^7.5.3" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/make-dir/node_modules/semver": { - "version": "7.7.3", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", - "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", - "dev": true, - "license": "ISC", - "bin": { - "semver": "bin/semver.js" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/make-event-props": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/make-event-props/-/make-event-props-2.0.0.tgz", - "integrity": "sha512-G/hncXrl4Qt7mauJEXSg3AcdYzmpkIITTNl5I+rH9sog5Yw0kK6vseJjCaPfOXqOqQuPUP89Rkhfz5kPS8ijtw==", - "license": "MIT", - "funding": { - "url": "https://github.com/wojtekmaj/make-event-props?sponsor=1" - } - }, - "node_modules/markdown-table": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/markdown-table/-/markdown-table-3.0.4.tgz", - "integrity": "sha512-wiYz4+JrLyb/DqW2hkFJxP7Vd7JuTDm77fvbM8VfEQdmSMqcImWeeRbHwZjBjIFki/VaMK2BhFi7oUUZeM5bqw==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/marked": { - "version": "16.4.2", - "resolved": "https://registry.npmjs.org/marked/-/marked-16.4.2.tgz", - "integrity": "sha512-TI3V8YYWvkVf3KJe1dRkpnjs68JUPyEa5vjKrp1XEEJUAOaQc+Qj+L1qWbPd0SJuAdQkFU0h73sXXqwDYxsiDA==", - "license": "MIT", - "bin": { - "marked": "bin/marked.js" - }, - "engines": { - "node": ">= 20" - } - }, - "node_modules/mdast-util-find-and-replace": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/mdast-util-find-and-replace/-/mdast-util-find-and-replace-3.0.2.tgz", - "integrity": "sha512-Tmd1Vg/m3Xz43afeNxDIhWRtFZgM2VLyaf4vSTYwudTyeuTneoL3qtWMA5jeLyz/O1vDJmmV4QuScFCA2tBPwg==", - "license": "MIT", - "dependencies": { - "@types/mdast": "^4.0.0", - "escape-string-regexp": "^5.0.0", - "unist-util-is": "^6.0.0", - "unist-util-visit-parents": "^6.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/mdast-util-find-and-replace/node_modules/escape-string-regexp": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-5.0.0.tgz", - "integrity": "sha512-/veY75JbMK4j1yjvuUxuVsiS/hr/4iHs9FTT6cgTexxdE0Ly/glccBAkloH/DofkjRbZU3bnoj38mOmhkZ0lHw==", - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/mdast-util-from-markdown": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/mdast-util-from-markdown/-/mdast-util-from-markdown-2.0.2.tgz", - "integrity": "sha512-uZhTV/8NBuw0WHkPTrCqDOl0zVe1BIng5ZtHoDk49ME1qqcjYmmLmOf0gELgcRMxN4w2iuIeVso5/6QymSrgmA==", - "license": "MIT", - "dependencies": { - "@types/mdast": "^4.0.0", - "@types/unist": "^3.0.0", - "decode-named-character-reference": "^1.0.0", - "devlop": "^1.0.0", - "mdast-util-to-string": "^4.0.0", - "micromark": "^4.0.0", - "micromark-util-decode-numeric-character-reference": "^2.0.0", - "micromark-util-decode-string": "^2.0.0", - "micromark-util-normalize-identifier": "^2.0.0", - "micromark-util-symbol": "^2.0.0", - "micromark-util-types": "^2.0.0", - "unist-util-stringify-position": "^4.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/mdast-util-gfm": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/mdast-util-gfm/-/mdast-util-gfm-3.1.0.tgz", - "integrity": "sha512-0ulfdQOM3ysHhCJ1p06l0b0VKlhU0wuQs3thxZQagjcjPrlFRqY215uZGHHJan9GEAXd9MbfPjFJz+qMkVR6zQ==", - "license": "MIT", - "dependencies": { - "mdast-util-from-markdown": "^2.0.0", - "mdast-util-gfm-autolink-literal": "^2.0.0", - "mdast-util-gfm-footnote": "^2.0.0", - "mdast-util-gfm-strikethrough": "^2.0.0", - "mdast-util-gfm-table": "^2.0.0", - "mdast-util-gfm-task-list-item": "^2.0.0", - "mdast-util-to-markdown": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/mdast-util-gfm-autolink-literal": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/mdast-util-gfm-autolink-literal/-/mdast-util-gfm-autolink-literal-2.0.1.tgz", - "integrity": "sha512-5HVP2MKaP6L+G6YaxPNjuL0BPrq9orG3TsrZ9YXbA3vDw/ACI4MEsnoDpn6ZNm7GnZgtAcONJyPhOP8tNJQavQ==", - "license": "MIT", - "dependencies": { - "@types/mdast": "^4.0.0", - "ccount": "^2.0.0", - "devlop": "^1.0.0", - "mdast-util-find-and-replace": "^3.0.0", - "micromark-util-character": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/mdast-util-gfm-footnote": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/mdast-util-gfm-footnote/-/mdast-util-gfm-footnote-2.1.0.tgz", - "integrity": "sha512-sqpDWlsHn7Ac9GNZQMeUzPQSMzR6Wv0WKRNvQRg0KqHh02fpTz69Qc1QSseNX29bhz1ROIyNyxExfawVKTm1GQ==", - "license": "MIT", - "dependencies": { - "@types/mdast": "^4.0.0", - "devlop": "^1.1.0", - "mdast-util-from-markdown": "^2.0.0", - "mdast-util-to-markdown": "^2.0.0", - "micromark-util-normalize-identifier": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/mdast-util-gfm-strikethrough": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/mdast-util-gfm-strikethrough/-/mdast-util-gfm-strikethrough-2.0.0.tgz", - "integrity": "sha512-mKKb915TF+OC5ptj5bJ7WFRPdYtuHv0yTRxK2tJvi+BDqbkiG7h7u/9SI89nRAYcmap2xHQL9D+QG/6wSrTtXg==", - "license": "MIT", - "dependencies": { - "@types/mdast": "^4.0.0", - "mdast-util-from-markdown": "^2.0.0", - "mdast-util-to-markdown": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/mdast-util-gfm-table": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/mdast-util-gfm-table/-/mdast-util-gfm-table-2.0.0.tgz", - "integrity": "sha512-78UEvebzz/rJIxLvE7ZtDd/vIQ0RHv+3Mh5DR96p7cS7HsBhYIICDBCu8csTNWNO6tBWfqXPWekRuj2FNOGOZg==", - "license": "MIT", - "dependencies": { - "@types/mdast": "^4.0.0", - "devlop": "^1.0.0", - "markdown-table": "^3.0.0", - "mdast-util-from-markdown": "^2.0.0", - "mdast-util-to-markdown": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/mdast-util-gfm-task-list-item": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/mdast-util-gfm-task-list-item/-/mdast-util-gfm-task-list-item-2.0.0.tgz", - "integrity": "sha512-IrtvNvjxC1o06taBAVJznEnkiHxLFTzgonUdy8hzFVeDun0uTjxxrRGVaNFqkU1wJR3RBPEfsxmU6jDWPofrTQ==", - "license": "MIT", - "dependencies": { - "@types/mdast": "^4.0.0", - "devlop": "^1.0.0", - "mdast-util-from-markdown": "^2.0.0", - "mdast-util-to-markdown": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/mdast-util-mdx-expression": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/mdast-util-mdx-expression/-/mdast-util-mdx-expression-2.0.1.tgz", - "integrity": "sha512-J6f+9hUp+ldTZqKRSg7Vw5V6MqjATc+3E4gf3CFNcuZNWD8XdyI6zQ8GqH7f8169MM6P7hMBRDVGnn7oHB9kXQ==", - "license": "MIT", - "dependencies": { - "@types/estree-jsx": "^1.0.0", - "@types/hast": "^3.0.0", - "@types/mdast": "^4.0.0", - "devlop": "^1.0.0", - "mdast-util-from-markdown": "^2.0.0", - "mdast-util-to-markdown": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/mdast-util-mdx-jsx": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/mdast-util-mdx-jsx/-/mdast-util-mdx-jsx-3.2.0.tgz", - "integrity": "sha512-lj/z8v0r6ZtsN/cGNNtemmmfoLAFZnjMbNyLzBafjzikOM+glrjNHPlf6lQDOTccj9n5b0PPihEBbhneMyGs1Q==", - "license": "MIT", - "dependencies": { - "@types/estree-jsx": "^1.0.0", - "@types/hast": "^3.0.0", - "@types/mdast": "^4.0.0", - "@types/unist": "^3.0.0", - "ccount": "^2.0.0", - "devlop": "^1.1.0", - "mdast-util-from-markdown": "^2.0.0", - "mdast-util-to-markdown": "^2.0.0", - "parse-entities": "^4.0.0", - "stringify-entities": "^4.0.0", - "unist-util-stringify-position": "^4.0.0", - "vfile-message": "^4.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/mdast-util-mdxjs-esm": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/mdast-util-mdxjs-esm/-/mdast-util-mdxjs-esm-2.0.1.tgz", - "integrity": "sha512-EcmOpxsZ96CvlP03NghtH1EsLtr0n9Tm4lPUJUBccV9RwUOneqSycg19n5HGzCf+10LozMRSObtVr3ee1WoHtg==", - "license": "MIT", - "dependencies": { - "@types/estree-jsx": "^1.0.0", - "@types/hast": "^3.0.0", - "@types/mdast": "^4.0.0", - "devlop": "^1.0.0", - "mdast-util-from-markdown": "^2.0.0", - "mdast-util-to-markdown": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/mdast-util-phrasing": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/mdast-util-phrasing/-/mdast-util-phrasing-4.1.0.tgz", - "integrity": "sha512-TqICwyvJJpBwvGAMZjj4J2n0X8QWp21b9l0o7eXyVJ25YNWYbJDVIyD1bZXE6WtV6RmKJVYmQAKWa0zWOABz2w==", - "license": "MIT", - "dependencies": { - "@types/mdast": "^4.0.0", - "unist-util-is": "^6.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/mdast-util-to-hast": { - "version": "13.2.1", - "resolved": "https://registry.npmjs.org/mdast-util-to-hast/-/mdast-util-to-hast-13.2.1.tgz", - "integrity": "sha512-cctsq2wp5vTsLIcaymblUriiTcZd0CwWtCbLvrOzYCDZoWyMNV8sZ7krj09FSnsiJi3WVsHLM4k6Dq/yaPyCXA==", - "license": "MIT", - "dependencies": { - "@types/hast": "^3.0.0", - "@types/mdast": "^4.0.0", - "@ungap/structured-clone": "^1.0.0", - "devlop": "^1.0.0", - "micromark-util-sanitize-uri": "^2.0.0", - "trim-lines": "^3.0.0", - "unist-util-position": "^5.0.0", - "unist-util-visit": "^5.0.0", - "vfile": "^6.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/mdast-util-to-markdown": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/mdast-util-to-markdown/-/mdast-util-to-markdown-2.1.2.tgz", - "integrity": "sha512-xj68wMTvGXVOKonmog6LwyJKrYXZPvlwabaryTjLh9LuvovB/KAH+kvi8Gjj+7rJjsFi23nkUxRQv1KqSroMqA==", - "license": "MIT", - "dependencies": { - "@types/mdast": "^4.0.0", - "@types/unist": "^3.0.0", - "longest-streak": "^3.0.0", - "mdast-util-phrasing": "^4.0.0", - "mdast-util-to-string": "^4.0.0", - "micromark-util-classify-character": "^2.0.0", - "micromark-util-decode-string": "^2.0.0", - "unist-util-visit": "^5.0.0", - "zwitch": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/mdast-util-to-string": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/mdast-util-to-string/-/mdast-util-to-string-4.0.0.tgz", - "integrity": "sha512-0H44vDimn51F0YwvxSJSm0eCDOJTRlmN0R1yBh4HLj9wiV1Dn0QoXGbvFAWj2hSItVTlCmBF1hqKlIyUBVFLPg==", - "license": "MIT", - "dependencies": { - "@types/mdast": "^4.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/merge-refs": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/merge-refs/-/merge-refs-2.0.0.tgz", - "integrity": "sha512-3+B21mYK2IqUWnd2EivABLT7ueDhb0b8/dGK8LoFQPrU61YITeCMn14F7y7qZafWNZhUEKb24cJdiT5Wxs3prg==", - "license": "MIT", - "funding": { - "url": "https://github.com/wojtekmaj/merge-refs?sponsor=1" - }, - "peerDependencies": { - "@types/react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" - }, - "peerDependenciesMeta": { - "@types/react": { - "optional": true - } - } - }, - "node_modules/merge2": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", - "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 8" - } - }, - "node_modules/mermaid": { - "version": "11.12.1", - "resolved": "https://registry.npmjs.org/mermaid/-/mermaid-11.12.1.tgz", - "integrity": "sha512-UlIZrRariB11TY1RtTgUWp65tphtBv4CSq7vyS2ZZ2TgoMjs2nloq+wFqxiwcxlhHUvs7DPGgMjs2aeQxz5h9g==", - "license": "MIT", - "dependencies": { - "@braintree/sanitize-url": "^7.1.1", - "@iconify/utils": "^3.0.1", - "@mermaid-js/parser": "^0.6.3", - "@types/d3": "^7.4.3", - "cytoscape": "^3.29.3", - "cytoscape-cose-bilkent": "^4.1.0", - "cytoscape-fcose": "^2.2.0", - "d3": "^7.9.0", - "d3-sankey": "^0.12.3", - "dagre-d3-es": "7.0.13", - "dayjs": "^1.11.18", - "dompurify": "^3.2.5", - "katex": "^0.16.22", - "khroma": "^2.1.0", - "lodash-es": "^4.17.21", - "marked": "^16.2.1", - "roughjs": "^4.6.6", - "stylis": "^4.3.6", - "ts-dedent": "^2.2.0", - "uuid": "^11.1.0" - } - }, - "node_modules/micromark": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/micromark/-/micromark-4.0.2.tgz", - "integrity": "sha512-zpe98Q6kvavpCr1NPVSCMebCKfD7CA2NqZ+rykeNhONIJBpc1tFKt9hucLGwha3jNTNI8lHpctWJWoimVF4PfA==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "@types/debug": "^4.0.0", - "debug": "^4.0.0", - "decode-named-character-reference": "^1.0.0", - "devlop": "^1.0.0", - "micromark-core-commonmark": "^2.0.0", - "micromark-factory-space": "^2.0.0", - "micromark-util-character": "^2.0.0", - "micromark-util-chunked": "^2.0.0", - "micromark-util-combine-extensions": "^2.0.0", - "micromark-util-decode-numeric-character-reference": "^2.0.0", - "micromark-util-encode": "^2.0.0", - "micromark-util-normalize-identifier": "^2.0.0", - "micromark-util-resolve-all": "^2.0.0", - "micromark-util-sanitize-uri": "^2.0.0", - "micromark-util-subtokenize": "^2.0.0", - "micromark-util-symbol": "^2.0.0", - "micromark-util-types": "^2.0.0" - } - }, - "node_modules/micromark-core-commonmark": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/micromark-core-commonmark/-/micromark-core-commonmark-2.0.3.tgz", - "integrity": "sha512-RDBrHEMSxVFLg6xvnXmb1Ayr2WzLAWjeSATAoxwKYJV94TeNavgoIdA0a9ytzDSVzBy2YKFK+emCPOEibLeCrg==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "decode-named-character-reference": "^1.0.0", - "devlop": "^1.0.0", - "micromark-factory-destination": "^2.0.0", - "micromark-factory-label": "^2.0.0", - "micromark-factory-space": "^2.0.0", - "micromark-factory-title": "^2.0.0", - "micromark-factory-whitespace": "^2.0.0", - "micromark-util-character": "^2.0.0", - "micromark-util-chunked": "^2.0.0", - "micromark-util-classify-character": "^2.0.0", - "micromark-util-html-tag-name": "^2.0.0", - "micromark-util-normalize-identifier": "^2.0.0", - "micromark-util-resolve-all": "^2.0.0", - "micromark-util-subtokenize": "^2.0.0", - "micromark-util-symbol": "^2.0.0", - "micromark-util-types": "^2.0.0" - } - }, - "node_modules/micromark-extension-gfm": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/micromark-extension-gfm/-/micromark-extension-gfm-3.0.0.tgz", - "integrity": "sha512-vsKArQsicm7t0z2GugkCKtZehqUm31oeGBV/KVSorWSy8ZlNAv7ytjFhvaryUiCUJYqs+NoE6AFhpQvBTM6Q4w==", - "license": "MIT", - "dependencies": { - "micromark-extension-gfm-autolink-literal": "^2.0.0", - "micromark-extension-gfm-footnote": "^2.0.0", - "micromark-extension-gfm-strikethrough": "^2.0.0", - "micromark-extension-gfm-table": "^2.0.0", - "micromark-extension-gfm-tagfilter": "^2.0.0", - "micromark-extension-gfm-task-list-item": "^2.0.0", - "micromark-util-combine-extensions": "^2.0.0", - "micromark-util-types": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/micromark-extension-gfm-autolink-literal": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/micromark-extension-gfm-autolink-literal/-/micromark-extension-gfm-autolink-literal-2.1.0.tgz", - "integrity": "sha512-oOg7knzhicgQ3t4QCjCWgTmfNhvQbDDnJeVu9v81r7NltNCVmhPy1fJRX27pISafdjL+SVc4d3l48Gb6pbRypw==", - "license": "MIT", - "dependencies": { - "micromark-util-character": "^2.0.0", - "micromark-util-sanitize-uri": "^2.0.0", - "micromark-util-symbol": "^2.0.0", - "micromark-util-types": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/micromark-extension-gfm-footnote": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/micromark-extension-gfm-footnote/-/micromark-extension-gfm-footnote-2.1.0.tgz", - "integrity": "sha512-/yPhxI1ntnDNsiHtzLKYnE3vf9JZ6cAisqVDauhp4CEHxlb4uoOTxOCJ+9s51bIB8U1N1FJ1RXOKTIlD5B/gqw==", - "license": "MIT", - "dependencies": { - "devlop": "^1.0.0", - "micromark-core-commonmark": "^2.0.0", - "micromark-factory-space": "^2.0.0", - "micromark-util-character": "^2.0.0", - "micromark-util-normalize-identifier": "^2.0.0", - "micromark-util-sanitize-uri": "^2.0.0", - "micromark-util-symbol": "^2.0.0", - "micromark-util-types": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/micromark-extension-gfm-strikethrough": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/micromark-extension-gfm-strikethrough/-/micromark-extension-gfm-strikethrough-2.1.0.tgz", - "integrity": "sha512-ADVjpOOkjz1hhkZLlBiYA9cR2Anf8F4HqZUO6e5eDcPQd0Txw5fxLzzxnEkSkfnD0wziSGiv7sYhk/ktvbf1uw==", - "license": "MIT", - "dependencies": { - "devlop": "^1.0.0", - "micromark-util-chunked": "^2.0.0", - "micromark-util-classify-character": "^2.0.0", - "micromark-util-resolve-all": "^2.0.0", - "micromark-util-symbol": "^2.0.0", - "micromark-util-types": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/micromark-extension-gfm-table": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/micromark-extension-gfm-table/-/micromark-extension-gfm-table-2.1.1.tgz", - "integrity": "sha512-t2OU/dXXioARrC6yWfJ4hqB7rct14e8f7m0cbI5hUmDyyIlwv5vEtooptH8INkbLzOatzKuVbQmAYcbWoyz6Dg==", - "license": "MIT", - "dependencies": { - "devlop": "^1.0.0", - "micromark-factory-space": "^2.0.0", - "micromark-util-character": "^2.0.0", - "micromark-util-symbol": "^2.0.0", - "micromark-util-types": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/micromark-extension-gfm-tagfilter": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/micromark-extension-gfm-tagfilter/-/micromark-extension-gfm-tagfilter-2.0.0.tgz", - "integrity": "sha512-xHlTOmuCSotIA8TW1mDIM6X2O1SiX5P9IuDtqGonFhEK0qgRI4yeC6vMxEV2dgyr2TiD+2PQ10o+cOhdVAcwfg==", - "license": "MIT", - "dependencies": { - "micromark-util-types": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/micromark-extension-gfm-task-list-item": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/micromark-extension-gfm-task-list-item/-/micromark-extension-gfm-task-list-item-2.1.0.tgz", - "integrity": "sha512-qIBZhqxqI6fjLDYFTBIa4eivDMnP+OZqsNwmQ3xNLE4Cxwc+zfQEfbs6tzAo2Hjq+bh6q5F+Z8/cksrLFYWQQw==", - "license": "MIT", - "dependencies": { - "devlop": "^1.0.0", - "micromark-factory-space": "^2.0.0", - "micromark-util-character": "^2.0.0", - "micromark-util-symbol": "^2.0.0", - "micromark-util-types": "^2.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/micromark-factory-destination": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/micromark-factory-destination/-/micromark-factory-destination-2.0.1.tgz", - "integrity": "sha512-Xe6rDdJlkmbFRExpTOmRj9N3MaWmbAgdpSrBQvCFqhezUn4AHqJHbaEnfbVYYiexVSs//tqOdY/DxhjdCiJnIA==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "micromark-util-character": "^2.0.0", - "micromark-util-symbol": "^2.0.0", - "micromark-util-types": "^2.0.0" - } - }, - "node_modules/micromark-factory-label": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/micromark-factory-label/-/micromark-factory-label-2.0.1.tgz", - "integrity": "sha512-VFMekyQExqIW7xIChcXn4ok29YE3rnuyveW3wZQWWqF4Nv9Wk5rgJ99KzPvHjkmPXF93FXIbBp6YdW3t71/7Vg==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "devlop": "^1.0.0", - "micromark-util-character": "^2.0.0", - "micromark-util-symbol": "^2.0.0", - "micromark-util-types": "^2.0.0" - } - }, - "node_modules/micromark-factory-space": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/micromark-factory-space/-/micromark-factory-space-2.0.1.tgz", - "integrity": "sha512-zRkxjtBxxLd2Sc0d+fbnEunsTj46SWXgXciZmHq0kDYGnck/ZSGj9/wULTV95uoeYiK5hRXP2mJ98Uo4cq/LQg==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "micromark-util-character": "^2.0.0", - "micromark-util-types": "^2.0.0" - } - }, - "node_modules/micromark-factory-title": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/micromark-factory-title/-/micromark-factory-title-2.0.1.tgz", - "integrity": "sha512-5bZ+3CjhAd9eChYTHsjy6TGxpOFSKgKKJPJxr293jTbfry2KDoWkhBb6TcPVB4NmzaPhMs1Frm9AZH7OD4Cjzw==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "micromark-factory-space": "^2.0.0", - "micromark-util-character": "^2.0.0", - "micromark-util-symbol": "^2.0.0", - "micromark-util-types": "^2.0.0" - } - }, - "node_modules/micromark-factory-whitespace": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/micromark-factory-whitespace/-/micromark-factory-whitespace-2.0.1.tgz", - "integrity": "sha512-Ob0nuZ3PKt/n0hORHyvoD9uZhr+Za8sFoP+OnMcnWK5lngSzALgQYKMr9RJVOWLqQYuyn6ulqGWSXdwf6F80lQ==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "micromark-factory-space": "^2.0.0", - "micromark-util-character": "^2.0.0", - "micromark-util-symbol": "^2.0.0", - "micromark-util-types": "^2.0.0" - } - }, - "node_modules/micromark-util-character": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/micromark-util-character/-/micromark-util-character-2.1.1.tgz", - "integrity": "sha512-wv8tdUTJ3thSFFFJKtpYKOYiGP2+v96Hvk4Tu8KpCAsTMs6yi+nVmGh1syvSCsaxz45J6Jbw+9DD6g97+NV67Q==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "micromark-util-symbol": "^2.0.0", - "micromark-util-types": "^2.0.0" - } - }, - "node_modules/micromark-util-chunked": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/micromark-util-chunked/-/micromark-util-chunked-2.0.1.tgz", - "integrity": "sha512-QUNFEOPELfmvv+4xiNg2sRYeS/P84pTW0TCgP5zc9FpXetHY0ab7SxKyAQCNCc1eK0459uoLI1y5oO5Vc1dbhA==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "micromark-util-symbol": "^2.0.0" - } - }, - "node_modules/micromark-util-classify-character": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/micromark-util-classify-character/-/micromark-util-classify-character-2.0.1.tgz", - "integrity": "sha512-K0kHzM6afW/MbeWYWLjoHQv1sgg2Q9EccHEDzSkxiP/EaagNzCm7T/WMKZ3rjMbvIpvBiZgwR3dKMygtA4mG1Q==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "micromark-util-character": "^2.0.0", - "micromark-util-symbol": "^2.0.0", - "micromark-util-types": "^2.0.0" - } - }, - "node_modules/micromark-util-combine-extensions": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/micromark-util-combine-extensions/-/micromark-util-combine-extensions-2.0.1.tgz", - "integrity": "sha512-OnAnH8Ujmy59JcyZw8JSbK9cGpdVY44NKgSM7E9Eh7DiLS2E9RNQf0dONaGDzEG9yjEl5hcqeIsj4hfRkLH/Bg==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "micromark-util-chunked": "^2.0.0", - "micromark-util-types": "^2.0.0" - } - }, - "node_modules/micromark-util-decode-numeric-character-reference": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/micromark-util-decode-numeric-character-reference/-/micromark-util-decode-numeric-character-reference-2.0.2.tgz", - "integrity": "sha512-ccUbYk6CwVdkmCQMyr64dXz42EfHGkPQlBj5p7YVGzq8I7CtjXZJrubAYezf7Rp+bjPseiROqe7G6foFd+lEuw==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "micromark-util-symbol": "^2.0.0" - } - }, - "node_modules/micromark-util-decode-string": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/micromark-util-decode-string/-/micromark-util-decode-string-2.0.1.tgz", - "integrity": "sha512-nDV/77Fj6eH1ynwscYTOsbK7rR//Uj0bZXBwJZRfaLEJ1iGBR6kIfNmlNqaqJf649EP0F3NWNdeJi03elllNUQ==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "decode-named-character-reference": "^1.0.0", - "micromark-util-character": "^2.0.0", - "micromark-util-decode-numeric-character-reference": "^2.0.0", - "micromark-util-symbol": "^2.0.0" - } - }, - "node_modules/micromark-util-encode": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/micromark-util-encode/-/micromark-util-encode-2.0.1.tgz", - "integrity": "sha512-c3cVx2y4KqUnwopcO9b/SCdo2O67LwJJ/UyqGfbigahfegL9myoEFoDYZgkT7f36T0bLrM9hZTAaAyH+PCAXjw==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT" - }, - "node_modules/micromark-util-html-tag-name": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/micromark-util-html-tag-name/-/micromark-util-html-tag-name-2.0.1.tgz", - "integrity": "sha512-2cNEiYDhCWKI+Gs9T0Tiysk136SnR13hhO8yW6BGNyhOC4qYFnwF1nKfD3HFAIXA5c45RrIG1ub11GiXeYd1xA==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT" - }, - "node_modules/micromark-util-normalize-identifier": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/micromark-util-normalize-identifier/-/micromark-util-normalize-identifier-2.0.1.tgz", - "integrity": "sha512-sxPqmo70LyARJs0w2UclACPUUEqltCkJ6PhKdMIDuJ3gSf/Q+/GIe3WKl0Ijb/GyH9lOpUkRAO2wp0GVkLvS9Q==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "micromark-util-symbol": "^2.0.0" - } - }, - "node_modules/micromark-util-resolve-all": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/micromark-util-resolve-all/-/micromark-util-resolve-all-2.0.1.tgz", - "integrity": "sha512-VdQyxFWFT2/FGJgwQnJYbe1jjQoNTS4RjglmSjTUlpUMa95Htx9NHeYW4rGDJzbjvCsl9eLjMQwGeElsqmzcHg==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "micromark-util-types": "^2.0.0" - } - }, - "node_modules/micromark-util-sanitize-uri": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/micromark-util-sanitize-uri/-/micromark-util-sanitize-uri-2.0.1.tgz", - "integrity": "sha512-9N9IomZ/YuGGZZmQec1MbgxtlgougxTodVwDzzEouPKo3qFWvymFHWcnDi2vzV1ff6kas9ucW+o3yzJK9YB1AQ==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "micromark-util-character": "^2.0.0", - "micromark-util-encode": "^2.0.0", - "micromark-util-symbol": "^2.0.0" - } - }, - "node_modules/micromark-util-subtokenize": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/micromark-util-subtokenize/-/micromark-util-subtokenize-2.1.0.tgz", - "integrity": "sha512-XQLu552iSctvnEcgXw6+Sx75GflAPNED1qx7eBJ+wydBb2KCbRZe+NwvIEEMM83uml1+2WSXpBAcp9IUCgCYWA==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT", - "dependencies": { - "devlop": "^1.0.0", - "micromark-util-chunked": "^2.0.0", - "micromark-util-symbol": "^2.0.0", - "micromark-util-types": "^2.0.0" - } - }, - "node_modules/micromark-util-symbol": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/micromark-util-symbol/-/micromark-util-symbol-2.0.1.tgz", - "integrity": "sha512-vs5t8Apaud9N28kgCrRUdEed4UJ+wWNvicHLPxCa9ENlYuAY31M0ETy5y1vA33YoNPDFTghEbnh6efaE8h4x0Q==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT" - }, - "node_modules/micromark-util-types": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/micromark-util-types/-/micromark-util-types-2.0.2.tgz", - "integrity": "sha512-Yw0ECSpJoViF1qTU4DC6NwtC4aWGt1EkzaQB8KPPyCRR8z9TWeV0HbEFGTO+ZY1wB22zmxnJqhPyTpOVCpeHTA==", - "funding": [ - { - "type": "GitHub Sponsors", - "url": "https://github.com/sponsors/unifiedjs" - }, - { - "type": "OpenCollective", - "url": "https://opencollective.com/unified" - } - ], - "license": "MIT" - }, - "node_modules/micromatch": { - "version": "4.0.8", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", - "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", - "dev": true, - "license": "MIT", - "dependencies": { - "braces": "^3.0.3", - "picomatch": "^2.3.1" - }, - "engines": { - "node": ">=8.6" - } - }, - "node_modules/minimatch": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", - "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", - "dev": true, - "license": "ISC", - "dependencies": { - "brace-expansion": "^1.1.7" - }, - "engines": { - "node": "*" - } - }, - "node_modules/minipass": { - "version": "7.1.2", - "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz", - "integrity": "sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==", - "dev": true, - "license": "ISC", - "engines": { - "node": ">=16 || 14 >=14.17" - } - }, - "node_modules/mlly": { - "version": "1.8.0", - "resolved": "https://registry.npmjs.org/mlly/-/mlly-1.8.0.tgz", - "integrity": "sha512-l8D9ODSRWLe2KHJSifWGwBqpTZXIXTeo8mlKjY+E2HAakaTeNpqAyBZ8GSqLzHgw4XmHmC8whvpjJNMbFZN7/g==", - "license": "MIT", - "dependencies": { - "acorn": "^8.15.0", - "pathe": "^2.0.3", - "pkg-types": "^1.3.1", - "ufo": "^1.6.1" - } - }, - "node_modules/mlly/node_modules/confbox": { - "version": "0.1.8", - "resolved": "https://registry.npmjs.org/confbox/-/confbox-0.1.8.tgz", - "integrity": "sha512-RMtmw0iFkeR4YV+fUOSucriAQNb9g8zFR52MWCtl+cCZOFRNL6zeB395vPzFhEjjn4fMxXudmELnl/KF/WrK6w==", - "license": "MIT" - }, - "node_modules/mlly/node_modules/pkg-types": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/pkg-types/-/pkg-types-1.3.1.tgz", - "integrity": "sha512-/Jm5M4RvtBFVkKWRu2BLUTNP8/M2a+UwuAX+ae4770q1qVGtfjG+WTCupoZixokjmHiry8uI+dlY8KXYV5HVVQ==", - "license": "MIT", - "dependencies": { - "confbox": "^0.1.8", - "mlly": "^1.7.4", - "pathe": "^2.0.1" - } - }, - "node_modules/motion-dom": { - "version": "12.23.23", - "resolved": "https://registry.npmjs.org/motion-dom/-/motion-dom-12.23.23.tgz", - "integrity": "sha512-n5yolOs0TQQBRUFImrRfs/+6X4p3Q4n1dUEqt/H58Vx7OW6RF+foWEgmTVDhIWJIMXOuNNL0apKH2S16en9eiA==", - "license": "MIT", - "dependencies": { - "motion-utils": "^12.23.6" - } - }, - "node_modules/motion-utils": { - "version": "12.23.6", - "resolved": "https://registry.npmjs.org/motion-utils/-/motion-utils-12.23.6.tgz", - "integrity": "sha512-eAWoPgr4eFEOFfg2WjIsMoqJTW6Z8MTUCgn/GZ3VRpClWBdnbjryiA3ZSNLyxCTmCQx4RmYX6jX1iWHbenUPNQ==", - "license": "MIT" - }, - "node_modules/ms": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", - "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", - "license": "MIT" - }, - "node_modules/mz": { - "version": "2.7.0", - "resolved": "https://registry.npmjs.org/mz/-/mz-2.7.0.tgz", - "integrity": "sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "any-promise": "^1.0.0", - "object-assign": "^4.0.1", - "thenify-all": "^1.0.0" - } - }, - "node_modules/nanoid": { - "version": "3.3.11", - "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.11.tgz", - "integrity": "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/ai" - } - ], - "license": "MIT", - "bin": { - "nanoid": "bin/nanoid.cjs" - }, - "engines": { - "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" - } - }, - "node_modules/natural-compare": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", - "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", - "dev": true, - "license": "MIT" - }, - "node_modules/node-releases": { - "version": "2.0.27", - "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.27.tgz", - "integrity": "sha512-nmh3lCkYZ3grZvqcCH+fjmQ7X+H0OeZgP40OierEaAptX4XofMh5kwNbWh7lBduUzCcV/8kZ+NDLCwm2iorIlA==", - "dev": true, - "license": "MIT" - }, - "node_modules/normalize-path": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", - "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/normalize-range": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/normalize-range/-/normalize-range-0.1.2.tgz", - "integrity": "sha512-bdok/XvKII3nUpklnV6P2hxtMNrCboOjAcyBuQnWEhO665FwrSNRxU+AqpsyvO6LgGYPspN+lu5CLtw4jPRKNA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/object-assign": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", - "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/object-hash": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/object-hash/-/object-hash-3.0.0.tgz", - "integrity": "sha512-RSn9F68PjH9HqtltsSnqYC1XXoWe9Bju5+213R98cNGttag9q9yAOTzdbsqvIa7aNm5WffBZFpWYr2aWrklWAw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 6" - } - }, - "node_modules/optionator": { - "version": "0.9.4", - "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", - "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", - "dev": true, - "license": "MIT", - "dependencies": { - "deep-is": "^0.1.3", - "fast-levenshtein": "^2.0.6", - "levn": "^0.4.1", - "prelude-ls": "^1.2.1", - "type-check": "^0.4.0", - "word-wrap": "^1.2.5" - }, - "engines": { - "node": ">= 0.8.0" - } - }, - "node_modules/p-limit": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", - "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "yocto-queue": "^0.1.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/p-locate": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", - "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", - "dev": true, - "license": "MIT", - "dependencies": { - "p-limit": "^3.0.2" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/package-json-from-dist": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/package-json-from-dist/-/package-json-from-dist-1.0.1.tgz", - "integrity": "sha512-UEZIS3/by4OC8vL3P2dTXRETpebLI2NiI5vIrjaD/5UtrkFX/tNbwjTSRAGC/+7CAo2pIcBaRgWmcBBHcsaCIw==", - "dev": true, - "license": "BlueOak-1.0.0" - }, - "node_modules/package-manager-detector": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/package-manager-detector/-/package-manager-detector-1.5.0.tgz", - "integrity": "sha512-uBj69dVlYe/+wxj8JOpr97XfsxH/eumMt6HqjNTmJDf/6NO9s+0uxeOneIz3AsPt2m6y9PqzDzd3ATcU17MNfw==", - "license": "MIT" - }, - "node_modules/parent-module": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", - "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", - "dev": true, - "license": "MIT", - "dependencies": { - "callsites": "^3.0.0" - }, - "engines": { - "node": ">=6" - } - }, - "node_modules/parse-entities": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/parse-entities/-/parse-entities-4.0.2.tgz", - "integrity": "sha512-GG2AQYWoLgL877gQIKeRPGO1xF9+eG1ujIb5soS5gPvLQ1y2o8FL90w2QWNdf9I361Mpp7726c+lj3U0qK1uGw==", - "license": "MIT", - "dependencies": { - "@types/unist": "^2.0.0", - "character-entities-legacy": "^3.0.0", - "character-reference-invalid": "^2.0.0", - "decode-named-character-reference": "^1.0.0", - "is-alphanumerical": "^2.0.0", - "is-decimal": "^2.0.0", - "is-hexadecimal": "^2.0.0" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/parse-entities/node_modules/@types/unist": { - "version": "2.0.11", - "resolved": "https://registry.npmjs.org/@types/unist/-/unist-2.0.11.tgz", - "integrity": "sha512-CmBKiL6NNo/OqgmMn95Fk9Whlp2mtvIv+KNpQKN2F4SjvrEesubTRWGYSg+BnWZOnlCaSTU1sMpsBOzgbYhnsA==", - "license": "MIT" - }, - "node_modules/path-data-parser": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/path-data-parser/-/path-data-parser-0.1.0.tgz", - "integrity": "sha512-NOnmBpt5Y2RWbuv0LMzsayp3lVylAHLPUTut412ZA3l+C4uw4ZVkQbjShYCQ8TCpUMdPapr4YjUqLYD6v68j+w==", - "license": "MIT" - }, - "node_modules/path-exists": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", - "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/path-key": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", - "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/path-parse": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", - "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", - "dev": true, - "license": "MIT" - }, - "node_modules/path-scurry": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-1.11.1.tgz", - "integrity": "sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==", - "dev": true, - "license": "BlueOak-1.0.0", - "dependencies": { - "lru-cache": "^10.2.0", - "minipass": "^5.0.0 || ^6.0.2 || ^7.0.0" - }, - "engines": { - "node": ">=16 || 14 >=14.18" - }, - "funding": { - "url": "https://github.com/sponsors/isaacs" - } - }, - "node_modules/path-scurry/node_modules/lru-cache": { - "version": "10.4.3", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-10.4.3.tgz", - "integrity": "sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==", - "dev": true, - "license": "ISC" - }, - "node_modules/pathe": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/pathe/-/pathe-2.0.3.tgz", - "integrity": "sha512-WUjGcAqP1gQacoQe+OBJsFA7Ld4DyXuUIjZ5cc75cLHvJ7dtNsTugphxIADwspS+AraAUePCKrSVtPLFj/F88w==", - "license": "MIT" - }, - "node_modules/pathval": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/pathval/-/pathval-2.0.1.tgz", - "integrity": "sha512-//nshmD55c46FuFw26xV/xFAaB5HF9Xdap7HJBBnrKdAd6/GxDBaNA1870O79+9ueg61cZLSVc+OaFlfmObYVQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 14.16" - } - }, - "node_modules/pdfjs-dist": { - "version": "5.4.530", - "resolved": "https://registry.npmjs.org/pdfjs-dist/-/pdfjs-dist-5.4.530.tgz", - "integrity": "sha512-r1hWsSIGGmyYUAHR26zSXkxYWLXLMd6AwqcaFYG9YUZ0GBf5GvcjJSeo512tabM4GYFhxhl5pMCmPr7Q72Rq2Q==", - "license": "Apache-2.0", - "engines": { - "node": ">=20.16.0 || >=22.3.0" - }, - "optionalDependencies": { - "@napi-rs/canvas": "^0.1.84" - } - }, - "node_modules/picocolors": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", - "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==", - "dev": true, - "license": "ISC" - }, - "node_modules/picomatch": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", - "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8.6" - }, - "funding": { - "url": "https://github.com/sponsors/jonschlinkert" - } - }, - "node_modules/pify": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", - "integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/pirates": { - "version": "4.0.7", - "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.7.tgz", - "integrity": "sha512-TfySrs/5nm8fQJDcBDuUng3VOUKsd7S+zqvbOTiGXHfxX4wK31ard+hoNuvkicM/2YFzlpDgABOevKSsB4G/FA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 6" - } - }, - "node_modules/pkg-types": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/pkg-types/-/pkg-types-2.3.0.tgz", - "integrity": "sha512-SIqCzDRg0s9npO5XQ3tNZioRY1uK06lA41ynBC1YmFTmnY6FjUjVt6s4LoADmwoig1qqD0oK8h1p/8mlMx8Oig==", - "license": "MIT", - "dependencies": { - "confbox": "^0.2.2", - "exsolve": "^1.0.7", - "pathe": "^2.0.3" - } - }, - "node_modules/points-on-curve": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/points-on-curve/-/points-on-curve-0.2.0.tgz", - "integrity": "sha512-0mYKnYYe9ZcqMCWhUjItv/oHjvgEsfKvnUTg8sAtnHr3GVy7rGkXCb6d5cSyqrWqL4k81b9CPg3urd+T7aop3A==", - "license": "MIT" - }, - "node_modules/points-on-path": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/points-on-path/-/points-on-path-0.2.1.tgz", - "integrity": "sha512-25ClnWWuw7JbWZcgqY/gJ4FQWadKxGWk+3kR/7kD0tCaDtPPMj7oHu2ToLaVhfpnHrZzYby2w6tUA0eOIuUg8g==", - "license": "MIT", - "dependencies": { - "path-data-parser": "0.1.0", - "points-on-curve": "0.2.0" - } - }, - "node_modules/postcss": { - "version": "8.5.6", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.6.tgz", - "integrity": "sha512-3Ybi1tAuwAP9s0r1UQ2J4n5Y0G05bJkpUIO0/bI9MhwmD70S5aTWbXGBwxHrelT+XM1k6dM0pk+SwNkpTRN7Pg==", - "dev": true, - "funding": [ - { - "type": "opencollective", - "url": "https://opencollective.com/postcss/" - }, - { - "type": "tidelift", - "url": "https://tidelift.com/funding/github/npm/postcss" - }, - { - "type": "github", - "url": "https://github.com/sponsors/ai" - } - ], - "license": "MIT", - "dependencies": { - "nanoid": "^3.3.11", - "picocolors": "^1.1.1", - "source-map-js": "^1.2.1" - }, - "engines": { - "node": "^10 || ^12 || >=14" - } - }, - "node_modules/postcss-import": { - "version": "15.1.0", - "resolved": "https://registry.npmjs.org/postcss-import/-/postcss-import-15.1.0.tgz", - "integrity": "sha512-hpr+J05B2FVYUAXHeK1YyI267J/dDDhMU6B6civm8hSY1jYJnBXxzKDKDswzJmtLHryrjhnDjqqp/49t8FALew==", - "dev": true, - "license": "MIT", - "dependencies": { - "postcss-value-parser": "^4.0.0", - "read-cache": "^1.0.0", - "resolve": "^1.1.7" - }, - "engines": { - "node": ">=14.0.0" - }, - "peerDependencies": { - "postcss": "^8.0.0" - } - }, - "node_modules/postcss-js": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/postcss-js/-/postcss-js-4.1.0.tgz", - "integrity": "sha512-oIAOTqgIo7q2EOwbhb8UalYePMvYoIeRY2YKntdpFQXNosSu3vLrniGgmH9OKs/qAkfoj5oB3le/7mINW1LCfw==", - "dev": true, - "funding": [ - { - "type": "opencollective", - "url": "https://opencollective.com/postcss/" - }, - { - "type": "github", - "url": "https://github.com/sponsors/ai" - } - ], - "license": "MIT", - "dependencies": { - "camelcase-css": "^2.0.1" - }, - "engines": { - "node": "^12 || ^14 || >= 16" - }, - "peerDependencies": { - "postcss": "^8.4.21" - } - }, - "node_modules/postcss-load-config": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/postcss-load-config/-/postcss-load-config-6.0.1.tgz", - "integrity": "sha512-oPtTM4oerL+UXmx+93ytZVN82RrlY/wPUV8IeDxFrzIjXOLF1pN+EmKPLbubvKHT2HC20xXsCAH2Z+CKV6Oz/g==", - "dev": true, - "funding": [ - { - "type": "opencollective", - "url": "https://opencollective.com/postcss/" - }, - { - "type": "github", - "url": "https://github.com/sponsors/ai" - } - ], - "license": "MIT", - "dependencies": { - "lilconfig": "^3.1.1" - }, - "engines": { - "node": ">= 18" - }, - "peerDependencies": { - "jiti": ">=1.21.0", - "postcss": ">=8.0.9", - "tsx": "^4.8.1", - "yaml": "^2.4.2" - }, - "peerDependenciesMeta": { - "jiti": { - "optional": true - }, - "postcss": { - "optional": true - }, - "tsx": { - "optional": true - }, - "yaml": { - "optional": true - } - } - }, - "node_modules/postcss-nested": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/postcss-nested/-/postcss-nested-6.2.0.tgz", - "integrity": "sha512-HQbt28KulC5AJzG+cZtj9kvKB93CFCdLvog1WFLf1D+xmMvPGlBstkpTEZfK5+AN9hfJocyBFCNiqyS48bpgzQ==", - "dev": true, - "funding": [ - { - "type": "opencollective", - "url": "https://opencollective.com/postcss/" - }, - { - "type": "github", - "url": "https://github.com/sponsors/ai" - } - ], - "license": "MIT", - "dependencies": { - "postcss-selector-parser": "^6.1.1" - }, - "engines": { - "node": ">=12.0" - }, - "peerDependencies": { - "postcss": "^8.2.14" - } - }, - "node_modules/postcss-selector-parser": { - "version": "6.1.2", - "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.1.2.tgz", - "integrity": "sha512-Q8qQfPiZ+THO/3ZrOrO0cJJKfpYCagtMUkXbnEfmgUjwXg6z/WBeOyS9APBBPCTSiDV+s4SwQGu8yFsiMRIudg==", - "dev": true, - "license": "MIT", - "dependencies": { - "cssesc": "^3.0.0", - "util-deprecate": "^1.0.2" - }, - "engines": { - "node": ">=4" - } - }, - "node_modules/postcss-value-parser": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz", - "integrity": "sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/prelude-ls": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", - "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.8.0" - } - }, - "node_modules/prettier": { - "version": "3.6.2", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.6.2.tgz", - "integrity": "sha512-I7AIg5boAr5R0FFtJ6rCfD+LFsWHp81dolrFD8S79U9tb8Az2nGrJncnMSnys+bpQJfRUzqs9hnA81OAA3hCuQ==", - "dev": true, - "license": "MIT", - "bin": { - "prettier": "bin/prettier.cjs" - }, - "engines": { - "node": ">=14" - }, - "funding": { - "url": "https://github.com/prettier/prettier?sponsor=1" - } - }, - "node_modules/prettier-linter-helpers": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/prettier-linter-helpers/-/prettier-linter-helpers-1.0.0.tgz", - "integrity": "sha512-GbK2cP9nraSSUF9N2XwUwqfzlAFlMNYYl+ShE/V+H8a9uNl/oUqB1w2EL54Jh0OlyRSd8RfWYJ3coVS4TROP2w==", - "dev": true, - "license": "MIT", - "dependencies": { - "fast-diff": "^1.1.2" - }, - "engines": { - "node": ">=6.0.0" - } - }, - "node_modules/prismjs": { - "version": "1.30.0", - "resolved": "https://registry.npmjs.org/prismjs/-/prismjs-1.30.0.tgz", - "integrity": "sha512-DEvV2ZF2r2/63V+tK8hQvrR2ZGn10srHbXviTlcv7Kpzw8jWiNTqbVgjO3IY8RxrrOUF8VPMQQFysYYYv0YZxw==", - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/property-information": { - "version": "7.1.0", - "resolved": "https://registry.npmjs.org/property-information/-/property-information-7.1.0.tgz", - "integrity": "sha512-TwEZ+X+yCJmYfL7TPUOcvBZ4QfoT5YenQiJuX//0th53DE6w0xxLEtfK3iyryQFddXuvkIk51EEgrJQ0WJkOmQ==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/punycode": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", - "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/quansync": { - "version": "0.2.11", - "resolved": "https://registry.npmjs.org/quansync/-/quansync-0.2.11.tgz", - "integrity": "sha512-AifT7QEbW9Nri4tAwR5M/uzpBuqfZf+zwaEM/QkzEjj7NBuFD2rBuy0K3dE+8wltbezDV7JMA0WfnCPYRSYbXA==", - "funding": [ - { - "type": "individual", - "url": "https://github.com/sponsors/antfu" - }, - { - "type": "individual", - "url": "https://github.com/sponsors/sxzz" - } - ], - "license": "MIT" - }, - "node_modules/queue-microtask": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", - "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT" - }, - "node_modules/react": { - "version": "19.2.0", - "resolved": "https://registry.npmjs.org/react/-/react-19.2.0.tgz", - "integrity": "sha512-tmbWg6W31tQLeB5cdIBOicJDJRR2KzXsV7uSK9iNfLWQ5bIZfxuPEHp7M8wiHyHnn0DD1i7w3Zmin0FtkrwoCQ==", - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/react-chartjs-2": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/react-chartjs-2/-/react-chartjs-2-5.3.1.tgz", - "integrity": "sha512-h5IPXKg9EXpjoBzUfyWJvllMjG2mQ4EiuHQFhms/AjUm0XSZHhyRy2xVmLXHKrtcdrPO4mnGqRtYoD0vp95A0A==", - "license": "MIT", - "peerDependencies": { - "chart.js": "^4.1.1", - "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" - } - }, - "node_modules/react-dom": { - "version": "19.2.0", - "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.2.0.tgz", - "integrity": "sha512-UlbRu4cAiGaIewkPyiRGJk0imDN2T3JjieT6spoL2UeSf5od4n5LB/mQ4ejmxhCFT1tYe8IvaFulzynWovsEFQ==", - "license": "MIT", - "dependencies": { - "scheduler": "^0.27.0" - }, - "peerDependencies": { - "react": "^19.2.0" - } - }, - "node_modules/react-markdown": { - "version": "10.1.0", - "resolved": "https://registry.npmjs.org/react-markdown/-/react-markdown-10.1.0.tgz", - "integrity": "sha512-qKxVopLT/TyA6BX3Ue5NwabOsAzm0Q7kAPwq6L+wWDwisYs7R8vZ0nRXqq6rkueboxpkjvLGU9fWifiX/ZZFxQ==", - "license": "MIT", - "dependencies": { - "@types/hast": "^3.0.0", - "@types/mdast": "^4.0.0", - "devlop": "^1.0.0", - "hast-util-to-jsx-runtime": "^2.0.0", - "html-url-attributes": "^3.0.0", - "mdast-util-to-hast": "^13.0.0", - "remark-parse": "^11.0.0", - "remark-rehype": "^11.0.0", - "unified": "^11.0.0", - "unist-util-visit": "^5.0.0", - "vfile": "^6.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - }, - "peerDependencies": { - "@types/react": ">=18", - "react": ">=18" - } - }, - "node_modules/react-pdf": { - "version": "10.3.0", - "resolved": "https://registry.npmjs.org/react-pdf/-/react-pdf-10.3.0.tgz", - "integrity": "sha512-2LQzC9IgNVAX8gM+6F+1t/70a9/5RWThYxc+CWAmT2LW/BRmnj+35x1os5j/nR2oldyf8L+hCAMBmVKU8wrYFA==", - "license": "MIT", - "dependencies": { - "clsx": "^2.0.0", - "dequal": "^2.0.3", - "make-cancellable-promise": "^2.0.0", - "make-event-props": "^2.0.0", - "merge-refs": "^2.0.0", - "pdfjs-dist": "5.4.296", - "tiny-invariant": "^1.0.0", - "warning": "^4.0.0" - }, - "funding": { - "url": "https://github.com/wojtekmaj/react-pdf?sponsor=1" - }, - "peerDependencies": { - "@types/react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", - "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", - "react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" - }, - "peerDependenciesMeta": { - "@types/react": { - "optional": true - } - } - }, - "node_modules/react-pdf/node_modules/pdfjs-dist": { - "version": "5.4.296", - "resolved": "https://registry.npmjs.org/pdfjs-dist/-/pdfjs-dist-5.4.296.tgz", - "integrity": "sha512-DlOzet0HO7OEnmUmB6wWGJrrdvbyJKftI1bhMitK7O2N8W2gc757yyYBbINy9IDafXAV9wmKr9t7xsTaNKRG5Q==", - "license": "Apache-2.0", - "engines": { - "node": ">=20.16.0 || >=22.3.0" - }, - "optionalDependencies": { - "@napi-rs/canvas": "^0.1.80" - } - }, - "node_modules/react-refresh": { - "version": "0.18.0", - "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.18.0.tgz", - "integrity": "sha512-QgT5//D3jfjJb6Gsjxv0Slpj23ip+HtOpnNgnb2S5zU3CB26G/IDPGoy4RJB42wzFE46DRsstbW6tKHoKbhAxw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/react-resizable-panels": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/react-resizable-panels/-/react-resizable-panels-3.0.6.tgz", - "integrity": "sha512-b3qKHQ3MLqOgSS+FRYKapNkJZf5EQzuf6+RLiq1/IlTHw99YrZ2NJZLk4hQIzTnnIkRg2LUqyVinu6YWWpUYew==", - "license": "MIT", - "peerDependencies": { - "react": "^16.14.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc", - "react-dom": "^16.14.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc" - } - }, - "node_modules/react-router": { - "version": "7.12.0", - "resolved": "https://registry.npmjs.org/react-router/-/react-router-7.12.0.tgz", - "integrity": "sha512-kTPDYPFzDVGIIGNLS5VJykK0HfHLY5MF3b+xj0/tTyNYL1gF1qs7u67Z9jEhQk2sQ98SUaHxlG31g1JtF7IfVw==", - "license": "MIT", - "dependencies": { - "cookie": "^1.0.1", - "set-cookie-parser": "^2.6.0" - }, - "engines": { - "node": ">=20.0.0" - }, - "peerDependencies": { - "react": ">=18", - "react-dom": ">=18" - }, - "peerDependenciesMeta": { - "react-dom": { - "optional": true - } - } - }, - "node_modules/react-router-dom": { - "version": "7.12.0", - "resolved": "https://registry.npmjs.org/react-router-dom/-/react-router-dom-7.12.0.tgz", - "integrity": "sha512-pfO9fiBcpEfX4Tx+iTYKDtPbrSLLCbwJ5EqP+SPYQu1VYCXdy79GSj0wttR0U4cikVdlImZuEZ/9ZNCgoaxwBA==", - "license": "MIT", - "dependencies": { - "react-router": "7.12.0" - }, - "engines": { - "node": ">=20.0.0" - }, - "peerDependencies": { - "react": ">=18", - "react-dom": ">=18" - } - }, - "node_modules/react-syntax-highlighter": { - "version": "16.1.0", - "resolved": "https://registry.npmjs.org/react-syntax-highlighter/-/react-syntax-highlighter-16.1.0.tgz", - "integrity": "sha512-E40/hBiP5rCNwkeBN1vRP+xow1X0pndinO+z3h7HLsHyjztbyjfzNWNKuAsJj+7DLam9iT4AaaOZnueCU+Nplg==", - "license": "MIT", - "dependencies": { - "@babel/runtime": "^7.28.4", - "highlight.js": "^10.4.1", - "highlightjs-vue": "^1.0.0", - "lowlight": "^1.17.0", - "prismjs": "^1.30.0", - "refractor": "^5.0.0" - }, - "engines": { - "node": ">= 16.20.2" - }, - "peerDependencies": { - "react": ">= 0.14.0" - } - }, - "node_modules/react-virtuoso": { - "version": "4.14.1", - "resolved": "https://registry.npmjs.org/react-virtuoso/-/react-virtuoso-4.14.1.tgz", - "integrity": "sha512-NRUF1ak8lY+Tvc6WN9cce59gU+lilzVtOozP+pm9J7iHshLGGjsiAB4rB2qlBPHjFbcXOQpT+7womNHGDUql8w==", - "license": "MIT", - "peerDependencies": { - "react": ">=16 || >=17 || >= 18 || >= 19", - "react-dom": ">=16 || >=17 || >= 18 || >=19" - } - }, - "node_modules/react-zoom-pan-pinch": { - "version": "3.7.0", - "resolved": "https://registry.npmjs.org/react-zoom-pan-pinch/-/react-zoom-pan-pinch-3.7.0.tgz", - "integrity": "sha512-UmReVZ0TxlKzxSbYiAj+LeGRW8s8LraAFTXRAxzMYnNRgGPsxCudwZKVkjvGmjtx7SW/hZamt69NUmGf4xrkXA==", - "license": "MIT", - "engines": { - "node": ">=8", - "npm": ">=5" - }, - "peerDependencies": { - "react": "*", - "react-dom": "*" - } - }, - "node_modules/read-cache": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz", - "integrity": "sha512-Owdv/Ft7IjOgm/i0xvNDZ1LrRANRfew4b2prF3OWMQLxLfu3bS8FVhCsrSCMK4lR56Y9ya+AThoTpDCTxCmpRA==", - "dev": true, - "license": "MIT", - "dependencies": { - "pify": "^2.3.0" - } - }, - "node_modules/readdirp": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", - "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", - "dev": true, - "license": "MIT", - "dependencies": { - "picomatch": "^2.2.1" - }, - "engines": { - "node": ">=8.10.0" - } - }, - "node_modules/refractor": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/refractor/-/refractor-5.0.0.tgz", - "integrity": "sha512-QXOrHQF5jOpjjLfiNk5GFnWhRXvxjUVnlFxkeDmewR5sXkr3iM46Zo+CnRR8B+MDVqkULW4EcLVcRBNOPXHosw==", - "license": "MIT", - "dependencies": { - "@types/hast": "^3.0.0", - "@types/prismjs": "^1.0.0", - "hastscript": "^9.0.0", - "parse-entities": "^4.0.0" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/remark-gfm": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/remark-gfm/-/remark-gfm-4.0.1.tgz", - "integrity": "sha512-1quofZ2RQ9EWdeN34S79+KExV1764+wCUGop5CPL1WGdD0ocPpu91lzPGbwWMECpEpd42kJGQwzRfyov9j4yNg==", - "license": "MIT", - "dependencies": { - "@types/mdast": "^4.0.0", - "mdast-util-gfm": "^3.0.0", - "micromark-extension-gfm": "^3.0.0", - "remark-parse": "^11.0.0", - "remark-stringify": "^11.0.0", - "unified": "^11.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/remark-parse": { - "version": "11.0.0", - "resolved": "https://registry.npmjs.org/remark-parse/-/remark-parse-11.0.0.tgz", - "integrity": "sha512-FCxlKLNGknS5ba/1lmpYijMUzX2esxW5xQqjWxw2eHFfS2MSdaHVINFmhjo+qN1WhZhNimq0dZATN9pH0IDrpA==", - "license": "MIT", - "dependencies": { - "@types/mdast": "^4.0.0", - "mdast-util-from-markdown": "^2.0.0", - "micromark-util-types": "^2.0.0", - "unified": "^11.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/remark-rehype": { - "version": "11.1.2", - "resolved": "https://registry.npmjs.org/remark-rehype/-/remark-rehype-11.1.2.tgz", - "integrity": "sha512-Dh7l57ianaEoIpzbp0PC9UKAdCSVklD8E5Rpw7ETfbTl3FqcOOgq5q2LVDhgGCkaBv7p24JXikPdvhhmHvKMsw==", - "license": "MIT", - "dependencies": { - "@types/hast": "^3.0.0", - "@types/mdast": "^4.0.0", - "mdast-util-to-hast": "^13.0.0", - "unified": "^11.0.0", - "vfile": "^6.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/remark-stringify": { - "version": "11.0.0", - "resolved": "https://registry.npmjs.org/remark-stringify/-/remark-stringify-11.0.0.tgz", - "integrity": "sha512-1OSmLd3awB/t8qdoEOMazZkNsfVTeY4fTsgzcQFdXNq8ToTN4ZGwrMnlda4K6smTFKD+GRV6O48i6Z4iKgPPpw==", - "license": "MIT", - "dependencies": { - "@types/mdast": "^4.0.0", - "mdast-util-to-markdown": "^2.0.0", - "unified": "^11.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/resolve": { - "version": "1.22.11", - "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.11.tgz", - "integrity": "sha512-RfqAvLnMl313r7c9oclB1HhUEAezcpLjz95wFH4LVuhk9JF/r22qmVP9AMmOU4vMX7Q8pN8jwNg/CSpdFnMjTQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-core-module": "^2.16.1", - "path-parse": "^1.0.7", - "supports-preserve-symlinks-flag": "^1.0.0" - }, - "bin": { - "resolve": "bin/resolve" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/resolve-from": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", - "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=4" - } - }, - "node_modules/reusify": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz", - "integrity": "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==", - "dev": true, - "license": "MIT", - "engines": { - "iojs": ">=1.0.0", - "node": ">=0.10.0" - } - }, - "node_modules/robust-predicates": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/robust-predicates/-/robust-predicates-3.0.2.tgz", - "integrity": "sha512-IXgzBWvWQwE6PrDI05OvmXUIruQTcoMDzRsOd5CDvHCVLcLHMTSYvOK5Cm46kWqlV3yAbuSpBZdJ5oP5OUoStg==", - "license": "Unlicense" - }, - "node_modules/rollup": { - "version": "4.53.2", - "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.53.2.tgz", - "integrity": "sha512-MHngMYwGJVi6Fmnk6ISmnk7JAHRNF0UkuucA0CUW3N3a4KnONPEZz+vUanQP/ZC/iY1Qkf3bwPWzyY84wEks1g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/estree": "1.0.8" - }, - "bin": { - "rollup": "dist/bin/rollup" - }, - "engines": { - "node": ">=18.0.0", - "npm": ">=8.0.0" - }, - "optionalDependencies": { - "@rollup/rollup-android-arm-eabi": "4.53.2", - "@rollup/rollup-android-arm64": "4.53.2", - "@rollup/rollup-darwin-arm64": "4.53.2", - "@rollup/rollup-darwin-x64": "4.53.2", - "@rollup/rollup-freebsd-arm64": "4.53.2", - "@rollup/rollup-freebsd-x64": "4.53.2", - "@rollup/rollup-linux-arm-gnueabihf": "4.53.2", - "@rollup/rollup-linux-arm-musleabihf": "4.53.2", - "@rollup/rollup-linux-arm64-gnu": "4.53.2", - "@rollup/rollup-linux-arm64-musl": "4.53.2", - "@rollup/rollup-linux-loong64-gnu": "4.53.2", - "@rollup/rollup-linux-ppc64-gnu": "4.53.2", - "@rollup/rollup-linux-riscv64-gnu": "4.53.2", - "@rollup/rollup-linux-riscv64-musl": "4.53.2", - "@rollup/rollup-linux-s390x-gnu": "4.53.2", - "@rollup/rollup-linux-x64-gnu": "4.53.2", - "@rollup/rollup-linux-x64-musl": "4.53.2", - "@rollup/rollup-openharmony-arm64": "4.53.2", - "@rollup/rollup-win32-arm64-msvc": "4.53.2", - "@rollup/rollup-win32-ia32-msvc": "4.53.2", - "@rollup/rollup-win32-x64-gnu": "4.53.2", - "@rollup/rollup-win32-x64-msvc": "4.53.2", - "fsevents": "~2.3.2" - } - }, - "node_modules/roughjs": { - "version": "4.6.6", - "resolved": "https://registry.npmjs.org/roughjs/-/roughjs-4.6.6.tgz", - "integrity": "sha512-ZUz/69+SYpFN/g/lUlo2FXcIjRkSu3nDarreVdGGndHEBJ6cXPdKguS8JGxwj5HA5xIbVKSmLgr5b3AWxtRfvQ==", - "license": "MIT", - "dependencies": { - "hachure-fill": "^0.5.2", - "path-data-parser": "^0.1.0", - "points-on-curve": "^0.2.0", - "points-on-path": "^0.2.1" - } - }, - "node_modules/run-parallel": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", - "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT", - "dependencies": { - "queue-microtask": "^1.2.2" - } - }, - "node_modules/rw": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/rw/-/rw-1.3.3.tgz", - "integrity": "sha512-PdhdWy89SiZogBLaw42zdeqtRJ//zFd2PgQavcICDUgJT5oW10QCRKbJ6bg4r0/UY2M6BWd5tkxuGFRvCkgfHQ==", - "license": "BSD-3-Clause" - }, - "node_modules/safer-buffer": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", - "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", - "license": "MIT" - }, - "node_modules/scheduler": { - "version": "0.27.0", - "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.27.0.tgz", - "integrity": "sha512-eNv+WrVbKu1f3vbYJT/xtiF5syA5HPIMtf9IgY/nKg0sWqzAUEvqY/xm7OcZc/qafLx/iO9FgOmeSAp4v5ti/Q==", - "license": "MIT" - }, - "node_modules/semver": { - "version": "6.3.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", - "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", - "dev": true, - "license": "ISC", - "bin": { - "semver": "bin/semver.js" - } - }, - "node_modules/set-cookie-parser": { - "version": "2.7.2", - "resolved": "https://registry.npmjs.org/set-cookie-parser/-/set-cookie-parser-2.7.2.tgz", - "integrity": "sha512-oeM1lpU/UvhTxw+g3cIfxXHyJRc/uidd3yK1P242gzHds0udQBYzs3y8j4gCCW+ZJ7ad0yctld8RYO+bdurlvw==", - "license": "MIT" - }, - "node_modules/shebang-command": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", - "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", - "dev": true, - "license": "MIT", - "dependencies": { - "shebang-regex": "^3.0.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/shebang-regex": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", - "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/siginfo": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/siginfo/-/siginfo-2.0.0.tgz", - "integrity": "sha512-ybx0WO1/8bSBLEWXZvEd7gMW3Sn3JFlW3TvX1nREbDLRNQNaeNN8WK0meBwPdAaOI7TtRRRJn/Es1zhrrCHu7g==", - "dev": true, - "license": "ISC" - }, - "node_modules/signal-exit": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz", - "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==", - "dev": true, - "license": "ISC", - "engines": { - "node": ">=14" - }, - "funding": { - "url": "https://github.com/sponsors/isaacs" - } - }, - "node_modules/source-map-js": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz", - "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==", - "dev": true, - "license": "BSD-3-Clause", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/space-separated-tokens": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/space-separated-tokens/-/space-separated-tokens-2.0.2.tgz", - "integrity": "sha512-PEGlAwrG8yXGXRjW32fGbg66JAlOAwbObuqVoJpv/mRgoWDQfgH1wDPvtzWyUSNAXBGSk8h755YDbbcEy3SH2Q==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/stackback": { - "version": "0.0.2", - "resolved": "https://registry.npmjs.org/stackback/-/stackback-0.0.2.tgz", - "integrity": "sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw==", - "dev": true, - "license": "MIT" - }, - "node_modules/std-env": { - "version": "3.10.0", - "resolved": "https://registry.npmjs.org/std-env/-/std-env-3.10.0.tgz", - "integrity": "sha512-5GS12FdOZNliM5mAOxFRg7Ir0pWz8MdpYm6AY6VPkGpbA7ZzmbzNcBJQ0GPvvyWgcY7QAhCgf9Uy89I03faLkg==", - "dev": true, - "license": "MIT" - }, - "node_modules/string-width": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.2.tgz", - "integrity": "sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==", - "dev": true, - "license": "MIT", - "dependencies": { - "eastasianwidth": "^0.2.0", - "emoji-regex": "^9.2.2", - "strip-ansi": "^7.0.1" - }, - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/string-width-cjs": { - "name": "string-width", - "version": "4.2.3", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", - "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", - "dev": true, - "license": "MIT", - "dependencies": { - "emoji-regex": "^8.0.0", - "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.1" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/string-width-cjs/node_modules/ansi-regex": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", - "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/string-width-cjs/node_modules/emoji-regex": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", - "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", - "dev": true, - "license": "MIT" - }, - "node_modules/string-width-cjs/node_modules/strip-ansi": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", - "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", - "dev": true, - "license": "MIT", - "dependencies": { - "ansi-regex": "^5.0.1" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/stringify-entities": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/stringify-entities/-/stringify-entities-4.0.4.tgz", - "integrity": "sha512-IwfBptatlO+QCJUo19AqvrPNqlVMpW9YEL2LIVY+Rpv2qsjCGxaDLNRgeGsQWJhfItebuJhsGSLjaBbNSQ+ieg==", - "license": "MIT", - "dependencies": { - "character-entities-html4": "^2.0.0", - "character-entities-legacy": "^3.0.0" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/strip-ansi": { - "version": "7.1.2", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.2.tgz", - "integrity": "sha512-gmBGslpoQJtgnMAvOVqGZpEz9dyoKTCzy2nfz/n8aIFhN/jCE/rCmcxabB6jOOHV+0WNnylOxaxBQPSvcWklhA==", - "dev": true, - "license": "MIT", - "dependencies": { - "ansi-regex": "^6.0.1" - }, - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/chalk/strip-ansi?sponsor=1" - } - }, - "node_modules/strip-ansi-cjs": { - "name": "strip-ansi", - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", - "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", - "dev": true, - "license": "MIT", - "dependencies": { - "ansi-regex": "^5.0.1" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/strip-ansi-cjs/node_modules/ansi-regex": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", - "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/strip-json-comments": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", - "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/strip-literal": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/strip-literal/-/strip-literal-3.1.0.tgz", - "integrity": "sha512-8r3mkIM/2+PpjHoOtiAW8Rg3jJLHaV7xPwG+YRGrv6FP0wwk/toTpATxWYOW0BKdWwl82VT2tFYi5DlROa0Mxg==", - "dev": true, - "license": "MIT", - "dependencies": { - "js-tokens": "^9.0.1" - }, - "funding": { - "url": "https://github.com/sponsors/antfu" - } - }, - "node_modules/strip-literal/node_modules/js-tokens": { - "version": "9.0.1", - "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-9.0.1.tgz", - "integrity": "sha512-mxa9E9ITFOt0ban3j6L5MpjwegGz6lBQmM1IJkWeBZGcMxto50+eWdjC/52xDbS2vy0k7vIMK0Fe2wfL9OQSpQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/style-to-js": { - "version": "1.1.19", - "resolved": "https://registry.npmjs.org/style-to-js/-/style-to-js-1.1.19.tgz", - "integrity": "sha512-Ev+SgeqiNGT1ufsXyVC5RrJRXdrkRJ1Gol9Qw7Pb72YCKJXrBvP0ckZhBeVSrw2m06DJpei2528uIpjMb4TsoQ==", - "license": "MIT", - "dependencies": { - "style-to-object": "1.0.12" - } - }, - "node_modules/style-to-object": { - "version": "1.0.12", - "resolved": "https://registry.npmjs.org/style-to-object/-/style-to-object-1.0.12.tgz", - "integrity": "sha512-ddJqYnoT4t97QvN2C95bCgt+m7AAgXjVnkk/jxAfmp7EAB8nnqqZYEbMd3em7/vEomDb2LAQKAy1RFfv41mdNw==", - "license": "MIT", - "dependencies": { - "inline-style-parser": "0.2.6" - } - }, - "node_modules/stylis": { - "version": "4.3.6", - "resolved": "https://registry.npmjs.org/stylis/-/stylis-4.3.6.tgz", - "integrity": "sha512-yQ3rwFWRfwNUY7H5vpU0wfdkNSnvnJinhF9830Swlaxl03zsOjCfmX0ugac+3LtK0lYSgwL/KXc8oYL3mG4YFQ==", - "license": "MIT" - }, - "node_modules/sucrase": { - "version": "3.35.0", - "resolved": "https://registry.npmjs.org/sucrase/-/sucrase-3.35.0.tgz", - "integrity": "sha512-8EbVDiu9iN/nESwxeSxDKe0dunta1GOlHufmSSXxMD2z2/tMZpDMpvXQGsc+ajGo8y2uYUmixaSRUc/QPoQ0GA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@jridgewell/gen-mapping": "^0.3.2", - "commander": "^4.0.0", - "glob": "^10.3.10", - "lines-and-columns": "^1.1.6", - "mz": "^2.7.0", - "pirates": "^4.0.1", - "ts-interface-checker": "^0.1.9" - }, - "bin": { - "sucrase": "bin/sucrase", - "sucrase-node": "bin/sucrase-node" - }, - "engines": { - "node": ">=16 || 14 >=14.17" - } - }, - "node_modules/supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", - "dev": true, - "license": "MIT", - "dependencies": { - "has-flag": "^4.0.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/supports-preserve-symlinks-flag": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", - "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/synckit": { - "version": "0.11.11", - "resolved": "https://registry.npmjs.org/synckit/-/synckit-0.11.11.tgz", - "integrity": "sha512-MeQTA1r0litLUf0Rp/iisCaL8761lKAZHaimlbGK4j0HysC4PLfqygQj9srcs0m2RdtDYnF8UuYyKpbjHYp7Jw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@pkgr/core": "^0.2.9" - }, - "engines": { - "node": "^14.18.0 || >=16.0.0" - }, - "funding": { - "url": "https://opencollective.com/synckit" - } - }, - "node_modules/tailwind-merge": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/tailwind-merge/-/tailwind-merge-3.4.0.tgz", - "integrity": "sha512-uSaO4gnW+b3Y2aWoWfFpX62vn2sR3skfhbjsEnaBI81WD1wBLlHZe5sWf0AqjksNdYTbGBEd0UasQMT3SNV15g==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/dcastil" - } - }, - "node_modules/tailwindcss": { - "version": "3.4.18", - "resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-3.4.18.tgz", - "integrity": "sha512-6A2rnmW5xZMdw11LYjhcI5846rt9pbLSabY5XPxo+XWdxwZaFEn47Go4NzFiHu9sNNmr/kXivP1vStfvMaK1GQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@alloc/quick-lru": "^5.2.0", - "arg": "^5.0.2", - "chokidar": "^3.6.0", - "didyoumean": "^1.2.2", - "dlv": "^1.1.3", - "fast-glob": "^3.3.2", - "glob-parent": "^6.0.2", - "is-glob": "^4.0.3", - "jiti": "^1.21.7", - "lilconfig": "^3.1.3", - "micromatch": "^4.0.8", - "normalize-path": "^3.0.0", - "object-hash": "^3.0.0", - "picocolors": "^1.1.1", - "postcss": "^8.4.47", - "postcss-import": "^15.1.0", - "postcss-js": "^4.0.1", - "postcss-load-config": "^4.0.2 || ^5.0 || ^6.0", - "postcss-nested": "^6.2.0", - "postcss-selector-parser": "^6.1.2", - "resolve": "^1.22.8", - "sucrase": "^3.35.0" - }, - "bin": { - "tailwind": "lib/cli.js", - "tailwindcss": "lib/cli.js" - }, - "engines": { - "node": ">=14.0.0" - } - }, - "node_modules/tailwindcss/node_modules/jiti": { - "version": "1.21.7", - "resolved": "https://registry.npmjs.org/jiti/-/jiti-1.21.7.tgz", - "integrity": "sha512-/imKNG4EbWNrVjoNC/1H5/9GFy+tqjGBHCaSsN+P2RnPqjsLmv6UD3Ej+Kj8nBWaRAwyk7kK5ZUc+OEatnTR3A==", - "dev": true, - "license": "MIT", - "bin": { - "jiti": "bin/jiti.js" - } - }, - "node_modules/test-exclude": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-7.0.1.tgz", - "integrity": "sha512-pFYqmTw68LXVjeWJMST4+borgQP2AyMNbg1BpZh9LbyhUeNkeaPF9gzfPGUAnSMV3qPYdWUwDIjjCLiSDOl7vg==", - "dev": true, - "license": "ISC", - "dependencies": { - "@istanbuljs/schema": "^0.1.2", - "glob": "^10.4.1", - "minimatch": "^9.0.4" - }, - "engines": { - "node": ">=18" - } - }, - "node_modules/test-exclude/node_modules/brace-expansion": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.2.tgz", - "integrity": "sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "balanced-match": "^1.0.0" - } - }, - "node_modules/test-exclude/node_modules/minimatch": { - "version": "9.0.5", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", - "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", - "dev": true, - "license": "ISC", - "dependencies": { - "brace-expansion": "^2.0.1" - }, - "engines": { - "node": ">=16 || 14 >=14.17" - }, - "funding": { - "url": "https://github.com/sponsors/isaacs" - } - }, - "node_modules/thenify": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/thenify/-/thenify-3.3.1.tgz", - "integrity": "sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==", - "dev": true, - "license": "MIT", - "dependencies": { - "any-promise": "^1.0.0" - } - }, - "node_modules/thenify-all": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/thenify-all/-/thenify-all-1.6.0.tgz", - "integrity": "sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==", - "dev": true, - "license": "MIT", - "dependencies": { - "thenify": ">= 3.1.0 < 4" - }, - "engines": { - "node": ">=0.8" - } - }, - "node_modules/tiny-invariant": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/tiny-invariant/-/tiny-invariant-1.3.3.tgz", - "integrity": "sha512-+FbBPE1o9QAYvviau/qC5SE3caw21q3xkvWKBtja5vgqOWIHHJ3ioaq1VPfn/Szqctz2bU/oYeKd9/z5BL+PVg==", - "license": "MIT" - }, - "node_modules/tinybench": { - "version": "2.9.0", - "resolved": "https://registry.npmjs.org/tinybench/-/tinybench-2.9.0.tgz", - "integrity": "sha512-0+DUvqWMValLmha6lr4kD8iAMK1HzV0/aKnCtWb9v9641TnP/MFb7Pc2bxoxQjTXAErryXVgUOfv2YqNllqGeg==", - "dev": true, - "license": "MIT" - }, - "node_modules/tinyexec": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/tinyexec/-/tinyexec-1.0.2.tgz", - "integrity": "sha512-W/KYk+NFhkmsYpuHq5JykngiOCnxeVL8v8dFnqxSD8qEEdRfXk1SDM6JzNqcERbcGYj9tMrDQBYV9cjgnunFIg==", - "license": "MIT", - "engines": { - "node": ">=18" - } - }, - "node_modules/tinyglobby": { - "version": "0.2.15", - "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.15.tgz", - "integrity": "sha512-j2Zq4NyQYG5XMST4cbs02Ak8iJUdxRM0XI5QyxXuZOzKOINmWurp3smXu3y5wDcJrptwpSjgXHzIQxR0omXljQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "fdir": "^6.5.0", - "picomatch": "^4.0.3" - }, - "engines": { - "node": ">=12.0.0" - }, - "funding": { - "url": "https://github.com/sponsors/SuperchupuDev" - } - }, - "node_modules/tinyglobby/node_modules/fdir": { - "version": "6.5.0", - "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz", - "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=12.0.0" - }, - "peerDependencies": { - "picomatch": "^3 || ^4" - }, - "peerDependenciesMeta": { - "picomatch": { - "optional": true - } - } - }, - "node_modules/tinyglobby/node_modules/picomatch": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz", - "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/jonschlinkert" - } - }, - "node_modules/tinypool": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/tinypool/-/tinypool-1.1.1.tgz", - "integrity": "sha512-Zba82s87IFq9A9XmjiX5uZA/ARWDrB03OHlq+Vw1fSdt0I+4/Kutwy8BP4Y/y/aORMo61FQ0vIb5j44vSo5Pkg==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.0.0 || >=20.0.0" - } - }, - "node_modules/tinyrainbow": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/tinyrainbow/-/tinyrainbow-2.0.0.tgz", - "integrity": "sha512-op4nsTR47R6p0vMUUoYl/a+ljLFVtlfaXkLQmqfLR1qHma1h/ysYk4hEXZ880bf2CYgTskvTa/e196Vd5dDQXw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=14.0.0" - } - }, - "node_modules/tinyspy": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/tinyspy/-/tinyspy-4.0.4.tgz", - "integrity": "sha512-azl+t0z7pw/z958Gy9svOTuzqIk6xq+NSheJzn5MMWtWTFywIacg2wUlzKFGtt3cthx0r2SxMK0yzJOR0IES7Q==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=14.0.0" - } - }, - "node_modules/to-regex-range": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", - "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-number": "^7.0.0" - }, - "engines": { - "node": ">=8.0" - } - }, - "node_modules/trim-lines": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/trim-lines/-/trim-lines-3.0.1.tgz", - "integrity": "sha512-kRj8B+YHZCc9kQYdWfJB2/oUl9rA99qbowYYBtr4ui4mZyAQ2JpvVBd/6U2YloATfqBhBTSMhTpgBHtU0Mf3Rg==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/trough": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/trough/-/trough-2.2.0.tgz", - "integrity": "sha512-tmMpK00BjZiUyVyvrBK7knerNgmgvcV/KLVyuma/SC+TQN167GrMRciANTz09+k3zW8L8t60jWO1GpfkZdjTaw==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - }, - "node_modules/ts-api-utils": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.1.0.tgz", - "integrity": "sha512-CUgTZL1irw8u29bzrOD/nH85jqyc74D6SshFgujOIA7osm2Rz7dYH77agkx7H4FBNxDq7Cjf+IjaX/8zwFW+ZQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=18.12" - }, - "peerDependencies": { - "typescript": ">=4.8.4" - } - }, - "node_modules/ts-dedent": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/ts-dedent/-/ts-dedent-2.2.0.tgz", - "integrity": "sha512-q5W7tVM71e2xjHZTlgfTDoPF/SmqKG5hddq9SzR49CH2hayqRKJtQ4mtRlSxKaJlR/+9rEM+mnBHf7I2/BQcpQ==", - "license": "MIT", - "engines": { - "node": ">=6.10" - } - }, - "node_modules/ts-interface-checker": { - "version": "0.1.13", - "resolved": "https://registry.npmjs.org/ts-interface-checker/-/ts-interface-checker-0.1.13.tgz", - "integrity": "sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==", - "dev": true, - "license": "Apache-2.0" - }, - "node_modules/tslib": { - "version": "2.8.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", - "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", - "license": "0BSD" - }, - "node_modules/type-check": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", - "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", - "dev": true, - "license": "MIT", - "dependencies": { - "prelude-ls": "^1.2.1" - }, - "engines": { - "node": ">= 0.8.0" - } - }, - "node_modules/typescript": { - "version": "5.9.3", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz", - "integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==", - "dev": true, - "license": "Apache-2.0", - "bin": { - "tsc": "bin/tsc", - "tsserver": "bin/tsserver" - }, - "engines": { - "node": ">=14.17" - } - }, - "node_modules/typescript-eslint": { - "version": "8.46.4", - "resolved": "https://registry.npmjs.org/typescript-eslint/-/typescript-eslint-8.46.4.tgz", - "integrity": "sha512-KALyxkpYV5Ix7UhvjTwJXZv76VWsHG+NjNlt/z+a17SOQSiOcBdUXdbJdyXi7RPxrBFECtFOiPwUJQusJuCqrg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/eslint-plugin": "8.46.4", - "@typescript-eslint/parser": "8.46.4", - "@typescript-eslint/typescript-estree": "8.46.4", - "@typescript-eslint/utils": "8.46.4" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0", - "typescript": ">=4.8.4 <6.0.0" - } - }, - "node_modules/ufo": { - "version": "1.6.1", - "resolved": "https://registry.npmjs.org/ufo/-/ufo-1.6.1.tgz", - "integrity": "sha512-9a4/uxlTWJ4+a5i0ooc1rU7C7YOw3wT+UGqdeNNHWnOF9qcMBgLRS+4IYUqbczewFx4mLEig6gawh7X6mFlEkA==", - "license": "MIT" - }, - "node_modules/undici-types": { - "version": "7.16.0", - "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.16.0.tgz", - "integrity": "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw==", - "license": "MIT" - }, - "node_modules/unified": { - "version": "11.0.5", - "resolved": "https://registry.npmjs.org/unified/-/unified-11.0.5.tgz", - "integrity": "sha512-xKvGhPWw3k84Qjh8bI3ZeJjqnyadK+GEFtazSfZv/rKeTkTjOJho6mFqh2SM96iIcZokxiOpg78GazTSg8+KHA==", - "license": "MIT", - "dependencies": { - "@types/unist": "^3.0.0", - "bail": "^2.0.0", - "devlop": "^1.0.0", - "extend": "^3.0.0", - "is-plain-obj": "^4.0.0", - "trough": "^2.0.0", - "vfile": "^6.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/unist-util-is": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/unist-util-is/-/unist-util-is-6.0.1.tgz", - "integrity": "sha512-LsiILbtBETkDz8I9p1dQ0uyRUWuaQzd/cuEeS1hoRSyW5E5XGmTzlwY1OrNzzakGowI9Dr/I8HVaw4hTtnxy8g==", - "license": "MIT", - "dependencies": { - "@types/unist": "^3.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/unist-util-position": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/unist-util-position/-/unist-util-position-5.0.0.tgz", - "integrity": "sha512-fucsC7HjXvkB5R3kTCO7kUjRdrS0BJt3M/FPxmHMBOm8JQi2BsHAHFsy27E0EolP8rp0NzXsJ+jNPyDWvOJZPA==", - "license": "MIT", - "dependencies": { - "@types/unist": "^3.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/unist-util-stringify-position": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/unist-util-stringify-position/-/unist-util-stringify-position-4.0.0.tgz", - "integrity": "sha512-0ASV06AAoKCDkS2+xw5RXJywruurpbC4JZSm7nr7MOt1ojAzvyyaO+UxZf18j8FCF6kmzCZKcAgN/yu2gm2XgQ==", - "license": "MIT", - "dependencies": { - "@types/unist": "^3.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/unist-util-visit": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/unist-util-visit/-/unist-util-visit-5.0.0.tgz", - "integrity": "sha512-MR04uvD+07cwl/yhVuVWAtw+3GOR/knlL55Nd/wAdblk27GCVt3lqpTivy/tkJcZoNPzTwS1Y+KMojlLDhoTzg==", - "license": "MIT", - "dependencies": { - "@types/unist": "^3.0.0", - "unist-util-is": "^6.0.0", - "unist-util-visit-parents": "^6.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/unist-util-visit-parents": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/unist-util-visit-parents/-/unist-util-visit-parents-6.0.2.tgz", - "integrity": "sha512-goh1s1TBrqSqukSc8wrjwWhL0hiJxgA8m4kFxGlQ+8FYQ3C/m11FcTs4YYem7V664AhHVvgoQLk890Ssdsr2IQ==", - "license": "MIT", - "dependencies": { - "@types/unist": "^3.0.0", - "unist-util-is": "^6.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/update-browserslist-db": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.1.4.tgz", - "integrity": "sha512-q0SPT4xyU84saUX+tomz1WLkxUbuaJnR1xWt17M7fJtEJigJeWUNGUqrauFXsHnqev9y9JTRGwk13tFBuKby4A==", - "dev": true, - "funding": [ - { - "type": "opencollective", - "url": "https://opencollective.com/browserslist" - }, - { - "type": "tidelift", - "url": "https://tidelift.com/funding/github/npm/browserslist" - }, - { - "type": "github", - "url": "https://github.com/sponsors/ai" - } - ], - "license": "MIT", - "dependencies": { - "escalade": "^3.2.0", - "picocolors": "^1.1.1" - }, - "bin": { - "update-browserslist-db": "cli.js" - }, - "peerDependencies": { - "browserslist": ">= 4.21.0" - } - }, - "node_modules/uri-js": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", - "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "punycode": "^2.1.0" - } - }, - "node_modules/use-sync-external-store": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.6.0.tgz", - "integrity": "sha512-Pp6GSwGP/NrPIrxVFAIkOQeyw8lFenOHijQWkUTrDvrF4ALqylP2C/KCkeS9dpUM3KvYRQhna5vt7IL95+ZQ9w==", - "license": "MIT", - "peerDependencies": { - "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" - } - }, - "node_modules/util-deprecate": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", - "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", - "dev": true, - "license": "MIT" - }, - "node_modules/uuid": { - "version": "11.1.0", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-11.1.0.tgz", - "integrity": "sha512-0/A9rDy9P7cJ+8w1c9WD9V//9Wj15Ce2MPz8Ri6032usz+NfePxx5AcN3bN+r6ZL6jEo066/yNYB3tn4pQEx+A==", - "funding": [ - "https://github.com/sponsors/broofa", - "https://github.com/sponsors/ctavan" - ], - "license": "MIT", - "bin": { - "uuid": "dist/esm/bin/uuid" - } - }, - "node_modules/vfile": { - "version": "6.0.3", - "resolved": "https://registry.npmjs.org/vfile/-/vfile-6.0.3.tgz", - "integrity": "sha512-KzIbH/9tXat2u30jf+smMwFCsno4wHVdNmzFyL+T/L3UGqqk6JKfVqOFOZEpZSHADH1k40ab6NUIXZq422ov3Q==", - "license": "MIT", - "dependencies": { - "@types/unist": "^3.0.0", - "vfile-message": "^4.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/vfile-message": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/vfile-message/-/vfile-message-4.0.3.tgz", - "integrity": "sha512-QTHzsGd1EhbZs4AsQ20JX1rC3cOlt/IWJruk893DfLRr57lcnOeMaWG4K0JrRta4mIJZKth2Au3mM3u03/JWKw==", - "license": "MIT", - "dependencies": { - "@types/unist": "^3.0.0", - "unist-util-stringify-position": "^4.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/unified" - } - }, - "node_modules/vite": { - "version": "7.2.2", - "resolved": "https://registry.npmjs.org/vite/-/vite-7.2.2.tgz", - "integrity": "sha512-BxAKBWmIbrDgrokdGZH1IgkIk/5mMHDreLDmCJ0qpyJaAteP8NvMhkwr/ZCQNqNH97bw/dANTE9PDzqwJghfMQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "esbuild": "^0.25.0", - "fdir": "^6.5.0", - "picomatch": "^4.0.3", - "postcss": "^8.5.6", - "rollup": "^4.43.0", - "tinyglobby": "^0.2.15" - }, - "bin": { - "vite": "bin/vite.js" - }, - "engines": { - "node": "^20.19.0 || >=22.12.0" - }, - "funding": { - "url": "https://github.com/vitejs/vite?sponsor=1" - }, - "optionalDependencies": { - "fsevents": "~2.3.3" - }, - "peerDependencies": { - "@types/node": "^20.19.0 || >=22.12.0", - "jiti": ">=1.21.0", - "less": "^4.0.0", - "lightningcss": "^1.21.0", - "sass": "^1.70.0", - "sass-embedded": "^1.70.0", - "stylus": ">=0.54.8", - "sugarss": "^5.0.0", - "terser": "^5.16.0", - "tsx": "^4.8.1", - "yaml": "^2.4.2" - }, - "peerDependenciesMeta": { - "@types/node": { - "optional": true - }, - "jiti": { - "optional": true - }, - "less": { - "optional": true - }, - "lightningcss": { - "optional": true - }, - "sass": { - "optional": true - }, - "sass-embedded": { - "optional": true - }, - "stylus": { - "optional": true - }, - "sugarss": { - "optional": true - }, - "terser": { - "optional": true - }, - "tsx": { - "optional": true - }, - "yaml": { - "optional": true - } - } - }, - "node_modules/vite-node": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/vite-node/-/vite-node-3.2.4.tgz", - "integrity": "sha512-EbKSKh+bh1E1IFxeO0pg1n4dvoOTt0UDiXMd/qn++r98+jPO1xtJilvXldeuQ8giIB5IkpjCgMleHMNEsGH6pg==", - "dev": true, - "license": "MIT", - "dependencies": { - "cac": "^6.7.14", - "debug": "^4.4.1", - "es-module-lexer": "^1.7.0", - "pathe": "^2.0.3", - "vite": "^5.0.0 || ^6.0.0 || ^7.0.0-0" - }, - "bin": { - "vite-node": "vite-node.mjs" - }, - "engines": { - "node": "^18.0.0 || ^20.0.0 || >=22.0.0" - }, - "funding": { - "url": "https://opencollective.com/vitest" - } - }, - "node_modules/vite/node_modules/fdir": { - "version": "6.5.0", - "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz", - "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=12.0.0" - }, - "peerDependencies": { - "picomatch": "^3 || ^4" - }, - "peerDependenciesMeta": { - "picomatch": { - "optional": true - } - } - }, - "node_modules/vite/node_modules/picomatch": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz", - "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/jonschlinkert" - } - }, - "node_modules/vitest": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/vitest/-/vitest-3.2.4.tgz", - "integrity": "sha512-LUCP5ev3GURDysTWiP47wRRUpLKMOfPh+yKTx3kVIEiu5KOMeqzpnYNsKyOoVrULivR8tLcks4+lga33Whn90A==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/chai": "^5.2.2", - "@vitest/expect": "3.2.4", - "@vitest/mocker": "3.2.4", - "@vitest/pretty-format": "^3.2.4", - "@vitest/runner": "3.2.4", - "@vitest/snapshot": "3.2.4", - "@vitest/spy": "3.2.4", - "@vitest/utils": "3.2.4", - "chai": "^5.2.0", - "debug": "^4.4.1", - "expect-type": "^1.2.1", - "magic-string": "^0.30.17", - "pathe": "^2.0.3", - "picomatch": "^4.0.2", - "std-env": "^3.9.0", - "tinybench": "^2.9.0", - "tinyexec": "^0.3.2", - "tinyglobby": "^0.2.14", - "tinypool": "^1.1.1", - "tinyrainbow": "^2.0.0", - "vite": "^5.0.0 || ^6.0.0 || ^7.0.0-0", - "vite-node": "3.2.4", - "why-is-node-running": "^2.3.0" - }, - "bin": { - "vitest": "vitest.mjs" - }, - "engines": { - "node": "^18.0.0 || ^20.0.0 || >=22.0.0" - }, - "funding": { - "url": "https://opencollective.com/vitest" - }, - "peerDependencies": { - "@edge-runtime/vm": "*", - "@types/debug": "^4.1.12", - "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0", - "@vitest/browser": "3.2.4", - "@vitest/ui": "3.2.4", - "happy-dom": "*", - "jsdom": "*" - }, - "peerDependenciesMeta": { - "@edge-runtime/vm": { - "optional": true - }, - "@types/debug": { - "optional": true - }, - "@types/node": { - "optional": true - }, - "@vitest/browser": { - "optional": true - }, - "@vitest/ui": { - "optional": true - }, - "happy-dom": { - "optional": true - }, - "jsdom": { - "optional": true - } - } - }, - "node_modules/vitest/node_modules/picomatch": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz", - "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/jonschlinkert" - } - }, - "node_modules/vitest/node_modules/tinyexec": { - "version": "0.3.2", - "resolved": "https://registry.npmjs.org/tinyexec/-/tinyexec-0.3.2.tgz", - "integrity": "sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA==", - "dev": true, - "license": "MIT" - }, - "node_modules/vscode-jsonrpc": { - "version": "8.2.0", - "resolved": "https://registry.npmjs.org/vscode-jsonrpc/-/vscode-jsonrpc-8.2.0.tgz", - "integrity": "sha512-C+r0eKJUIfiDIfwJhria30+TYWPtuHJXHtI7J0YlOmKAo7ogxP20T0zxB7HZQIFhIyvoBPwWskjxrvAtfjyZfA==", - "license": "MIT", - "engines": { - "node": ">=14.0.0" - } - }, - "node_modules/vscode-languageserver": { - "version": "9.0.1", - "resolved": "https://registry.npmjs.org/vscode-languageserver/-/vscode-languageserver-9.0.1.tgz", - "integrity": "sha512-woByF3PDpkHFUreUa7Hos7+pUWdeWMXRd26+ZX2A8cFx6v/JPTtd4/uN0/jB6XQHYaOlHbio03NTHCqrgG5n7g==", - "license": "MIT", - "dependencies": { - "vscode-languageserver-protocol": "3.17.5" - }, - "bin": { - "installServerIntoExtension": "bin/installServerIntoExtension" - } - }, - "node_modules/vscode-languageserver-protocol": { - "version": "3.17.5", - "resolved": "https://registry.npmjs.org/vscode-languageserver-protocol/-/vscode-languageserver-protocol-3.17.5.tgz", - "integrity": "sha512-mb1bvRJN8SVznADSGWM9u/b07H7Ecg0I3OgXDuLdn307rl/J3A9YD6/eYOssqhecL27hK1IPZAsaqh00i/Jljg==", - "license": "MIT", - "dependencies": { - "vscode-jsonrpc": "8.2.0", - "vscode-languageserver-types": "3.17.5" - } - }, - "node_modules/vscode-languageserver-textdocument": { - "version": "1.0.12", - "resolved": "https://registry.npmjs.org/vscode-languageserver-textdocument/-/vscode-languageserver-textdocument-1.0.12.tgz", - "integrity": "sha512-cxWNPesCnQCcMPeenjKKsOCKQZ/L6Tv19DTRIGuLWe32lyzWhihGVJ/rcckZXJxfdKCFvRLS3fpBIsV/ZGX4zA==", - "license": "MIT" - }, - "node_modules/vscode-languageserver-types": { - "version": "3.17.5", - "resolved": "https://registry.npmjs.org/vscode-languageserver-types/-/vscode-languageserver-types-3.17.5.tgz", - "integrity": "sha512-Ld1VelNuX9pdF39h2Hgaeb5hEZM2Z3jUrrMgWQAu82jMtZp7p3vJT3BzToKtZI7NgQssZje5o0zryOrhQvzQAg==", - "license": "MIT" - }, - "node_modules/vscode-uri": { - "version": "3.0.8", - "resolved": "https://registry.npmjs.org/vscode-uri/-/vscode-uri-3.0.8.tgz", - "integrity": "sha512-AyFQ0EVmsOZOlAnxoFOGOq1SQDWAB7C6aqMGS23svWAllfOaxbuFvcT8D1i8z3Gyn8fraVeZNNmN6e9bxxXkKw==", - "license": "MIT" - }, - "node_modules/warning": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/warning/-/warning-4.0.3.tgz", - "integrity": "sha512-rpJyN222KWIvHJ/F53XSZv0Zl/accqHR8et1kpaMTD/fLCRxtV8iX8czMzY7sVZupTI3zcUTg8eycS2kNF9l6w==", - "license": "MIT", - "dependencies": { - "loose-envify": "^1.0.0" - } - }, - "node_modules/which": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", - "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", - "dev": true, - "license": "ISC", - "dependencies": { - "isexe": "^2.0.0" - }, - "bin": { - "node-which": "bin/node-which" - }, - "engines": { - "node": ">= 8" - } - }, - "node_modules/why-is-node-running": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/why-is-node-running/-/why-is-node-running-2.3.0.tgz", - "integrity": "sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w==", - "dev": true, - "license": "MIT", - "dependencies": { - "siginfo": "^2.0.0", - "stackback": "0.0.2" - }, - "bin": { - "why-is-node-running": "cli.js" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/word-wrap": { - "version": "1.2.5", - "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", - "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/wrap-ansi": { - "version": "8.1.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-8.1.0.tgz", - "integrity": "sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "ansi-styles": "^6.1.0", - "string-width": "^5.0.1", - "strip-ansi": "^7.0.1" - }, - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/chalk/wrap-ansi?sponsor=1" - } - }, - "node_modules/wrap-ansi-cjs": { - "name": "wrap-ansi", - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", - "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "ansi-styles": "^4.0.0", - "string-width": "^4.1.0", - "strip-ansi": "^6.0.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/wrap-ansi?sponsor=1" - } - }, - "node_modules/wrap-ansi-cjs/node_modules/ansi-regex": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", - "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/wrap-ansi-cjs/node_modules/emoji-regex": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", - "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", - "dev": true, - "license": "MIT" - }, - "node_modules/wrap-ansi-cjs/node_modules/string-width": { - "version": "4.2.3", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", - "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", - "dev": true, - "license": "MIT", - "dependencies": { - "emoji-regex": "^8.0.0", - "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.1" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/wrap-ansi-cjs/node_modules/strip-ansi": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", - "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", - "dev": true, - "license": "MIT", - "dependencies": { - "ansi-regex": "^5.0.1" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/wrap-ansi/node_modules/ansi-styles": { - "version": "6.2.3", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.2.3.tgz", - "integrity": "sha512-4Dj6M28JB+oAH8kFkTLUo+a2jwOFkuqb3yucU0CANcRRUbxS0cP0nZYCGjcc3BNXwRIsUVmDGgzawme7zvJHvg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, - "node_modules/ws": { - "version": "8.18.3", - "resolved": "https://registry.npmjs.org/ws/-/ws-8.18.3.tgz", - "integrity": "sha512-PEIGCY5tSlUt50cqyMXfCzX+oOPqN0vuGqWzbcJ2xvnkzkq46oOpz7dQaTDBdfICb4N14+GARUDw2XV2N4tvzg==", - "license": "MIT", - "engines": { - "node": ">=10.0.0" - }, - "peerDependencies": { - "bufferutil": "^4.0.1", - "utf-8-validate": ">=5.0.2" - }, - "peerDependenciesMeta": { - "bufferutil": { - "optional": true - }, - "utf-8-validate": { - "optional": true - } - } - }, - "node_modules/yallist": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", - "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", - "dev": true, - "license": "ISC" - }, - "node_modules/yocto-queue": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", - "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/zustand": { - "version": "5.0.8", - "resolved": "https://registry.npmjs.org/zustand/-/zustand-5.0.8.tgz", - "integrity": "sha512-gyPKpIaxY9XcO2vSMrLbiER7QMAMGOQZVRdJ6Zi782jkbzZygq5GI9nG8g+sMgitRtndwaBSl7uiqC49o1SSiw==", - "license": "MIT", - "engines": { - "node": ">=12.20.0" - }, - "peerDependencies": { - "@types/react": ">=18.0.0", - "immer": ">=9.0.6", - "react": ">=18.0.0", - "use-sync-external-store": ">=1.2.0" - }, - "peerDependenciesMeta": { - "@types/react": { - "optional": true - }, - "immer": { - "optional": true - }, - "react": { - "optional": true - }, - "use-sync-external-store": { - "optional": true - } - } - }, - "node_modules/zwitch": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/zwitch/-/zwitch-2.0.4.tgz", - "integrity": "sha512-bXE4cR/kVZhKZX/RjPEflHaKVhUVl85noU3v6b8apfQEc1x4A+zBxjZ4lN8LqGd6WZ3dl98pY4o717VFmoPp+A==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/wooorm" - } - } - } -} diff --git a/frontend/package.json b/frontend/package.json deleted file mode 100644 index 38820395..00000000 --- a/frontend/package.json +++ /dev/null @@ -1,76 +0,0 @@ -{ - "name": "frontend", - "private": true, - "version": "0.0.0", - "type": "module", - "scripts": { - "dev": "vite", - "build": "tsc -b && vite build", - "preview": "vite preview", - "lint": "eslint . --max-warnings=0", - "lint:fix": "eslint . --fix", - "format": "prettier --write \"src/**/*.{ts,tsx,js,jsx,json,css,md}\"", - "format:check": "prettier --check \"src/**/*.{ts,tsx,js,jsx,json,css,md}\"", - "type-check": "tsc --noEmit", - "test": "vitest", - "test:run": "vitest run", - "test:coverage": "vitest run --coverage" - }, - "dependencies": { - "@dagrejs/dagre": "^1.1.8", - "@marsidev/react-turnstile": "^1.3.1", - "@radix-ui/react-slot": "^1.2.4", - "@supabase/supabase-js": "^2.81.1", - "@types/dagre": "^0.7.53", - "@xyflow/react": "^12.9.3", - "chart.js": "^4.5.1", - "class-variance-authority": "^0.7.1", - "clsx": "^2.1.1", - "dagre": "^0.8.5", - "dompurify": "^3.3.0", - "driver.js": "^1.4.0", - "framer-motion": "^12.23.24", - "html-to-image": "^1.11.13", - "idb": "^8.0.3", - "lucide-react": "^0.553.0", - "mermaid": "^11.12.1", - "pdfjs-dist": "^5.4.530", - "react": "^19.2.0", - "react-chartjs-2": "^5.3.1", - "react-dom": "^19.2.0", - "react-markdown": "^10.1.0", - "react-pdf": "^10.3.0", - "react-resizable-panels": "^3.0.6", - "react-router-dom": "^7.9.5", - "react-syntax-highlighter": "^16.1.0", - "react-virtuoso": "^4.14.1", - "react-zoom-pan-pinch": "^3.7.0", - "remark-gfm": "^4.0.1", - "tailwind-merge": "^3.4.0", - "zustand": "^5.0.8" - }, - "devDependencies": { - "@eslint/js": "^9.39.1", - "@types/dompurify": "^3.0.5", - "@types/node": "^24.10.0", - "@types/react": "^19.2.2", - "@types/react-dom": "^19.2.2", - "@types/react-syntax-highlighter": "^15.5.13", - "@vitejs/plugin-react": "^5.1.0", - "@vitest/coverage-v8": "^3.2.4", - "autoprefixer": "^10.4.22", - "eslint": "^9.39.1", - "eslint-config-prettier": "^10.1.8", - "eslint-plugin-prettier": "^5.5.4", - "eslint-plugin-react-hooks": "^5.2.0", - "eslint-plugin-react-refresh": "^0.4.24", - "globals": "^16.5.0", - "postcss": "^8.5.6", - "prettier": "^3.6.2", - "tailwindcss": "^3.4.17", - "typescript": "~5.9.3", - "typescript-eslint": "^8.46.3", - "vite": "^7.2.2", - "vitest": "^3.2.4" - } -} diff --git a/frontend/public/clara-welcome (1).png b/frontend/public/clara-welcome (1).png deleted file mode 100644 index 2651c7a3..00000000 Binary files a/frontend/public/clara-welcome (1).png and /dev/null differ diff --git a/frontend/public/clara-welcome.png b/frontend/public/clara-welcome.png deleted file mode 100644 index 2651c7a3..00000000 Binary files a/frontend/public/clara-welcome.png and /dev/null differ diff --git a/frontend/public/image-1.webp b/frontend/public/image-1.webp deleted file mode 100644 index 05989afc..00000000 Binary files a/frontend/public/image-1.webp and /dev/null differ diff --git a/frontend/public/image-2.webp b/frontend/public/image-2.webp deleted file mode 100644 index 68410ca0..00000000 Binary files a/frontend/public/image-2.webp and /dev/null differ diff --git a/frontend/public/image-3.webp b/frontend/public/image-3.webp deleted file mode 100644 index 124e7865..00000000 Binary files a/frontend/public/image-3.webp and /dev/null differ diff --git a/frontend/public/image-4.webp b/frontend/public/image-4.webp deleted file mode 100644 index c490fc7d..00000000 Binary files a/frontend/public/image-4.webp and /dev/null differ diff --git a/frontend/public/image-banner.png b/frontend/public/image-banner.png deleted file mode 100644 index 53b0e3f2..00000000 Binary files a/frontend/public/image-banner.png and /dev/null differ diff --git a/frontend/public/vite.svg b/frontend/public/vite.svg deleted file mode 100644 index e7b8dfb1..00000000 --- a/frontend/public/vite.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/frontend/src/App.tsx b/frontend/src/App.tsx deleted file mode 100644 index 8dde825a..00000000 --- a/frontend/src/App.tsx +++ /dev/null @@ -1,35 +0,0 @@ -import { useEffect } from 'react'; -import { AppRouter } from '@/routes'; -import { ToastContainer } from '@/components/design-system'; -import { UpgradePromptModal, WelcomePopupModal } from '@/components/subscription'; - -function App() { - // Apply Christmas theme during Dec 15 - Jan 1 - useEffect(() => { - const now = new Date(); - const month = now.getMonth(); // 0-11 - const day = now.getDate(); - - // Dec 15 - Jan 1 (month 11 = Dec, month 0 = Jan) - const isChristmasSeason = (month === 11 && day >= 15) || (month === 0 && day <= 1); - - if (isChristmasSeason) { - document.documentElement.setAttribute('data-theme', 'christmas'); - } - - return () => { - document.documentElement.removeAttribute('data-theme'); - }; - }, []); - - return ( - <> - - - - - - ); -} - -export default App; diff --git a/frontend/src/assets/favicon-16x16.png b/frontend/src/assets/favicon-16x16.png deleted file mode 100644 index dc506b7b..00000000 Binary files a/frontend/src/assets/favicon-16x16.png and /dev/null differ diff --git a/frontend/src/assets/favicon-32x32.png b/frontend/src/assets/favicon-32x32.png deleted file mode 100644 index 11243999..00000000 Binary files a/frontend/src/assets/favicon-32x32.png and /dev/null differ diff --git a/frontend/src/assets/favicon.ico b/frontend/src/assets/favicon.ico deleted file mode 100644 index 22d39428..00000000 Binary files a/frontend/src/assets/favicon.ico and /dev/null differ diff --git a/frontend/src/assets/mascot/image-banner.png b/frontend/src/assets/mascot/image-banner.png deleted file mode 100644 index 53b0e3f2..00000000 Binary files a/frontend/src/assets/mascot/image-banner.png and /dev/null differ diff --git a/frontend/src/assets/spinner.gif b/frontend/src/assets/spinner.gif deleted file mode 100644 index e1f7a754..00000000 Binary files a/frontend/src/assets/spinner.gif and /dev/null differ diff --git a/frontend/src/assets/spinner.webm b/frontend/src/assets/spinner.webm deleted file mode 100644 index 2970e556..00000000 Binary files a/frontend/src/assets/spinner.webm and /dev/null differ diff --git a/frontend/src/components/ErrorBoundary/ErrorBoundary.module.css b/frontend/src/components/ErrorBoundary/ErrorBoundary.module.css deleted file mode 100644 index dd2586ab..00000000 --- a/frontend/src/components/ErrorBoundary/ErrorBoundary.module.css +++ /dev/null @@ -1,248 +0,0 @@ -.container { - min-height: 100vh; - display: flex; - align-items: center; - justify-content: center; - background: var(--color-bg-primary); - padding: var(--space-6); -} - -.content { - max-width: 500px; - width: 100%; - display: flex; - flex-direction: column; - align-items: center; - text-align: center; -} - -.mascot { - width: 200px; - height: auto; - margin-bottom: var(--space-6); - animation: float 3s ease-in-out infinite; -} - -@keyframes float { - 0%, - 100% { - transform: translateY(0); - } - 50% { - transform: translateY(-10px); - } -} - -.title { - font-size: var(--text-2xl); - font-weight: 600; - color: var(--color-text-primary); - margin: 0 0 var(--space-3) 0; -} - -.subtitle { - font-size: var(--text-base); - color: var(--color-text-secondary); - margin: 0 0 var(--space-6) 0; - line-height: 1.6; -} - -.checkboxLabel { - display: flex; - align-items: center; - gap: var(--space-2); - font-size: var(--text-sm); - color: var(--color-text-primary); - cursor: pointer; - margin-bottom: var(--space-3); - user-select: none; -} - -.checkbox { - width: 18px; - height: 18px; - accent-color: var(--color-accent); - cursor: pointer; -} - -.privacyNote { - font-size: var(--text-xs); - color: var(--color-text-tertiary); - max-width: 380px; - line-height: 1.5; - margin: 0 0 var(--space-5) 0; - text-align: center; -} - -.primaryButton, -.secondaryButton { - display: inline-flex; - align-items: center; - justify-content: center; - gap: var(--space-2); - padding: var(--space-3) var(--space-6); - border-radius: var(--radius-lg); - font-size: var(--text-sm); - font-weight: 500; - cursor: pointer; - transition: all var(--transition-fast); - border: none; - width: 100%; - max-width: 300px; -} - -.primaryButton { - background: linear-gradient(135deg, var(--color-accent), var(--color-accent-secondary, #8b5cf6)); - color: white; - padding: var(--space-4) var(--space-6); - font-size: var(--text-base); - font-weight: 600; - border-radius: var(--radius-xl); - box-shadow: 0 4px 14px rgba(var(--color-accent-rgb, 139, 92, 246), 0.3); - margin-bottom: var(--space-3); -} - -.primaryButton:hover:not(:disabled) { - transform: translateY(-2px); - box-shadow: 0 6px 20px rgba(var(--color-accent-rgb, 139, 92, 246), 0.4); -} - -.primaryButton:disabled { - opacity: 0.7; - cursor: not-allowed; -} - -.secondaryButton { - background: var(--color-surface-elevated); - color: var(--color-text-primary); - border: 1px solid var(--color-border); - margin-bottom: var(--space-4); -} - -.secondaryButton:hover:not(:disabled) { - background: var(--color-surface-hover); - transform: translateY(-1px); -} - -.secondaryButton:disabled { - opacity: 0.7; - cursor: not-allowed; -} - -.errorText { - color: var(--color-error); - font-size: var(--text-sm); - margin: var(--space-2) 0; -} - -.successText { - color: var(--color-success); - font-size: var(--text-sm); - margin: var(--space-2) 0; -} - -.detailsToggle { - display: inline-flex; - align-items: center; - gap: var(--space-1); - background: none; - border: none; - color: var(--color-text-tertiary); - font-size: var(--text-sm); - cursor: pointer; - padding: var(--space-2); - transition: color var(--transition-fast); -} - -.detailsToggle:hover { - color: var(--color-text-secondary); -} - -.details { - width: 100%; - margin-top: var(--space-4); - text-align: left; - background: var(--color-surface); - border: 1px solid var(--color-border); - border-radius: var(--radius-lg); - overflow: hidden; -} - -.detailSection { - padding: var(--space-4); - border-bottom: 1px solid var(--color-border); -} - -.detailSection:last-child { - border-bottom: none; -} - -.detailSection h3 { - font-size: var(--text-xs); - font-weight: 600; - color: var(--color-text-secondary); - text-transform: uppercase; - letter-spacing: 0.05em; - margin: 0 0 var(--space-2) 0; -} - -.detailSection pre { - font-family: 'SF Mono', 'Fira Code', 'Consolas', monospace; - font-size: var(--text-xs); - color: var(--color-text-tertiary); - background: var(--color-bg-secondary); - padding: var(--space-3); - border-radius: var(--radius-md); - overflow-x: auto; - white-space: pre-wrap; - word-break: break-word; - margin: 0; - max-height: 200px; - overflow-y: auto; -} - -.spinner { - width: 16px; - height: 16px; - border: 2px solid transparent; - border-top-color: currentColor; - border-radius: 50%; - animation: spin 0.8s linear infinite; -} - -@keyframes spin { - to { - transform: rotate(360deg); - } -} - -/* Responsive */ -@media (max-width: 480px) { - .container { - padding: var(--space-4); - } - - .mascot { - width: 150px; - } - - .title { - font-size: var(--text-xl); - } - - .actions { - flex-direction: column; - width: 100%; - } - - .primaryButton, - .secondaryButton { - width: 100%; - justify-content: center; - } - - .newChatButton { - width: 100%; - justify-content: center; - } -} diff --git a/frontend/src/components/ErrorBoundary/ErrorBoundary.tsx b/frontend/src/components/ErrorBoundary/ErrorBoundary.tsx deleted file mode 100644 index b61d589f..00000000 --- a/frontend/src/components/ErrorBoundary/ErrorBoundary.tsx +++ /dev/null @@ -1,286 +0,0 @@ -import { Component, type ErrorInfo, type ReactNode } from 'react'; -import { Copy, RefreshCw, ChevronDown, ChevronUp, Check } from 'lucide-react'; -import ErrorClara from '@/assets/mascot/Error_Clara.png'; -import styles from './ErrorBoundary.module.css'; - -interface Props { - children: ReactNode; -} - -interface State { - hasError: boolean; - error: Error | null; - errorInfo: ErrorInfo | null; - showDetails: boolean; - copySuccess: boolean; - sendingReport: boolean; - reportSent: boolean; - reportError: string | null; - sendDiagnostics: boolean; -} - -const DISCORD_WEBHOOK_URL = import.meta.env.VITE_DISCORD_ERROR_WEBHOOK || ''; - -class ErrorBoundary extends Component { - constructor(props: Props) { - super(props); - this.state = { - hasError: false, - error: null, - errorInfo: null, - showDetails: false, - copySuccess: false, - sendingReport: false, - reportSent: false, - reportError: null, - sendDiagnostics: true, // Default to sending diagnostics - }; - } - - static getDerivedStateFromError(error: Error): Partial { - return { hasError: true, error }; - } - - componentDidCatch(error: Error, errorInfo: ErrorInfo): void { - this.setState({ errorInfo }); - console.error('ErrorBoundary caught an error:', error, errorInfo); - } - - generateErrorReport = (): string => { - const { error, errorInfo } = this.state; - const timestamp = new Date().toISOString(); - const appVersion = import.meta.env.VITE_APP_VERSION || 'unknown'; - - return ` -═══════════════════════════════════════════════════ - CLARAVERSE ERROR REPORT -═══════════════════════════════════════════════════ - -📅 Timestamp: ${timestamp} -🌐 URL: ${window.location.href} -📱 Browser: ${navigator.userAgent} -📦 App Version: ${appVersion} - -─────────────────────────────────────────────────── -❌ ERROR MESSAGE -─────────────────────────────────────────────────── -${error?.message || 'Unknown error'} - -─────────────────────────────────────────────────── -📚 STACK TRACE -─────────────────────────────────────────────────── -${error?.stack || 'No stack trace available'} - -─────────────────────────────────────────────────── -🧩 COMPONENT STACK -─────────────────────────────────────────────────── -${errorInfo?.componentStack || 'No component stack available'} - -═══════════════════════════════════════════════════ -`.trim(); - }; - - handleCopyReport = async (): Promise => { - try { - const report = this.generateErrorReport(); - await navigator.clipboard.writeText(report); - this.setState({ copySuccess: true }); - setTimeout(() => this.setState({ copySuccess: false }), 2000); - } catch (err) { - console.error('Failed to copy report:', err); - } - }; - - sendToDiscord = async (): Promise => { - if (!DISCORD_WEBHOOK_URL) { - console.warn('Discord webhook not configured'); - return false; - } - - const { error, errorInfo } = this.state; - const timestamp = new Date().toISOString(); - - // Truncate stack trace if too long for Discord embed - const stackTrace = error?.stack || 'No stack trace'; - const truncatedStack = - stackTrace.length > 1000 ? stackTrace.substring(0, 1000) + '\n... (truncated)' : stackTrace; - - const payload = { - embeds: [ - { - title: '🚨 ClaraVerse Error Report', - color: 15158332, // Red color - fields: [ - { - name: '❌ Error', - value: `\`\`\`${error?.message || 'Unknown error'}\`\`\``, - inline: false, - }, - { - name: '🌐 URL', - value: window.location.href, - inline: true, - }, - { - name: '📅 Time', - value: timestamp, - inline: true, - }, - { - name: '📱 Browser', - value: navigator.userAgent.substring(0, 100) + '...', - inline: false, - }, - ], - description: `**Stack Trace:**\n\`\`\`\n${truncatedStack}\n\`\`\``, - footer: { - text: `Component: ${errorInfo?.componentStack?.split('\n')[1]?.trim() || 'Unknown'}`, - }, - }, - ], - }; - - try { - const response = await fetch(DISCORD_WEBHOOK_URL, { - method: 'POST', - headers: { 'Content-Type': 'application/json' }, - body: JSON.stringify(payload), - }); - - return response.ok; - } catch (err) { - console.error('Failed to send to Discord:', err); - return false; - } - }; - - handleRestartAndSend = async (): Promise => { - const { sendDiagnostics } = this.state; - - this.setState({ sendingReport: true, reportError: null }); - - // Send diagnostics if checkbox is checked - if (sendDiagnostics && DISCORD_WEBHOOK_URL) { - await this.sendToDiscord(); - } - - // Restart to chat page - window.location.href = '/chat'; - }; - - toggleDetails = (): void => { - this.setState(prev => ({ showDetails: !prev.showDetails })); - }; - - toggleSendDiagnostics = (): void => { - this.setState(prev => ({ sendDiagnostics: !prev.sendDiagnostics })); - }; - - render(): ReactNode { - const { hasError, error, errorInfo, showDetails, copySuccess, sendingReport, sendDiagnostics } = - this.state; - const { children } = this.props; - - if (!hasError) { - return children; - } - - return ( -
-
- {/* Mascot */} - Clara looking apologetic - - {/* Error Message */} -

Oops! Something went wrong

-

- Clara ran into an unexpected error. You can help us fix this by sending an error report. -

- - {/* Diagnostics Checkbox */} - - - {/* Privacy Note */} -

- Error reports are sent to ClaraVerse's Discord and contain only technical error - information (error message, stack trace). No personal data or chat content is included. -

- - {/* Primary Action - Restart & Send */} - - - {/* Secondary Action - Copy Report */} - - - {/* Technical Details Toggle */} - - - {/* Technical Details */} - {showDetails && ( -
-
-

Error Message

-
{error?.message || 'Unknown error'}
-
-
-

Stack Trace

-
{error?.stack || 'No stack trace available'}
-
- {errorInfo?.componentStack && ( -
-

Component Stack

-
{errorInfo.componentStack}
-
- )} -
- )} -
-
- ); - } -} - -export default ErrorBoundary; diff --git a/frontend/src/components/ErrorBoundary/index.ts b/frontend/src/components/ErrorBoundary/index.ts deleted file mode 100644 index e5d6dda2..00000000 --- a/frontend/src/components/ErrorBoundary/index.ts +++ /dev/null @@ -1 +0,0 @@ -export { default as ErrorBoundary } from './ErrorBoundary'; diff --git a/frontend/src/components/admin/AdminLayout.tsx b/frontend/src/components/admin/AdminLayout.tsx deleted file mode 100644 index e02d9dca..00000000 --- a/frontend/src/components/admin/AdminLayout.tsx +++ /dev/null @@ -1,165 +0,0 @@ -import { Outlet, NavLink, useNavigate } from 'react-router-dom'; -import { useState } from 'react'; -import { useAuthStore } from '@/store/useAuthStore'; -import { useOnboardingTour } from './OnboardingTour'; -import faviconIcon from '@/assets/favicon-32x32.png'; -import { - LayoutDashboard, - Plug, - BarChart3, - Box, - Users, - ChevronLeft, - ChevronRight, - LogOut, - ArrowLeft, -} from 'lucide-react'; - -export const AdminLayout = () => { - const [isSidebarOpen, setIsSidebarOpen] = useState(true); - const { user, signOut } = useAuthStore(); - const navigate = useNavigate(); - useOnboardingTour(); - - const handleSignOut = async () => { - await signOut(); - navigate('/signin'); - }; - - const navItems = [ - { path: '/admin/dashboard', label: 'Dashboard', icon: LayoutDashboard }, - { path: '/admin/providers', label: 'Providers', icon: Plug }, - { path: '/admin/models', label: 'Models', icon: Box }, - { path: '/admin/analytics', label: 'Analytics', icon: BarChart3 }, - { path: '/admin/users', label: 'Users', icon: Users }, - ]; - - return ( -
- {/* Sidebar */} - - - {/* Main Content */} -
- {/* Top Header */} -
-
-
- Admin -
-
- - -
- - {/* Page Content */} -
- -
-
-
- ); -}; diff --git a/frontend/src/components/admin/OnboardingTour.tsx b/frontend/src/components/admin/OnboardingTour.tsx deleted file mode 100644 index 9c421669..00000000 --- a/frontend/src/components/admin/OnboardingTour.tsx +++ /dev/null @@ -1,127 +0,0 @@ -import { useEffect } from 'react'; -import { useLocation } from 'react-router-dom'; -import { driver } from 'driver.js'; -import 'driver.js/dist/driver.css'; - -const PROVIDERS_TOUR_KEY = 'admin_providers_tour_completed'; -const MODELS_TOUR_KEY = 'admin_models_tour_completed'; - -export const useOnboardingTour = () => { - const location = useLocation(); - - useEffect(() => { - // Providers page tour - independent - const hasCompletedProvidersTour = localStorage.getItem(PROVIDERS_TOUR_KEY); - if (!hasCompletedProvidersTour && location.pathname.includes('/admin/providers')) { - const timer = setTimeout(() => { - const driverObj = driver({ - showProgress: true, - steps: [ - { - element: '[data-tour="add-provider"]', - popover: { - title: 'Add Your First Provider', - description: 'Click here to add an AI provider like OpenAI, Anthropic, or Google. You\'ll need to provide an API key.', - side: 'bottom', - align: 'start', - }, - }, - { - popover: { - title: 'After Adding a Provider', - description: 'Once you add a provider, navigate to the Models page to fetch and enable models.', - }, - }, - { - element: '[data-tour="models-link"]', - popover: { - title: 'Models Page', - description: 'After adding a provider, click here to manage models. You\'ll be able to fetch models from your providers and toggle their visibility.', - side: 'right', - align: 'start', - }, - }, - ], - onDestroyStarted: () => { - localStorage.setItem(PROVIDERS_TOUR_KEY, 'true'); - driverObj.destroy(); - }, - }); - - driverObj.drive(); - }, 500); - - return () => clearTimeout(timer); - } - - // Models page tour - independent - const hasCompletedModelsTour = localStorage.getItem(MODELS_TOUR_KEY); - if (!hasCompletedModelsTour && location.pathname.includes('/admin/models')) { - const timer = setTimeout(() => { - const driverObj = driver({ - showProgress: true, - steps: [ - { - popover: { - title: 'Welcome to Models Management', - description: 'Here you can manage all your AI models. Let\'s walk through the key features.', - }, - }, - { - element: '[data-tour="toggle-visibility"]', - popover: { - title: 'Toggle Model Visibility', - description: 'Click the Visible/Hidden button to control which models users can see and use. Only visible models will appear in the chat interface.', - side: 'left', - align: 'start', - }, - }, - { - popover: { - title: 'Expand Model Details', - description: 'Click on any model row (the chevron icon or model name) to expand and see detailed settings including capabilities and aliases.', - }, - }, - { - element: '[data-tour="model-capabilities"]', - popover: { - title: 'Model Capabilities Explained', - description: '🔧 **Tools**: Enable function calling - lets the model use external APIs and functions.\n\n👁️ **Vision**: Allows the model to understand and analyze images.\n\n📡 **Streaming**: Enables real-time response streaming for better UX.\n\n⚡ **Smart Router**: Advanced routing for optimized model selection.\n\nClick each to toggle on/off.', - side: 'top', - align: 'start', - }, - }, - { - element: '[data-tour="model-aliases"]', - popover: { - title: 'Model Aliases', - description: 'Aliases let you create alternative configurations for the same model. For example, you can have one alias with vision enabled and another without. This allows fine-grained control over model capabilities for different use cases.', - side: 'top', - align: 'start', - }, - }, - { - popover: { - title: 'You\'re All Set!', - description: 'You can now start using your configured models. Remember to enable visibility for models you want users to access, and configure their capabilities based on your needs.', - }, - }, - ], - onDestroyStarted: () => { - localStorage.setItem(MODELS_TOUR_KEY, 'true'); - driverObj.destroy(); - }, - }); - - driverObj.drive(); - }, 800); - - return () => clearTimeout(timer); - } - }, [location.pathname]); -}; - -export const resetOnboarding = () => { - localStorage.removeItem(PROVIDERS_TOUR_KEY); - localStorage.removeItem(MODELS_TOUR_KEY); -}; diff --git a/frontend/src/components/admin/ProviderForm.tsx b/frontend/src/components/admin/ProviderForm.tsx deleted file mode 100644 index 6b53a62f..00000000 --- a/frontend/src/components/admin/ProviderForm.tsx +++ /dev/null @@ -1,398 +0,0 @@ -import { useState, useEffect } from 'react'; -import { Modal } from '@/components/design-system/feedback/Modal/Modal'; -import { Eye, EyeOff } from 'lucide-react'; -import type { ProviderConfig, CreateProviderRequest } from '@/types/admin'; - -export interface ProviderFormProps { - isOpen: boolean; - onClose: () => void; - onSave: (data: CreateProviderRequest) => Promise; - provider?: ProviderConfig | null; - mode?: 'create' | 'edit'; -} - -export const ProviderForm: React.FC = ({ - isOpen, - onClose, - onSave, - provider = null, - mode = 'create', -}) => { - const [formData, setFormData] = useState({ - name: '', - base_url: '', - api_key: '', - enabled: true, - audio_only: false, - image_only: false, - image_edit_only: false, - secure: false, - default_model: '', - system_prompt: '', - favicon: '', - }); - - const [showApiKey, setShowApiKey] = useState(false); - const [errors, setErrors] = useState>({}); - const [isSubmitting, setIsSubmitting] = useState(false); - - useEffect(() => { - if (isOpen) { - if (provider) { - setFormData({ - name: provider.name, - base_url: provider.base_url, - api_key: provider.api_key, - enabled: provider.enabled, - audio_only: provider.audio_only || false, - image_only: provider.image_only || false, - image_edit_only: provider.image_edit_only || false, - secure: provider.secure || false, - default_model: provider.default_model || '', - system_prompt: provider.system_prompt || '', - favicon: provider.favicon || '', - }); - } else { - // Reset form for create mode - setFormData({ - name: '', - base_url: '', - api_key: '', - enabled: true, - audio_only: false, - image_only: false, - image_edit_only: false, - secure: false, - default_model: '', - system_prompt: '', - favicon: '', - }); - } - setErrors({}); - setShowApiKey(false); - } - }, [isOpen, provider]); - - const validateForm = (): boolean => { - const newErrors: Record = {}; - - if (!formData.name.trim()) { - newErrors.name = 'Provider name is required'; - } else if (formData.name.length > 255) { - newErrors.name = 'Provider name must be less than 255 characters'; - } - - if (!formData.base_url.trim()) { - newErrors.base_url = 'Base URL is required'; - } else { - try { - new URL(formData.base_url); - } catch { - newErrors.base_url = 'Must be a valid URL'; - } - } - - if (!formData.api_key.trim()) { - newErrors.api_key = 'API key is required'; - } - - // Validate that only one special type is selected - const specialTypes = [formData.audio_only, formData.image_only, formData.image_edit_only]; - const selectedCount = specialTypes.filter(Boolean).length; - if (selectedCount > 1) { - newErrors.special_type = 'Only one special type can be selected'; - } - - setErrors(newErrors); - return Object.keys(newErrors).length === 0; - }; - - const handleSubmit = async (e: React.FormEvent) => { - e.preventDefault(); - - if (!validateForm()) { - return; - } - - setIsSubmitting(true); - try { - await onSave(formData); - onClose(); - } catch (error) { - console.error('Failed to save provider:', error); - setErrors({ submit: 'Failed to save provider. Please try again.' }); - } finally { - setIsSubmitting(false); - } - }; - - const handleSpecialTypeChange = (type: 'audio_only' | 'image_only' | 'image_edit_only', checked: boolean) => { - setFormData(prev => ({ - ...prev, - audio_only: type === 'audio_only' ? checked : false, - image_only: type === 'image_only' ? checked : false, - image_edit_only: type === 'image_edit_only' ? checked : false, - })); - }; - - return ( - -
- {/* Basic Information */} -
-

- Basic Information -

- -
- - setFormData({ ...formData, name: e.target.value })} - className={`w-full px-3 py-2 bg-[var(--color-surface)] rounded-lg text-[var(--color-text-primary)] placeholder-[var(--color-text-tertiary)] focus:outline-none focus:ring-2 focus:ring-[var(--color-accent)] ${ - errors.name ? 'border border-[var(--color-error)]' : '' - }`} - placeholder="e.g., OpenAI, Anthropic, Custom Provider" - disabled={isSubmitting} - /> - {errors.name &&

{errors.name}

} -
- -
- - setFormData({ ...formData, base_url: e.target.value })} - className={`w-full px-3 py-2 bg-[var(--color-surface)] rounded-lg text-[var(--color-text-primary)] placeholder-[var(--color-text-tertiary)] focus:outline-none focus:ring-2 focus:ring-[var(--color-accent)] ${ - errors.base_url ? 'border border-[var(--color-error)]' : '' - }`} - placeholder="https://api.provider.com/v1" - disabled={isSubmitting} - /> - {errors.base_url &&

{errors.base_url}

} -
- -
- -
- setFormData({ ...formData, api_key: e.target.value })} - className={`w-full px-3 py-2 pr-10 bg-[var(--color-surface)] rounded-lg text-[var(--color-text-primary)] placeholder-[var(--color-text-tertiary)] focus:outline-none focus:ring-2 focus:ring-[var(--color-accent)] ${ - errors.api_key ? 'border border-[var(--color-error)]' : '' - }`} - placeholder="sk-..." - disabled={isSubmitting} - /> - -
- {errors.api_key &&

{errors.api_key}

} -
-
- - {/* Special Provider Types */} -
-

- Special Provider Type -

-

- Select if this provider handles a specific type of content (only one can be selected) -

- -
- - - - - -
- {errors.special_type &&

{errors.special_type}

} -
- - {/* Security & Settings */} -
-

- Security & Settings -

- - - - -
- - {/* Optional Metadata */} -
-

- Optional Metadata -

- -
- - setFormData({ ...formData, default_model: e.target.value })} - className="w-full px-3 py-2 bg-[var(--color-surface)] rounded-lg text-[var(--color-text-primary)] placeholder-[var(--color-text-tertiary)] focus:outline-none focus:ring-2 focus:ring-[var(--color-accent)]" - placeholder="gpt-4, claude-3-opus, etc." - disabled={isSubmitting} - /> -
- -
- - setFormData({ ...formData, favicon: e.target.value })} - className="w-full px-3 py-2 bg-[var(--color-surface)] rounded-lg text-[var(--color-text-primary)] placeholder-[var(--color-text-tertiary)] focus:outline-none focus:ring-2 focus:ring-[var(--color-accent)]" - placeholder="https://example.com/icon.png" - disabled={isSubmitting} - /> -
- -
- - +

Enter JSON input for the workflow

+
+ + + + + +
+
+

Response

+ +
+ +
+
+ Duration: +
+
+
Output:
+

+                                
+
+ +
+
Error Message:
+

+
+
+
+ + +
+
+

Interactive API Documentation

+

+ View the full OpenAPI/Swagger documentation for this workflow with interactive testing capabilities. +

+ + + + + Open Swagger Documentation + +
+ +
+

Quick Reference

+ + +
+
+

cURL Command

+ +
+

+                            
+ + +
+
+

Python (requests)

+ +
+

+                            
+ + +
+
+

JavaScript (fetch)

+ +
+

+                            
+
+
+ + + + + + + + + + + diff --git a/sdk/server/schema-generator.js b/sdk/server/schema-generator.js new file mode 100644 index 00000000..53f2fe79 --- /dev/null +++ b/sdk/server/schema-generator.js @@ -0,0 +1,485 @@ +/** + * ClaraVerse Agent Runner - Schema Generator + * Auto-generates OpenAPI 3.0 schemas from workflow input/output nodes + */ + +/** + * Node type to JSON schema type mapping + */ +const NODE_TYPE_MAPPINGS = { + // Input nodes + 'input': { + text: { type: 'string', description: 'Text input' }, + number: { type: 'number', description: 'Number input' }, + json: { type: 'object', description: 'JSON object input' }, + }, + 'image-input': { + type: 'string', + format: 'base64', + description: 'Base64 encoded image data (supports png, jpg, jpeg, gif, webp)', + example: '...', + }, + 'pdf-input': { + type: 'string', + format: 'binary', + description: 'PDF file content or base64 encoded PDF', + }, + 'file-upload': { + type: 'string', + format: 'binary', + description: 'File upload content (base64 encoded)', + }, + // Output nodes + 'output': { + type: 'string', + description: 'Workflow output', + }, +}; + +/** + * Extract input schema from workflow nodes + * @param {Object} workflow - Workflow JSON + * @returns {Object} - Input schema definition + */ +export function extractInputSchema(workflow) { + const inputs = {}; + const required = []; + + // Find all input-type nodes + const inputNodes = workflow.nodes.filter(node => + ['input', 'image-input', 'pdf-input', 'file-upload'].includes(node.type) + ); + + for (const node of inputNodes) { + const nodeType = node.type; + const nodeData = node.data || {}; + + // Generate field name from node label or ID + const fieldName = generateFieldName(nodeData.label || nodeData.inputLabel || node.id); + + // Get schema for this node type + let schema; + if (nodeType === 'input') { + const inputType = nodeData.inputType || 'text'; + schema = NODE_TYPE_MAPPINGS['input'][inputType] || NODE_TYPE_MAPPINGS['input'].text; + } else { + schema = NODE_TYPE_MAPPINGS[nodeType]; + } + + // Add to inputs + inputs[fieldName] = { + ...schema, + 'x-node-id': node.id, + 'x-node-type': nodeType, + }; + + // Mark as required if node has required flag or default behavior + const isRequired = nodeData.required !== false; // Default to required + if (isRequired) { + required.push(fieldName); + } + + // Add default value if present + if (nodeData.value !== undefined && nodeData.value !== null && nodeData.value !== '') { + inputs[fieldName].default = nodeData.value; + } + + // Add custom description if present + if (nodeData.description) { + inputs[fieldName].description = nodeData.description; + } + } + + return { + type: 'object', + properties: inputs, + required: required.length > 0 ? required : undefined, + }; +} + +/** + * Extract output schema from workflow nodes + * @param {Object} workflow - Workflow JSON + * @returns {Object} - Output schema definition + */ +export function extractOutputSchema(workflow) { + const outputs = {}; + + // Find all output-type nodes + const outputNodes = workflow.nodes.filter(node => node.type === 'output'); + + if (outputNodes.length === 0) { + // Default output if no output nodes + return { + type: 'object', + properties: { + result: { + type: 'string', + description: 'Workflow execution result', + }, + }, + }; + } + + for (const node of outputNodes) { + const nodeData = node.data || {}; + const fieldName = generateFieldName(nodeData.outputLabel || node.id); + + outputs[fieldName] = { + type: 'string', + description: nodeData.description || 'Output value', + 'x-node-id': node.id, + 'x-node-type': 'output', + }; + + // Detect if output format hints at specific type + const format = nodeData.format; + if (format === 'json') { + outputs[fieldName].type = 'object'; + outputs[fieldName].description += ' (JSON format)'; + } + } + + return { + type: 'object', + properties: outputs, + }; +} + +/** + * Generate OpenAPI 3.0 schema for workflow + * @param {Object} workflow - Workflow JSON + * @param {string} slug - Workflow slug for URL + * @param {string} baseUrl - Base URL of the API + * @returns {Object} - OpenAPI 3.0 schema + */ +export function generateOpenAPISchema(workflow, slug, baseUrl) { + const inputSchema = extractInputSchema(workflow); + const outputSchema = extractOutputSchema(workflow); + + const openApiSchema = { + openapi: '3.0.0', + info: { + title: workflow.name || 'Clara Workflow', + description: workflow.description || 'Deployed Clara workflow API', + version: workflow.version || '1.0.0', + }, + servers: [ + { + url: baseUrl, + description: 'Clara Agent Runner API', + }, + ], + paths: { + [`/api/workflows/${slug}/execute`]: { + post: { + summary: `Execute ${workflow.name || 'workflow'}`, + description: workflow.description || 'Execute the deployed workflow with provided inputs', + operationId: `execute_${slug}`, + tags: ['Workflow Execution'], + security: [ + { + ApiKeyAuth: [], + }, + ], + requestBody: { + required: true, + content: { + 'application/json': { + schema: inputSchema, + examples: { + default: { + summary: 'Example input', + value: generateExampleInputs(inputSchema), + }, + }, + }, + }, + }, + responses: { + '200': { + description: 'Successful execution', + content: { + 'application/json': { + schema: { + type: 'object', + properties: { + success: { + type: 'boolean', + example: true, + }, + outputs: outputSchema, + metadata: { + type: 'object', + properties: { + executionId: { + type: 'string', + format: 'uuid', + }, + duration: { + type: 'string', + example: '1234ms', + }, + timestamp: { + type: 'string', + format: 'date-time', + }, + }, + }, + }, + }, + }, + }, + }, + '400': { + description: 'Invalid input', + content: { + 'application/json': { + schema: { + $ref: '#/components/schemas/Error', + }, + }, + }, + }, + '401': { + description: 'Invalid or missing API key', + content: { + 'application/json': { + schema: { + $ref: '#/components/schemas/Error', + }, + }, + }, + }, + '500': { + description: 'Execution error', + content: { + 'application/json': { + schema: { + $ref: '#/components/schemas/Error', + }, + }, + }, + }, + }, + }, + }, + }, + components: { + securitySchemes: { + ApiKeyAuth: { + type: 'apiKey', + in: 'header', + name: 'Authorization', + description: 'API key in format: Bearer clara_sk_...', + }, + }, + schemas: { + Error: { + type: 'object', + properties: { + success: { + type: 'boolean', + example: false, + }, + error: { + type: 'string', + description: 'Error message', + }, + timestamp: { + type: 'string', + format: 'date-time', + }, + }, + }, + }, + }, + }; + + return openApiSchema; +} + +/** + * Generate a simple JSON schema for input validation (lighter than OpenAPI) + * @param {Object} workflow - Workflow JSON + * @returns {Object} - JSON schema + */ +export function generateSimpleSchema(workflow) { + const inputSchema = extractInputSchema(workflow); + const outputSchema = extractOutputSchema(workflow); + + return { + input: inputSchema, + output: outputSchema, + metadata: { + nodeCount: workflow.nodes.length, + connectionCount: workflow.connections?.length || 0, + hasCustomNodes: workflow.customNodes && workflow.customNodes.length > 0, + }, + }; +} + +/** + * Map workflow inputs to node IDs for execution + * Creates a mapping from API field names to internal node IDs + * @param {Object} workflow - Workflow JSON + * @returns {Object} - Map of fieldName -> nodeId + */ +export function generateInputMapping(workflow) { + const mapping = {}; + + const inputNodes = workflow.nodes.filter(node => + ['input', 'image-input', 'pdf-input', 'file-upload'].includes(node.type) + ); + + for (const node of inputNodes) { + const nodeData = node.data || {}; + const fieldName = generateFieldName(nodeData.label || nodeData.inputLabel || node.id); + mapping[fieldName] = node.id; + } + + return mapping; +} + +/** + * Map node IDs to output field names for response formatting + * @param {Object} workflow - Workflow JSON + * @returns {Object} - Map of nodeId -> fieldName + */ +export function generateOutputMapping(workflow) { + const mapping = {}; + + const outputNodes = workflow.nodes.filter(node => node.type === 'output'); + + for (const node of outputNodes) { + const nodeData = node.data || {}; + const fieldName = generateFieldName(nodeData.outputLabel || node.id); + mapping[node.id] = fieldName; + } + + return mapping; +} + +/** + * Generate example inputs from schema + * @param {Object} schema - Input schema + * @returns {Object} - Example input values + */ +function generateExampleInputs(schema) { + const example = {}; + + if (schema.properties) { + for (const [key, prop] of Object.entries(schema.properties)) { + if (prop.example !== undefined) { + example[key] = prop.example; + } else if (prop.default !== undefined) { + example[key] = prop.default; + } else { + // Generate example based on type + switch (prop.type) { + case 'string': + if (prop.format === 'base64') { + example[key] = ''; + } else { + example[key] = 'Example text'; + } + break; + case 'number': + example[key] = 42; + break; + case 'boolean': + example[key] = true; + break; + case 'object': + example[key] = { key: 'value' }; + break; + case 'array': + example[key] = ['item1', 'item2']; + break; + default: + example[key] = null; + } + } + } + } + + return example; +} + +/** + * Generate API-friendly field name from node label + * Converts to camelCase and removes special characters + * @param {string} label - Node label or ID + * @returns {string} - Sanitized field name + */ +function generateFieldName(label) { + return label + .trim() + .replace(/[^\w\s-]/g, '') // Remove special chars except word chars, spaces, hyphens + .replace(/\s+(.)/g, (_, char) => char.toUpperCase()) // Convert to camelCase + .replace(/^[A-Z]/, char => char.toLowerCase()) // Ensure first char is lowercase + .replace(/[^a-zA-Z0-9]/g, '') // Remove any remaining special chars + || 'input'; +} + +/** + * Validate inputs against schema + * @param {Object} inputs - User-provided inputs + * @param {Object} schema - Input schema + * @returns {Object} - { valid: boolean, errors: string[] } + */ +export function validateInputs(inputs, schema) { + const errors = []; + + // Check required fields + if (schema.required) { + for (const field of schema.required) { + if (inputs[field] === undefined || inputs[field] === null || inputs[field] === '') { + errors.push(`Missing required field: ${field}`); + } + } + } + + // Check field types + if (schema.properties) { + for (const [field, value] of Object.entries(inputs)) { + const fieldSchema = schema.properties[field]; + + if (!fieldSchema) { + errors.push(`Unknown field: ${field}`); + continue; + } + + // Type validation + const actualType = typeof value; + const expectedType = fieldSchema.type; + + if (expectedType === 'number' && actualType !== 'number') { + errors.push(`Field '${field}' must be a number`); + } else if (expectedType === 'string' && actualType !== 'string') { + errors.push(`Field '${field}' must be a string`); + } else if (expectedType === 'boolean' && actualType !== 'boolean') { + errors.push(`Field '${field}' must be a boolean`); + } else if (expectedType === 'object' && (actualType !== 'object' || value === null)) { + errors.push(`Field '${field}' must be an object`); + } else if (expectedType === 'array' && !Array.isArray(value)) { + errors.push(`Field '${field}' must be an array`); + } + } + } + + return { + valid: errors.length === 0, + errors, + }; +} + +export default { + extractInputSchema, + extractOutputSchema, + generateOpenAPISchema, + generateSimpleSchema, + generateInputMapping, + generateOutputMapping, + validateInputs, +}; diff --git a/sdk/server/server.js b/sdk/server/server.js new file mode 100644 index 00000000..2f2e7f68 --- /dev/null +++ b/sdk/server/server.js @@ -0,0 +1,751 @@ +/** + * ClaraVerse Agent Runner - Main Server + * REST API for deploying and executing agent workflows + */ + +import express from 'express'; +import cors from 'cors'; +import rateLimit from 'express-rate-limit'; +import path from 'path'; +import { fileURLToPath } from 'url'; +import config, { validateConfig, getConfigSummary } from './config.js'; +import * as db from './database.js'; +import { generateOpenAPISchema, generateSimpleSchema } from './schema-generator.js'; +import { createExecutor } from './workflow-executor.js'; +import { authenticateApiKey, optionalAuth } from './middleware/auth.js'; +import { + validateDeployRequest, + validateExecutionInputs, + validateBodySize, + sanitizeInputs, + validatePagination, + validateUUID, + validateSlugFormat, +} from './middleware/validation.js'; + +// Get directory name for ES modules +const __filename = fileURLToPath(import.meta.url); +const __dirname = path.dirname(__filename); + +// Validate configuration on startup +try { + validateConfig(); + console.log('✅ Configuration validated successfully'); +} catch (error) { + console.error('❌ Configuration validation failed:', error.message); + process.exit(1); +} + +// Initialize Express app +const app = express(); + +// Trust proxy if configured (for rate limiting behind reverse proxy) +if (config.security.trustProxy) { + app.set('trust proxy', 1); +} + +// ==================== +// Middleware +// ==================== + +// CORS +app.use(cors({ + origin: config.security.corsOrigins, + credentials: true, +})); + +// Body parsing +app.use(express.json({ limit: '10mb' })); +app.use(express.urlencoded({ extended: true, limit: '10mb' })); + +// Serve static files from public directory +app.use(express.static(path.join(__dirname, 'public'))); + +// Rate limiting (if enabled) +if (config.rateLimit.enabled) { + const limiter = rateLimit({ + windowMs: config.rateLimit.windowMs, + max: config.rateLimit.maxRequests, + message: { + success: false, + error: 'Too many requests', + message: `Rate limit exceeded. Please try again later.`, + }, + standardHeaders: true, + legacyHeaders: false, + }); + + app.use('/api/workflows/:slug/execute', limiter); + console.log(`🚦 Rate limiting enabled: ${config.rateLimit.maxRequests} requests per ${config.rateLimit.windowMs}ms`); +} + +// Request logging +app.use((req, res, next) => { + const start = Date.now(); + res.on('finish', () => { + const duration = Date.now() - start; + console.log(`${req.method} ${req.path} ${res.statusCode} - ${duration}ms`); + }); + next(); +}); + +// ==================== +// Health & Info Endpoints +// ==================== + +/** + * Health check endpoint + */ +app.get('/health', async (req, res) => { + const dbStatus = await db.testConnection(); + + res.json({ + status: dbStatus.success ? 'healthy' : 'degraded', + service: 'Clara Agent Runner', + version: '1.0.0', + timestamp: new Date().toISOString(), + uptime: `${Math.floor(process.uptime())}s`, + database: dbStatus.success ? 'connected' : 'disconnected', + config: getConfigSummary(), + }); +}); + +/** + * Dashboard - Web UI for managing workflows + */ +app.get(['/', '/dashboard'], (req, res) => { + res.sendFile(path.join(__dirname, 'public', 'dashboard.html')); +}); + +/** + * Service information endpoint + */ +app.get('/api/info', (req, res) => { + res.json({ + service: 'Clara Agent Runner API', + version: '1.0.0', + description: 'Deploy and execute Clara workflows as REST APIs', + features: [ + 'Workflow deployment with auto-generated schemas', + 'Dynamic API endpoint creation', + 'OpenAPI 3.0 documentation', + 'Input/output validation', + 'Execution tracking and logging', + 'Rate limiting and authentication', + ], + endpoints: { + 'POST /api/deploy': 'Deploy a new workflow', + 'GET /api/deployments': 'List all deployed workflows', + 'GET /api/workflows/:slug': 'Get workflow details', + 'GET /api/workflows/:slug/schema': 'Get OpenAPI schema', + 'GET /api/workflows/:slug/docs': 'Interactive API documentation', + 'POST /api/workflows/:slug/execute': 'Execute workflow', + 'GET /api/workflows/:slug/executions': 'Get execution history', + 'DELETE /api/workflows/:id': 'Delete (deactivate) workflow', + 'POST /api/workflows/:id/regenerate-key': 'Regenerate API key', + 'GET /health': 'Health check', + 'GET /api/info': 'Service information', + }, + documentation: `${config.server.baseUrl}/api/docs`, + }); +}); + +// ==================== +// Deployment Endpoints +// ==================== + +/** + * Deploy a new workflow + * Creates a deployed workflow and returns endpoint URL + API key + */ +app.post('/api/deploy', + validateBodySize, + validateDeployRequest, + async (req, res) => { + try { + const { workflow, name, slug, description, userId } = req.body; + + // Generate slug if not provided + const finalSlug = slug || await db.generateUniqueSlug(name); + + // Check if slug is available + if (!(await db.isSlugAvailable(finalSlug))) { + return res.status(400).json({ + success: false, + error: 'Slug already in use', + message: `The slug '${finalSlug}' is already taken. Please choose a different name or slug.`, + timestamp: new Date().toISOString(), + }); + } + + // Generate schemas + const simpleSchema = generateSimpleSchema(workflow); + const openApiSchema = generateOpenAPISchema(workflow, finalSlug, config.server.baseUrl); + + // Create deployed workflow in database + const deployed = await db.createDeployedWorkflow({ + name, + slug: finalSlug, + description, + workflowJson: workflow, + schemaJson: { + ...simpleSchema, + openapi: openApiSchema, + }, + userId, + }); + + res.status(201).json({ + success: true, + deployment: { + id: deployed.id, + name: deployed.name, + slug: deployed.slug, + description: deployed.description, + endpoint: `${config.server.baseUrl}/api/workflows/${deployed.slug}/execute`, + apiKey: deployed.apiKey, // Only shown once! + schema: simpleSchema, + docs: `${config.server.baseUrl}/api/workflows/${deployed.slug}/docs`, + createdAt: deployed.created_at, + }, + message: 'Workflow deployed successfully. Save the API key - it will not be shown again!', + }); + } catch (error) { + console.error('❌ Deployment error:', error); + res.status(500).json({ + success: false, + error: 'Deployment failed', + message: error.message, + timestamp: new Date().toISOString(), + }); + } + } +); + +/** + * List all deployed workflows + */ +app.get('/api/deployments', + validatePagination, + async (req, res) => { + try { + const { userId } = req.query; + const limit = req.query.limit || 100; + const offset = req.query.offset || 0; + + const deployments = await db.getAllDeployedWorkflows(userId, { limit, offset }); + + res.json({ + success: true, + deployments: deployments.map(d => ({ + id: d.id, + name: d.name, + slug: d.slug, + description: d.description, + endpoint: `${config.server.baseUrl}/api/workflows/${d.slug}/execute`, + executionCount: d.execution_count, + lastExecuted: d.last_executed_at, + createdAt: d.created_at, + isActive: d.is_active, + })), + pagination: { + limit, + offset, + hasMore: deployments.length === limit, + }, + }); + } catch (error) { + console.error('❌ List deployments error:', error); + res.status(500).json({ + success: false, + error: 'Failed to list deployments', + message: error.message, + timestamp: new Date().toISOString(), + }); + } + } +); + +// ==================== +// Workflow Info Endpoints +// ==================== + +/** + * Get workflow details by slug + */ +app.get('/api/workflows/:slug', + validateSlugFormat('slug'), + async (req, res) => { + try { + const { slug } = req.params; + const workflow = await db.getDeployedWorkflowBySlug(slug); + + if (!workflow) { + return res.status(404).json({ + success: false, + error: 'Workflow not found', + timestamp: new Date().toISOString(), + }); + } + + res.json({ + success: true, + workflow: { + id: workflow.id, + name: workflow.name, + slug: workflow.slug, + description: workflow.description, + endpoint: `${config.server.baseUrl}/api/workflows/${workflow.slug}/execute`, + schema: workflow.schema_json, + executionCount: workflow.execution_count, + lastExecuted: workflow.last_executed_at, + createdAt: workflow.created_at, + updatedAt: workflow.updated_at, + }, + }); + } catch (error) { + console.error('❌ Get workflow error:', error); + res.status(500).json({ + success: false, + error: 'Failed to get workflow', + message: error.message, + timestamp: new Date().toISOString(), + }); + } + } +); + +/** + * Get OpenAPI schema for workflow + */ +app.get('/api/workflows/:slug/schema', + validateSlugFormat('slug'), + async (req, res) => { + try { + const { slug } = req.params; + const workflow = await db.getDeployedWorkflowBySlug(slug); + + if (!workflow) { + return res.status(404).json({ + success: false, + error: 'Workflow not found', + timestamp: new Date().toISOString(), + }); + } + + // Return OpenAPI schema + const openApiSchema = workflow.schema_json?.openapi || + generateOpenAPISchema(workflow.workflow_json, workflow.slug, config.server.baseUrl); + + res.json(openApiSchema); + } catch (error) { + console.error('❌ Get schema error:', error); + res.status(500).json({ + success: false, + error: 'Failed to get schema', + message: error.message, + timestamp: new Date().toISOString(), + }); + } + } +); + +/** + * Interactive API documentation (Swagger UI) + */ +app.get('/api/workflows/:slug/docs', + validateSlugFormat('slug'), + async (req, res) => { + try { + const { slug } = req.params; + const workflow = await db.getDeployedWorkflowBySlug(slug); + + if (!workflow) { + return res.status(404).send('

Workflow not found

'); + } + + const schemaUrl = `${config.server.baseUrl}/api/workflows/${slug}/schema`; + + // Simple Swagger UI HTML + const html = ` + + + + + ${workflow.name} - API Documentation + + + +
+ + + + + + `; + + res.send(html); + } catch (error) { + console.error('❌ Get docs error:', error); + res.status(500).send('

Error loading documentation

'); + } + } +); + +// ==================== +// Execution Endpoints +// ==================== + +/** + * Execute workflow + * Main endpoint for workflow execution with authentication + */ +app.post('/api/workflows/:slug/execute', + validateSlugFormat('slug'), + authenticateApiKey, + validateBodySize, + sanitizeInputs, + validateExecutionInputs, + async (req, res) => { + const executionId = null; + const startTime = Date.now(); + + try { + const inputs = req.body; + const workflow = req.workflow; + + console.log(`📋 Executing workflow: ${workflow.name} (${workflow.slug})`); + + // Create execution record + const execution = await db.createWorkflowExecution({ + workflowId: workflow.id, + inputs, + status: 'running', + }); + + // Execute workflow + const executor = createExecutor({ + enableLogging: config.execution.enableLogging, + timeout: config.execution.maxExecutionTime, + }); + + const result = await executor.execute(workflow, inputs); + + const duration = Date.now() - startTime; + + // Update execution record + await db.updateWorkflowExecution(execution.id, { + outputs: result.outputs, + status: result.success ? 'success' : 'error', + errorMessage: result.error || null, + durationMs: duration, + }); + + // Increment workflow execution count + await db.incrementExecutionCount(workflow.id); + + if (result.success) { + res.json({ + success: true, + outputs: result.outputs, + metadata: { + ...result.metadata, + executionId: execution.id, + workflow: { + id: workflow.id, + name: workflow.name, + slug: workflow.slug, + }, + }, + }); + } else { + res.status(500).json({ + success: false, + error: result.error, + logs: result.logs, + metadata: { + ...result.metadata, + executionId: execution.id, + }, + }); + } + } catch (error) { + const duration = Date.now() - startTime; + console.error('❌ Execution error:', error); + + // Update execution record if created + if (executionId) { + await db.updateWorkflowExecution(executionId, { + status: 'error', + errorMessage: error.message, + durationMs: duration, + }); + } + + res.status(500).json({ + success: false, + error: 'Execution failed', + message: error.message, + metadata: { + duration: `${duration}ms`, + timestamp: new Date().toISOString(), + }, + }); + } + } +); + +/** + * Get workflow execution history + */ +app.get('/api/workflows/:slug/executions', + validateSlugFormat('slug'), + authenticateApiKey, + validatePagination, + async (req, res) => { + try { + const workflow = req.workflow; + const limit = req.query.limit || 50; + const offset = req.query.offset || 0; + const status = req.query.status; + + const executions = await db.getWorkflowExecutions(workflow.id, { + limit, + offset, + status, + }); + + res.json({ + success: true, + executions: executions.map(e => ({ + id: e.id, + inputs: e.inputs, + outputs: e.outputs, + status: e.status, + error: e.error_message, + duration: `${e.duration_ms}ms`, + createdAt: e.created_at, + })), + pagination: { + limit, + offset, + hasMore: executions.length === limit, + }, + }); + } catch (error) { + console.error('❌ Get executions error:', error); + res.status(500).json({ + success: false, + error: 'Failed to get executions', + message: error.message, + timestamp: new Date().toISOString(), + }); + } + } +); + +// ==================== +// Management Endpoints +// ==================== + +/** + * Delete (deactivate) deployed workflow + */ +app.delete('/api/workflows/:id', + validateUUID('id'), + authenticateApiKey, + async (req, res) => { + try { + const { id } = req.params; + const workflow = req.workflow; + + // Verify ownership + if (workflow.id !== id) { + return res.status(403).json({ + success: false, + error: 'Access denied', + message: 'You do not have permission to delete this workflow', + timestamp: new Date().toISOString(), + }); + } + + const deleted = await db.deleteDeployedWorkflow(id); + + if (!deleted) { + return res.status(404).json({ + success: false, + error: 'Workflow not found', + timestamp: new Date().toISOString(), + }); + } + + res.json({ + success: true, + message: 'Workflow deleted successfully', + workflow: { + id: deleted.id, + slug: deleted.slug, + }, + }); + } catch (error) { + console.error('❌ Delete workflow error:', error); + res.status(500).json({ + success: false, + error: 'Failed to delete workflow', + message: error.message, + timestamp: new Date().toISOString(), + }); + } + } +); + +/** + * Regenerate API key for workflow + */ +app.post('/api/workflows/:id/regenerate-key', + validateUUID('id'), + authenticateApiKey, + async (req, res) => { + try { + const { id } = req.params; + const workflow = req.workflow; + + // Verify ownership + if (workflow.id !== id) { + return res.status(403).json({ + success: false, + error: 'Access denied', + message: 'You do not have permission to regenerate the API key for this workflow', + timestamp: new Date().toISOString(), + }); + } + + const result = await db.regenerateApiKey(id); + + if (!result) { + return res.status(404).json({ + success: false, + error: 'Workflow not found', + timestamp: new Date().toISOString(), + }); + } + + res.json({ + success: true, + message: 'API key regenerated successfully. Save it - it will not be shown again!', + workflow: { + id: result.id, + slug: result.slug, + apiKey: result.apiKey, + }, + }); + } catch (error) { + console.error('❌ Regenerate key error:', error); + res.status(500).json({ + success: false, + error: 'Failed to regenerate API key', + message: error.message, + timestamp: new Date().toISOString(), + }); + } + } +); + +// ==================== +// Error Handling +// ==================== + +// 404 handler +app.use((req, res) => { + res.status(404).json({ + success: false, + error: 'Endpoint not found', + message: `The endpoint ${req.method} ${req.path} does not exist`, + availableEndpoints: '/api/info', + timestamp: new Date().toISOString(), + }); +}); + +// Global error handler +app.use((error, req, res, next) => { + console.error('🚨 Unhandled error:', error); + res.status(500).json({ + success: false, + error: 'Internal server error', + message: config.server.env === 'development' ? error.message : 'An unexpected error occurred', + timestamp: new Date().toISOString(), + }); +}); + +// ==================== +// Server Startup +// ==================== + +/** + * Start server + */ +async function startServer() { + try { + // Test database connection + console.log('🔌 Connecting to database...'); + const dbStatus = await db.testConnection(); + if (!dbStatus.success) { + throw new Error(`Database connection failed: ${dbStatus.error}`); + } + console.log('✅ Database connected successfully'); + + // Start HTTP server + const server = app.listen(config.server.port, config.server.host, () => { + console.log(''); + console.log('🚀 Clara Agent Runner Server Started'); + console.log('═══════════════════════════════════════════════'); + console.log(`📍 Server URL: ${config.server.baseUrl}`); + console.log(`🏥 Health Check: ${config.server.baseUrl}/health`); + console.log(`📖 API Info: ${config.server.baseUrl}/api/info`); + console.log(`🌍 Environment: ${config.server.env}`); + console.log('═══════════════════════════════════════════════'); + console.log(''); + }); + + // Graceful shutdown + process.on('SIGTERM', async () => { + console.log('🛑 SIGTERM received, shutting down gracefully...'); + server.close(async () => { + await db.closePool(); + console.log('✅ Server shut down successfully'); + process.exit(0); + }); + }); + + process.on('SIGINT', async () => { + console.log('\n🛑 SIGINT received, shutting down gracefully...'); + server.close(async () => { + await db.closePool(); + console.log('✅ Server shut down successfully'); + process.exit(0); + }); + }); + } catch (error) { + console.error('❌ Failed to start server:', error.message); + process.exit(1); + } +} + +// Start server if run directly +if (import.meta.url === `file://${process.argv[1]}`) { + startServer(); +} + +export default app; diff --git a/sdk/server/workflow-executor.js b/sdk/server/workflow-executor.js new file mode 100644 index 00000000..a26bd4ac --- /dev/null +++ b/sdk/server/workflow-executor.js @@ -0,0 +1,354 @@ +/** + * ClaraVerse Agent Runner - Workflow Executor + * Handles workflow execution with ClaraFlowRunner and input/output mapping + */ + +import { ClaraFlowRunner } from '../dist/index.js'; +import { generateInputMapping, generateOutputMapping } from './schema-generator.js'; +import config from './config.js'; + +/** + * Workflow Executor Class + * Manages workflow execution with proper input mapping and error handling + */ +export class WorkflowExecutor { + constructor(options = {}) { + this.runner = new ClaraFlowRunner({ + enableLogging: options.enableLogging !== false, + timeout: options.timeout || config.execution.maxExecutionTime, + }); + + this.maxExecutionTime = options.timeout || config.execution.maxExecutionTime; + } + + /** + * Execute a deployed workflow + * @param {Object} workflow - Deployed workflow from database + * @param {Object} apiInputs - User-provided inputs (field names from API) + * @returns {Promise} - Execution result with outputs and metadata + */ + async execute(workflow, apiInputs) { + const startTime = Date.now(); + + try { + // Get the workflow JSON + const workflowJson = typeof workflow.workflow_json === 'string' + ? JSON.parse(workflow.workflow_json) + : workflow.workflow_json; + + // Generate input mapping (API field names -> node IDs) + const inputMapping = generateInputMapping(workflowJson); + const outputMapping = generateOutputMapping(workflowJson); + + // Map API inputs to node IDs + const nodeInputs = this.mapInputsToNodes(apiInputs, inputMapping); + + // Inject environment configuration for nodes that need external services + this.injectEnvironmentConfig(workflowJson); + + // Execute workflow with ClaraFlowRunner + const result = await Promise.race([ + this.runner.execute(workflowJson, nodeInputs), + this.createTimeoutPromise(this.maxExecutionTime), + ]); + + const duration = Date.now() - startTime; + + // Map outputs from node IDs to field names + // Note: SDK returns outputs directly, not wrapped in result.outputs + const outputs = this.mapOutputsToFields(result, outputMapping); + + return { + success: true, + outputs, + logs: [], + metadata: { + duration: `${duration}ms`, + durationMs: duration, + nodesExecuted: Object.keys(result || {}).length, + timestamp: new Date().toISOString(), + }, + }; + } catch (error) { + const duration = Date.now() - startTime; + + return { + success: false, + error: error.message, + logs: this.runner.getLogs ? this.runner.getLogs() : [], + metadata: { + duration: `${duration}ms`, + durationMs: duration, + timestamp: new Date().toISOString(), + }, + }; + } + } + + /** + * Map API inputs (with field names) to node IDs + * @param {Object} apiInputs - User-provided inputs + * @param {Object} inputMapping - Map of fieldName -> nodeId + * @returns {Object} - Inputs keyed by node ID + */ + mapInputsToNodes(apiInputs, inputMapping) { + const nodeInputs = {}; + + for (const [fieldName, value] of Object.entries(apiInputs)) { + const nodeId = inputMapping[fieldName]; + if (nodeId) { + nodeInputs[nodeId] = value; + } + } + + return nodeInputs; + } + + /** + * Map outputs from node IDs to field names + * @param {Object} nodeOutputs - Outputs from ClaraFlowRunner (keyed by node ID) + * @param {Object} outputMapping - Map of nodeId -> fieldName + * @returns {Object} - Outputs with friendly field names + */ + mapOutputsToFields(nodeOutputs, outputMapping) { + const outputs = {}; + + if (!nodeOutputs || typeof nodeOutputs !== 'object') { + return outputs; + } + + // Map output nodes + for (const [nodeId, value] of Object.entries(nodeOutputs)) { + const fieldName = outputMapping[nodeId]; + if (fieldName) { + // Extract actual output value from output node structure + outputs[fieldName] = this.extractOutputValue(value); + } + } + + // If no mapped outputs, try to find any output nodes + if (Object.keys(outputs).length === 0 && Object.keys(nodeOutputs).length > 0) { + // Fallback: return first output found + const firstOutput = Object.values(nodeOutputs)[0]; + outputs.result = this.extractOutputValue(firstOutput); + } + + return outputs; + } + + /** + * Extract the actual value from output node result + * Output nodes typically have structure like { input: actualValue } + * @param {any} value - Output node value + * @returns {any} - Extracted value + */ + extractOutputValue(value) { + if (value && typeof value === 'object') { + // Check for common output node structures + if (value.input !== undefined) { + return value.input; + } + if (value.value !== undefined) { + return value.value; + } + if (value.output !== undefined) { + return value.output; + } + if (value.result !== undefined) { + return value.result; + } + } + + return value; + } + + /** + * Inject environment configuration into workflow nodes + * This allows nodes to access API keys and service URLs without exposing them in the workflow + * @param {Object} workflow - Workflow JSON (modified in place) + */ + injectEnvironmentConfig(workflow) { + for (const node of workflow.nodes) { + const nodeType = node.type; + const nodeData = node.data || {}; + + // Inject ComfyUI URL + if (nodeType === 'comfyui-image-generator' || nodeType === 'comfyui') { + if (!nodeData.comfyUIBaseUrl && config.services.comfyui.enabled) { + node.data.comfyUIBaseUrl = config.services.comfyui.url; + } + } + + // Inject LLM API keys and endpoints + if (nodeType === 'llm-chat' || nodeType === 'structured-llm' || nodeType === 'llm') { + // Use OpenAI key if not provided + if (!nodeData.apiKey && config.apiKeys.openai) { + node.data.apiKey = config.apiKeys.openai; + } + // Default to OpenAI endpoint if not provided + if (!nodeData.apiBaseUrl) { + node.data.apiBaseUrl = 'https://api.openai.com/v1'; + } + } + + // Inject Whisper API key + if (nodeType === 'whisper-transcription' || nodeType === 'speech-to-text') { + if (!nodeData.apiKey && config.apiKeys.openai) { + node.data.apiKey = config.apiKeys.openai; + } + if (!nodeData.baseUrl && config.services.pythonBackend.enabled) { + node.data.baseUrl = config.services.pythonBackend.url; + } + } + + // Inject Python backend URL for TTS + if (nodeType === 'text-to-speech' || nodeType === 'tts') { + if (!nodeData.baseUrl && config.services.pythonBackend.enabled) { + node.data.baseUrl = config.services.pythonBackend.url; + } + } + + // Inject Agent Executor configuration + if (nodeType === 'agent-executor' || nodeType === 'agent') { + if (!nodeData.provider && config.services.ollama.enabled) { + node.data.provider = 'ollama'; + node.data.baseUrl = config.services.ollama.url; + } + // Inject API keys based on provider + if (nodeData.provider === 'openai' && !nodeData.apiKey && config.apiKeys.openai) { + node.data.apiKey = config.apiKeys.openai; + } + if (nodeData.provider === 'anthropic' && !nodeData.apiKey && config.apiKeys.anthropic) { + node.data.apiKey = config.apiKeys.anthropic; + } + if (nodeData.provider === 'openrouter' && !nodeData.apiKey && config.apiKeys.openrouter) { + node.data.apiKey = config.apiKeys.openrouter; + } + } + } + } + + /** + * Create a timeout promise + * @param {number} timeoutMs - Timeout in milliseconds + * @returns {Promise} - Promise that rejects after timeout + */ + createTimeoutPromise(timeoutMs) { + return new Promise((_, reject) => { + setTimeout(() => { + reject(new Error(`Workflow execution timeout after ${timeoutMs}ms`)); + }, timeoutMs); + }); + } + + /** + * Validate workflow before execution + * @param {Object} workflow - Workflow JSON + * @returns {Object} - { valid: boolean, errors: string[] } + */ + validateWorkflow(workflow) { + const errors = []; + + try { + // Check workflow structure + if (!workflow.nodes || !Array.isArray(workflow.nodes)) { + errors.push('Workflow must have a nodes array'); + } + + if (!workflow.connections && !workflow.edges) { + errors.push('Workflow must have connections or edges'); + } + + // Check for input nodes + const hasInputNodes = workflow.nodes?.some(node => + ['input', 'image-input', 'pdf-input', 'file-upload'].includes(node.type) + ); + + if (!hasInputNodes) { + errors.push('Workflow must have at least one input node'); + } + + // Check for output nodes + const hasOutputNodes = workflow.nodes?.some(node => node.type === 'output'); + + if (!hasOutputNodes) { + errors.push('Workflow must have at least one output node'); + } + + // Check for circular dependencies (basic check) + if (workflow.connections) { + const nodeMap = new Map(); + workflow.nodes.forEach(node => nodeMap.set(node.id, new Set())); + + workflow.connections.forEach(conn => { + nodeMap.get(conn.source)?.add(conn.target); + }); + + // Simple cycle detection using DFS + const visited = new Set(); + const recStack = new Set(); + + const hasCycle = (nodeId) => { + if (recStack.has(nodeId)) return true; + if (visited.has(nodeId)) return false; + + visited.add(nodeId); + recStack.add(nodeId); + + const neighbors = nodeMap.get(nodeId) || new Set(); + for (const neighbor of neighbors) { + if (hasCycle(neighbor)) return true; + } + + recStack.delete(nodeId); + return false; + }; + + for (const nodeId of nodeMap.keys()) { + if (hasCycle(nodeId)) { + errors.push('Workflow contains circular dependencies'); + break; + } + } + } + } catch (error) { + errors.push(`Workflow validation error: ${error.message}`); + } + + return { + valid: errors.length === 0, + errors, + }; + } + + /** + * Get execution logs + * @returns {Array} - Execution logs + */ + getLogs() { + return this.runner.getLogs ? this.runner.getLogs() : []; + } + + /** + * Clear execution logs + */ + clearLogs() { + if (this.runner.clearLogs) { + this.runner.clearLogs(); + } + } +} + +/** + * Create a workflow executor instance + * @param {Object} options - Executor options + * @returns {WorkflowExecutor} - New executor instance + */ +export function createExecutor(options = {}) { + return new WorkflowExecutor(options); +} + +export default { + WorkflowExecutor, + createExecutor, +}; diff --git a/sdk/simple-test.js b/sdk/simple-test.js new file mode 100644 index 00000000..96c01bfb --- /dev/null +++ b/sdk/simple-test.js @@ -0,0 +1,130 @@ +/** + * Clara Flow SDK v2.0 - Simple Developer Test + * Shows the EASIEST way to use workflows + */ + +import { ClaraFlowRunner } from './dist/index.js'; + +console.log('🚀 Clara Flow SDK v2.0 - Developer Test\n'); + +// Simple workflow JSON (exported from Clara Studio) +const workflow = { + "name": "Text Processor", + "nodes": [ + { + "id": "user-input", + "type": "input", + "name": "User Input", + "data": { "label": "Enter your text" }, + "outputs": [{ "id": "output" }] + }, + { + "id": "prefix-input", + "type": "input", + "name": "Prefix", + "data": { + "label": "Prefix text", + "value": "[PROCESSED]" // Has default value + }, + "outputs": [{ "id": "output" }] + }, + { + "id": "combiner", + "type": "combine-text", + "name": "Text Combiner", + "data": { "separator": " " }, + "inputs": [ + { "id": "text1", "name": "Prefix" }, + { "id": "text2", "name": "Text" } + ], + "outputs": [{ "id": "output" }] + }, + { + "id": "result", + "type": "output", + "name": "Result", + "inputs": [{ "id": "input" }] + } + ], + "connections": [ + { "sourceNodeId": "prefix-input", "sourcePortId": "output", "targetNodeId": "combiner", "targetPortId": "text1" }, + { "sourceNodeId": "user-input", "sourcePortId": "output", "targetNodeId": "combiner", "targetPortId": "text2" }, + { "sourceNodeId": "combiner", "sourcePortId": "output", "targetNodeId": "result", "targetPortId": "input" } + ] +}; + +async function testDeveloperExperience() { + const runner = new ClaraFlowRunner({ enableLogging: true }); + + console.log('📋 STEP 1: Analyze what this workflow needs'); + console.log('=' .repeat(50)); + + // Show workflow info + const info = runner.describe(workflow); + console.log(`Name: ${info.name}`); + console.log(`Complexity: ${info.complexity}`); + console.log(`Node Count: ${info.nodeCount}`); + console.log(); + + // Show required inputs + const inputs = runner.getRequiredInputs(workflow); + console.log('Required Inputs:'); + inputs.forEach(input => { + const status = input.required ? '🔴 REQUIRED' : '🟢 OPTIONAL'; + console.log(` ${status} ${input.name} (${input.type})`); + if (input.defaultValue) { + console.log(` ↳ Default: "${input.defaultValue}"`); + } + }); + console.log(); + + console.log('🚀 STEP 2: Run the workflow'); + console.log('=' .repeat(50)); + + try { + // Method 1: Provide required inputs + console.log('📍 Test 1: Provide required inputs'); + const result1 = await runner.run(workflow, { + 'user-input': 'Hello World!' + // prefix-input will use default value "[PROCESSED]" + }); + console.log('✅ Result:', result1); + console.log(); + + // Method 2: Provide all inputs + console.log('📍 Test 2: Override all inputs'); + const result2 = await runner.run(workflow, { + 'user-input': 'Custom message', + 'prefix-input': '[CUSTOM]' + }); + console.log('✅ Result:', result2); + console.log(); + + // Method 3: Show error for missing inputs + console.log('📍 Test 3: Missing required input (will show error)'); + try { + const result3 = await runner.run(workflow, { + 'prefix-input': '[ONLY PREFIX]' + // Missing user-input which is required + }); + } catch (error) { + console.log('❌ Expected Error:', error.message.split('\n')[0]); + } + + } catch (error) { + console.error('❌ Test failed:', error.message); + } + + console.log(); + console.log('=' .repeat(50)); + console.log('💡 SUMMARY: Super Easy for Developers!'); + console.log('=' .repeat(50)); + console.log('✨ Just 3 lines of code:'); + console.log(' const runner = new ClaraFlowRunner();'); + console.log(' const inputs = runner.getRequiredInputs(workflow);'); + console.log(' const result = await runner.run(workflow, myInputs);'); + console.log(); + console.log('🎯 The SDK handles everything automatically!'); +} + +testDeveloperExperience().catch(console.error); \ No newline at end of file diff --git a/sdk/src/ClaraFlowRunner.ts b/sdk/src/ClaraFlowRunner.ts new file mode 100644 index 00000000..e7a225c9 --- /dev/null +++ b/sdk/src/ClaraFlowRunner.ts @@ -0,0 +1,166 @@ +import { FlowExecutor, ExecutionContext, FlowExecutorOptions } from '../../src/shared/FlowEngine'; +import { FlowNode, Connection, ExecutionLog } from '../../src/types/agent/types'; + +export interface FlowData { + format: string; + version: string; + flow: { + id: string; + name: string; + description?: string; + nodes: FlowNode[]; + connections: Connection[]; + }; + customNodes?: any[]; + metadata?: any; +} + +export interface FlowExecutionResult { + outputs: Record; + logs: ExecutionLog[]; +} + +export interface ClaraFlowRunnerOptions extends FlowExecutorOptions { + // Additional SDK-specific options can be added here +} + +export class ClaraFlowRunner { + private executor: FlowExecutor; + private customNodes: any[] = []; + + constructor(options: ClaraFlowRunnerOptions = {}) { + this.executor = new FlowExecutor(options); + } + + /** + * Execute a flow from exported Clara Studio data + */ + async executeFlow(flowData: FlowData, inputs: Record = {}): Promise { + try { + // Extract flow from Agent Studio format if needed + const flow = flowData.flow || flowData; + + if (!flow.nodes || !Array.isArray(flow.nodes)) { + throw new Error('Invalid flow: missing nodes array'); + } + + // Merge custom nodes from flow data with registered custom nodes + const allCustomNodes = [ + ...(flowData.customNodes || []), + ...this.customNodes + ]; + + // Execute the flow using the shared engine + const outputs = await this.executor.executeFlow( + flow.nodes, + flow.connections || [], + inputs, + allCustomNodes + ); + + // Get execution logs + const logs = this.executor.getLogs(); + + return { + outputs, + logs + }; + } catch (error) { + throw new Error(`Flow execution failed: ${error instanceof Error ? error.message : 'Unknown error'}`); + } + } + + /** + * Register a custom node for use in flows + */ + registerCustomNode(customNode: any): void { + this.customNodes.push(customNode); + } + + /** + * Execute multiple flows in batch + */ + async executeBatch( + flowData: FlowData, + inputSets: Record[], + options: { concurrency?: number; onProgress?: (progress: any) => void } = {} + ): Promise { + const { concurrency = 3, onProgress } = options; + const results = []; + + for (let i = 0; i < inputSets.length; i += concurrency) { + const batch = inputSets.slice(i, i + concurrency); + const batchPromises = batch.map(inputs => this.executeFlow(flowData, inputs)); + const batchResults = await Promise.all(batchPromises); + results.push(...batchResults); + + if (onProgress) { + onProgress({ + completed: Math.min(i + concurrency, inputSets.length), + total: inputSets.length, + progress: Math.min(i + concurrency, inputSets.length) / inputSets.length + }); + } + } + + return results; + } + + /** + * Get execution logs from the last flow run + */ + getLogs(): ExecutionLog[] { + return this.executor.getLogs(); + } + + /** + * Clear execution logs + */ + clearLogs(): void { + this.executor.clearLogs(); + } + + /** + * Validate a flow structure without executing it + */ + validateFlow(flowData: FlowData): { isValid: boolean; errors: string[] } { + const errors: string[] = []; + + try { + const flow = flowData.flow || flowData; + + if (!flow.nodes || !Array.isArray(flow.nodes)) { + errors.push('Missing or invalid nodes array'); + } + + if (!flow.connections || !Array.isArray(flow.connections)) { + errors.push('Missing or invalid connections array'); + } + + // Check for circular dependencies + if (flow.nodes && flow.connections) { + // Basic validation - could be extended + const nodeIds = new Set(flow.nodes.map(n => n.id)); + for (const conn of flow.connections) { + if (!nodeIds.has(conn.sourceNodeId)) { + errors.push(`Invalid connection: source node ${conn.sourceNodeId} not found`); + } + if (!nodeIds.has(conn.targetNodeId)) { + errors.push(`Invalid connection: target node ${conn.targetNodeId} not found`); + } + } + } + + return { + isValid: errors.length === 0, + errors + }; + } catch (error) { + errors.push(`Validation error: ${error instanceof Error ? error.message : 'Unknown error'}`); + return { + isValid: false, + errors + }; + } + } +} \ No newline at end of file diff --git a/sdk/src/browser.js b/sdk/src/browser.js new file mode 100644 index 00000000..f94a531b --- /dev/null +++ b/sdk/src/browser.js @@ -0,0 +1,191 @@ +/** + * Browser-compatible entry point for Clara Flow SDK + * Excludes Node.js-specific modules and provides browser-safe functionality + */ + +// Import core functionality +import { ClaraFlowRunner } from './index.js'; + +// Browser-compatible logger (simplified) +class BrowserLogger { + constructor(options = {}) { + this.level = options.level || 'info'; + this.enableColors = false; // No colors in browser console by default + } + + log(level, message, data = null) { + const timestamp = new Date().toISOString(); + const logMessage = `[${timestamp}] [${level.toUpperCase()}] ${message}`; + + if (data) { + console.log(logMessage, data); + } else { + console.log(logMessage); + } + } + + debug(message, data) { this.log('debug', message, data); } + info(message, data) { this.log('info', message, data); } + warn(message, data) { this.log('warn', message, data); } + error(message, data) { this.log('error', message, data); } +} + +// Browser-compatible Flow Runner +class BrowserClaraFlowRunner extends ClaraFlowRunner { + constructor(options = {}) { + // Override logger for browser compatibility + const browserOptions = { + ...options, + logger: new BrowserLogger(options.logger || {}) + }; + + super(browserOptions); + + // Disable Node.js-specific features + this.isNodeEnvironment = false; + this.isBrowserEnvironment = true; + } + + // Override methods that use Node.js-specific APIs + async loadFlowFromFile(filePath) { + throw new Error('File system operations are not supported in browser environment. Use loadFlow() with flow data instead.'); + } + + async saveFlowToFile(flowData, filePath) { + throw new Error('File system operations are not supported in browser environment. Use exportFlow() to get flow data instead.'); + } + + // Browser-compatible flow loading + async loadFlowFromUrl(url) { + try { + const response = await fetch(url); + if (!response.ok) { + throw new Error(`Failed to fetch flow from ${url}: ${response.statusText}`); + } + const flowData = await response.json(); + return this.loadFlow(flowData); + } catch (error) { + throw new Error(`Failed to load flow from URL: ${error.message}`); + } + } + + // Browser-compatible file handling for file-upload node + async handleFileUpload(file, options = {}) { + if (!(file instanceof File)) { + throw new Error('Expected File object for browser file upload'); + } + + const { outputFormat = 'base64', maxSize = 10 * 1024 * 1024 } = options; + + // Check file size + if (file.size > maxSize) { + throw new Error(`File size (${file.size} bytes) exceeds maximum allowed size (${maxSize} bytes)`); + } + + try { + switch (outputFormat) { + case 'base64': + return new Promise((resolve, reject) => { + const reader = new FileReader(); + reader.onload = () => resolve(reader.result); + reader.onerror = () => reject(new Error('Failed to read file as base64')); + reader.readAsDataURL(file); + }); + + case 'text': + return new Promise((resolve, reject) => { + const reader = new FileReader(); + reader.onload = () => resolve(reader.result); + reader.onerror = () => reject(new Error('Failed to read file as text')); + reader.readAsText(file); + }); + + case 'binary': + return new Promise((resolve, reject) => { + const reader = new FileReader(); + reader.onload = () => resolve(reader.result); + reader.onerror = () => reject(new Error('Failed to read file as binary')); + reader.readAsArrayBuffer(file); + }); + + case 'url': + return URL.createObjectURL(file); + + default: + throw new Error(`Unsupported output format: ${outputFormat}`); + } + } catch (error) { + throw new Error(`File upload failed: ${error.message}`); + } + } +} + +// Export for UMD build +export { BrowserClaraFlowRunner as ClaraFlowRunner }; + +// Also export utility functions +export const createFlowRunner = (options = {}) => { + return new BrowserClaraFlowRunner(options); +}; + +export const validateFlow = (flowData) => { + const runner = new BrowserClaraFlowRunner(); + return runner.validateFlow(flowData); +}; + +// Browser-specific utilities +export const BrowserUtils = { + // Check if running in browser + isBrowser: () => typeof window !== 'undefined', + + // Get browser info + getBrowserInfo: () => { + if (typeof navigator === 'undefined') return null; + return { + userAgent: navigator.userAgent, + platform: navigator.platform, + language: navigator.language + }; + }, + + // Download flow as file (browser-specific) + downloadFlow: (flowData, filename = 'flow.json') => { + const dataStr = JSON.stringify(flowData, null, 2); + const dataBlob = new Blob([dataStr], { type: 'application/json' }); + const url = URL.createObjectURL(dataBlob); + + const link = document.createElement('a'); + link.href = url; + link.download = filename; + document.body.appendChild(link); + link.click(); + document.body.removeChild(link); + URL.revokeObjectURL(url); + }, + + // Load flow from file input + loadFlowFromFileInput: (fileInput) => { + return new Promise((resolve, reject) => { + const file = fileInput.files[0]; + if (!file) { + reject(new Error('No file selected')); + return; + } + + const reader = new FileReader(); + reader.onload = (e) => { + try { + const flowData = JSON.parse(e.target.result); + resolve(flowData); + } catch (error) { + reject(new Error('Invalid JSON file')); + } + }; + reader.onerror = () => reject(new Error('Failed to read file')); + reader.readAsText(file); + }); + } +}; + +// Default export +export default BrowserClaraFlowRunner; \ No newline at end of file diff --git a/sdk/src/customNodeManager.js b/sdk/src/customNodeManager.js new file mode 100644 index 00000000..d1474fc6 --- /dev/null +++ b/sdk/src/customNodeManager.js @@ -0,0 +1,447 @@ +/** + * Custom Node Manager - Handles registration and execution of custom nodes + */ + +export class CustomNodeManager { + constructor(logger, enableSandbox = true) { + this.logger = logger; + this.enableSandbox = enableSandbox; + this.customNodes = new Map(); + this.executionCache = new Map(); + } + + /** + * Register multiple custom nodes + * @param {Array} nodeDefinitions - Array of custom node definitions + */ + async registerNodes(nodeDefinitions) { + for (const nodeDefinition of nodeDefinitions) { + await this.registerNode(nodeDefinition); + } + } + + /** + * Register a single custom node + * @param {Object} nodeDefinition - Custom node definition + */ + async registerNode(nodeDefinition) { + const { type, name, executionCode } = nodeDefinition; + + if (!type || !name || !executionCode) { + throw new Error('Custom node must have type, name, and executionCode'); + } + + this.logger.info(`Registering custom node: ${name} (${type})`); + + try { + // Validate and prepare execution code + const executor = this.prepareExecutor(executionCode, nodeDefinition); + + // Store the node definition and executor + this.customNodes.set(type, { + definition: nodeDefinition, + executor: executor, + registeredAt: new Date().toISOString() + }); + + this.logger.info(`Successfully registered custom node: ${type}`); + + } catch (error) { + this.logger.error(`Failed to register custom node ${type}: ${error.message}`); + throw new Error(`Custom node registration failed: ${error.message}`); + } + } + + /** + * Execute a custom node + * @param {string} nodeType - Type of the custom node + * @param {Object} inputs - Input values + * @param {Object} properties - Node properties + * @returns {Promise} Execution result + */ + async executeNode(nodeType, inputs = {}, properties = {}) { + if (!this.hasNode(nodeType)) { + throw new Error(`Custom node type '${nodeType}' is not registered`); + } + + const nodeInfo = this.customNodes.get(nodeType); + const { definition, executor } = nodeInfo; + + this.logger.debug(`Executing custom node: ${definition.name} (${nodeType})`, { inputs, properties }); + + try { + // Create execution context + const context = this.createExecutionContext(definition.name); + + // Map inputs according to node definition + const mappedInputs = this.mapInputs(inputs, definition.inputs || []); + + // Map properties according to node definition + const mappedProperties = this.mapProperties(properties, definition.properties || []); + + // Execute the custom node + const result = await executor(mappedInputs, mappedProperties, context); + + // Validate and map outputs + const mappedResult = this.mapOutputs(result, definition.outputs || []); + + this.logger.debug(`Custom node execution completed: ${definition.name}`, { result: mappedResult }); + return mappedResult; + + } catch (error) { + this.logger.error(`Custom node execution failed: ${definition.name}`, { error: error.message }); + throw new Error(`Custom node '${definition.name}' execution failed: ${error.message}`); + } + } + + /** + * Prepare executor function from execution code + * @param {string} executionCode - JavaScript code for the node + * @param {Object} nodeDefinition - Node definition for validation + * @returns {Function} Prepared executor function + */ + prepareExecutor(executionCode, nodeDefinition) { + try { + // Extract the execute function from the code + let cleanCode = executionCode.trim(); + + // If the code doesn't start with 'async function execute', wrap it + if (!cleanCode.startsWith('async function execute')) { + // Look for just the function body + if (cleanCode.includes('async function execute')) { + // Code contains the full function + } else { + // Assume it's just the function body, wrap it + cleanCode = `async function execute(inputs, properties, context) {\n${cleanCode}\n}`; + } + } + + // Create a safe execution environment + if (this.enableSandbox) { + return this.createSandboxedExecutor(cleanCode, nodeDefinition); + } else { + return this.createDirectExecutor(cleanCode, nodeDefinition); + } + + } catch (error) { + throw new Error(`Failed to prepare executor: ${error.message}`); + } + } + + /** + * Create a sandboxed executor (safer but limited) + * @param {string} code - Execution code + * @param {Object} nodeDefinition - Node definition + * @returns {Function} Sandboxed executor + */ + createSandboxedExecutor(code, nodeDefinition) { + // Create a restricted global context + const sandbox = { + // Allow basic JavaScript features + console: { + log: (...args) => this.logger.debug(`[${nodeDefinition.name}]`, ...args), + warn: (...args) => this.logger.warn(`[${nodeDefinition.name}]`, ...args), + error: (...args) => this.logger.error(`[${nodeDefinition.name}]`, ...args), + }, + JSON: JSON, + Math: Math, + Date: Date, + Number: Number, + String: String, + Boolean: Boolean, + Array: Array, + Object: Object, + Promise: Promise, + setTimeout: setTimeout, + clearTimeout: clearTimeout, + // Add common utilities + btoa: typeof btoa !== 'undefined' ? btoa : undefined, + atob: typeof atob !== 'undefined' ? atob : undefined, + }; + + // Use Function constructor to create executor in restricted context + const executorFunction = new Function( + 'sandbox', + 'inputs', + 'properties', + 'context', + ` + with (sandbox) { + ${code} + return execute(inputs, properties, context); + } + ` + ); + + return async (inputs, properties, context) => { + return await executorFunction(sandbox, inputs, properties, context); + }; + } + + /** + * Create a direct executor (faster but less secure) + * @param {string} code - Execution code + * @param {Object} nodeDefinition - Node definition + * @returns {Function} Direct executor + */ + createDirectExecutor(code, nodeDefinition) { + // Create executor function directly + const executorFunction = new Function( + 'inputs', + 'properties', + 'context', + ` + ${code} + return execute(inputs, properties, context); + ` + ); + + return async (inputs, properties, context) => { + return await executorFunction(inputs, properties, context); + }; + } + + /** + * Create execution context for custom nodes + * @param {string} nodeName - Name of the node + * @returns {Object} Execution context + */ + createExecutionContext(nodeName) { + return { + log: (...args) => { + this.logger.info(`[${nodeName}]`, ...args); + }, + warn: (...args) => { + this.logger.warn(`[${nodeName}]`, ...args); + }, + error: (...args) => { + this.logger.error(`[${nodeName}]`, ...args); + }, + debug: (...args) => { + this.logger.debug(`[${nodeName}]`, ...args); + }, + nodeName: nodeName, + timestamp: Date.now(), + }; + } + + /** + * Map input values according to node definition + * @param {Object} inputs - Raw input values + * @param {Array} inputDefinitions - Node input definitions + * @returns {Object} Mapped inputs + */ + mapInputs(inputs, inputDefinitions) { + const mapped = {}; + + // If no input definitions, return inputs as-is + if (!inputDefinitions || inputDefinitions.length === 0) { + return inputs; + } + + // Map each defined input + for (const inputDef of inputDefinitions) { + const { id, name, dataType, defaultValue } = inputDef; + + // Try to find value by ID first (for exported flows), then by name + let value = inputs[id] || inputs[name]; + + // Use default if no value provided + if (value === undefined && defaultValue !== undefined) { + value = defaultValue; + } + + // Type conversion + if (value !== undefined && dataType) { + value = this.convertDataType(value, dataType); + } + + // Add both name and lowercase name for compatibility + mapped[name] = value; + if (name) { + mapped[name.toLowerCase()] = value; + } + } + + return mapped; + } + + /** + * Map property values according to node definition + * @param {Object} properties - Raw property values + * @param {Array} propertyDefinitions - Node property definitions + * @returns {Object} Mapped properties + */ + mapProperties(properties, propertyDefinitions) { + const mapped = {}; + + // If no property definitions, return properties as-is + if (!propertyDefinitions || propertyDefinitions.length === 0) { + return properties; + } + + // Map each defined property + for (const propDef of propertyDefinitions) { + const { name, defaultValue } = propDef; + let value = properties[name]; + + // Use default if no value provided + if (value === undefined && defaultValue !== undefined) { + value = defaultValue; + } + + mapped[name] = value; + } + + return mapped; + } + + /** + * Map output values according to node definition + * @param {any} result - Raw execution result + * @param {Array} outputDefinitions - Node output definitions + * @returns {any} Mapped outputs + */ + mapOutputs(result, outputDefinitions) { + // If no output definitions or single output, return result as-is + if (!outputDefinitions || outputDefinitions.length === 0) { + return result; + } + + // If single output definition and result is not an object, wrap it + if (outputDefinitions.length === 1 && (typeof result !== 'object' || result === null)) { + return { [outputDefinitions[0].name]: result }; + } + + // If result is an object, map according to definitions + if (typeof result === 'object' && result !== null) { + const mapped = {}; + + for (const outputDef of outputDefinitions) { + const { id, name, dataType } = outputDef; + + // Try multiple ways to find the output value: + // 1. By exact name + // 2. By lowercase name (for case-insensitive matching) + // 3. By ID + let value = result[name] || + result[name?.toLowerCase()] || + result[id] || + // Try all case variations + Object.keys(result).find(key => + key.toLowerCase() === name?.toLowerCase() + ) && result[Object.keys(result).find(key => + key.toLowerCase() === name?.toLowerCase() + )]; + + // Type conversion + if (value !== undefined && dataType) { + value = this.convertDataType(value, dataType); + } + + mapped[name] = value; + } + + return mapped; + } + + return result; + } + + /** + * Convert value to specified data type + * @param {any} value - Value to convert + * @param {string} dataType - Target data type + * @returns {any} Converted value + */ + convertDataType(value, dataType) { + try { + switch (dataType) { + case 'string': + return String(value); + case 'number': + return Number(value); + case 'boolean': + return Boolean(value); + case 'json': + case 'object': + return typeof value === 'string' ? JSON.parse(value) : value; + case 'array': + return Array.isArray(value) ? value : [value]; + default: + return value; + } + } catch (error) { + this.logger.warn(`Type conversion failed for ${dataType}:`, error.message); + return value; + } + } + + /** + * Check if a custom node type is registered + * @param {string} nodeType - Node type to check + * @returns {boolean} True if registered + */ + hasNode(nodeType) { + return this.customNodes.has(nodeType); + } + + /** + * Get list of registered custom node types + * @returns {Array} Registered node types + */ + getRegisteredNodeTypes() { + return Array.from(this.customNodes.keys()); + } + + /** + * Get custom node definition + * @param {string} nodeType - Node type + * @returns {Object|null} Node definition or null if not found + */ + getNodeDefinition(nodeType) { + const nodeInfo = this.customNodes.get(nodeType); + return nodeInfo ? nodeInfo.definition : null; + } + + /** + * Unregister a custom node + * @param {string} nodeType - Node type to unregister + */ + unregisterNode(nodeType) { + if (this.customNodes.has(nodeType)) { + this.customNodes.delete(nodeType); + this.executionCache.delete(nodeType); + this.logger.info(`Unregistered custom node: ${nodeType}`); + } + } + + /** + * Clear all registered custom nodes + */ + clear() { + const nodeCount = this.customNodes.size; + this.customNodes.clear(); + this.executionCache.clear(); + this.logger.info(`Cleared ${nodeCount} custom nodes`); + } + + /** + * Get execution statistics + * @returns {Object} Statistics + */ + getStats() { + return { + registeredNodes: this.customNodes.size, + cacheSize: this.executionCache.size, + sandboxEnabled: this.enableSandbox + }; + } + + /** + * Dispose resources and cleanup + */ + dispose() { + this.clear(); + } +} \ No newline at end of file diff --git a/sdk/src/flowEngine.js b/sdk/src/flowEngine.js new file mode 100644 index 00000000..be05de8a --- /dev/null +++ b/sdk/src/flowEngine.js @@ -0,0 +1,571 @@ +/** + * Flow Engine - Handles flow execution with proper node ordering and data flow + * Updated to match Agent Builder UI execution logic exactly + */ + +export class FlowEngine { + constructor(logger, nodeExecutor, validator) { + this.logger = logger; + this.nodeExecutor = nodeExecutor; + this.validator = validator; + } + + /** + * Execute a complete flow + * @param {Object} flowData - Flow definition + * @param {Object} inputs - Input values for the flow + * @returns {Promise} Flow execution result + */ + async executeFlow(flowData, inputs = {}) { + const { nodes, connections } = flowData; + + if (!nodes || nodes.length === 0) { + throw new Error('Flow has no nodes to execute'); + } + + this.logger.info('Starting flow execution', { nodeCount: nodes.length, connectionCount: connections?.length || 0 }); + + try { + // Get execution order using the same topological sort as the UI + const executionOrder = this.getExecutionOrder(nodes, connections || []); + this.logger.info('Execution order determined', { + order: executionOrder.map(n => `${n.name} (${n.type})`).join(' → ') + }); + + // Initialize results storage (like localResults in UI) + const nodeResults = new Map(); + + // Map flow inputs to input nodes (same as UI logic) + const flowInputMapping = this.mapFlowInputsToNodes(inputs, nodes); + for (const [nodeId, inputValue] of Object.entries(flowInputMapping)) { + nodeResults.set(nodeId, inputValue); + } + + // Execute nodes in order (exactly like UI) + for (const node of executionOrder) { + await this.executeNodeInFlow(node, nodes, connections || [], nodeResults); + } + + // Collect outputs (same as UI) + const outputs = this.collectFlowOutputs(nodes, nodeResults); + + this.logger.info('Flow execution completed successfully', { outputs }); + return outputs; + + } catch (error) { + this.logger.error('Flow execution failed', { error: error.message }); + throw error; + } + } + + /** + * Get execution order using Kahn's algorithm (same as Agent Builder UI) + * @param {Array} nodes - Flow nodes + * @param {Array} connections - Flow connections + * @returns {Array} Ordered list of nodes + */ + getExecutionOrder(nodes, connections) { + const inDegree = {}; + const adjList = {}; + + // Initialize (same as UI) + nodes.forEach(node => { + inDegree[node.id] = 0; + adjList[node.id] = []; + }); + + // Build adjacency list and count incoming edges (same as UI) + connections.forEach(conn => { + adjList[conn.sourceNodeId].push(conn.targetNodeId); + inDegree[conn.targetNodeId]++; + }); + + // Topological sort using Kahn's algorithm (same as UI) + const queue = []; + const result = []; + + // Start with nodes that have no incoming edges + Object.keys(inDegree).forEach(nodeId => { + if (inDegree[nodeId] === 0) { + queue.push(nodeId); + } + }); + + while (queue.length > 0) { + const nodeId = queue.shift(); + const node = nodes.find(n => n.id === nodeId); + if (node) { + result.push(node); + } + + adjList[nodeId].forEach(targetId => { + inDegree[targetId]--; + if (inDegree[targetId] === 0) { + queue.push(targetId); + } + }); + } + + return result; + } + + /** + * Map flow inputs to input nodes (same logic as UI) + * @param {Object} inputs - Flow inputs + * @param {Array} nodes - Flow nodes + * @returns {Object} Mapped inputs + */ + mapFlowInputsToNodes(inputs, nodes) { + const mapped = {}; + + // Find input nodes and map values (same as UI logic) + for (const node of nodes) { + if (node.type === 'input') { + // Try different input mapping strategies (same as UI) + const inputValue = inputs[node.id] || + inputs[node.name] || + inputs[node.data?.label] || + node.data?.value || + node.data?.defaultValue; + + if (inputValue !== undefined) { + mapped[node.id] = inputValue; + } + } + } + + // Also map inputs to any node that has a direct input provided (for source nodes like file-upload) + for (const [nodeId, inputValue] of Object.entries(inputs)) { + const node = nodes.find(n => n.id === nodeId || n.name === nodeId); + if (node && inputValue !== undefined) { + mapped[node.id] = inputValue; + } + } + + this.logger.debug('Flow inputs mapped', { mapped }); + return mapped; + } + + /** + * Get inputs for a node from connected outputs (exact same logic as UI) + * @param {string} nodeId - Target node ID + * @param {Array} nodes - All nodes + * @param {Array} connections - All connections + * @param {Map} nodeResults - Current node results + * @returns {Object} Node inputs + */ + getNodeInputs(nodeId, nodes, connections, nodeResults) { + const inputs = {}; + + this.logger.debug(`Getting inputs for node ${nodeId}`); + + // Find the target node to understand its input definitions (same as UI) + const targetNode = nodes.find(n => n.id === nodeId); + + connections.forEach(conn => { + if (conn.targetNodeId === nodeId) { + const sourceNodeResult = nodeResults.get(conn.sourceNodeId); + + if (sourceNodeResult !== undefined) { + // Extract the specific output port value from the source result (same as UI) + let sourceValue = sourceNodeResult; + + // If the source result is an object and we have a specific source port, extract that value + if (typeof sourceNodeResult === 'object' && sourceNodeResult !== null && conn.sourcePortId) { + // Find the source node to understand its output structure + const sourceNode = nodes.find(n => n.id === conn.sourceNodeId); + if (sourceNode) { + const sourceOutput = sourceNode.outputs?.find(output => output.id === conn.sourcePortId); + if (sourceOutput && sourceNodeResult.hasOwnProperty(sourceOutput.id)) { + sourceValue = sourceNodeResult[sourceOutput.id]; + this.logger.debug(`Extracted specific output port ${conn.sourcePortId}: ${sourceValue}`); + } + } + } + + // Map the target port ID to the logical input name (same as UI) + if (targetNode) { + const targetInput = targetNode.inputs?.find(input => input.id === conn.targetPortId); + if (targetInput) { + // Use the logical input name from the node definition + const logicalName = targetInput.id; + inputs[logicalName] = sourceValue; + + // Also add common fallback mappings for execution functions (same as UI) + if (logicalName === 'input' || targetInput.name?.toLowerCase().includes('input')) { + inputs.input = sourceValue; + } + if (logicalName === 'user' || targetInput.name?.toLowerCase().includes('user')) { + inputs.user = sourceValue; + } + if (logicalName === 'system' || targetInput.name?.toLowerCase().includes('system')) { + inputs.system = sourceValue; + } + if (logicalName === 'context' || targetInput.name?.toLowerCase().includes('context')) { + inputs.context = sourceValue; + } + if (logicalName === 'text' || targetInput.name?.toLowerCase().includes('text')) { + inputs.text = sourceValue; + inputs.input = sourceValue; // Text nodes often expect 'input' + } + } + } + + // Fallback: also store with the original port ID (same as UI) + inputs[conn.targetPortId] = sourceValue; + } + } + }); + + this.logger.debug(`Final inputs for ${nodeId}:`, inputs); + return inputs; + } + + /** + * Execute a single node within the flow context (same as UI logic) + * @param {Object} node - Node to execute + * @param {Array} nodes - All nodes + * @param {Array} connections - All connections + * @param {Map} nodeResults - Node results storage + */ + async executeNodeInFlow(node, nodes, connections, nodeResults) { + this.logger.debug(`Executing node: ${node.name} (${node.type})`); + + try { + // Get inputs for this node using current results (same as UI) + const nodeInputs = this.getNodeInputs(node.id, nodes, connections, nodeResults); + + // Debug: Log what we have for this node before merging + this.logger.debug(`Node ${node.name} - Before merging:`, { + nodeInputs, + hasStoredValue: nodeResults.has(node.id), + storedValue: nodeResults.get(node.id), + storedValueType: typeof nodeResults.get(node.id) + }); + + // For input nodes, check if we already have a result stored + if (node.type === 'input' && nodeResults.has(node.id)) { + // Use the stored input value + const storedValue = nodeResults.get(node.id); + this.logger.debug(`Using stored input value for ${node.name}:`, storedValue); + nodeResults.set(node.id, storedValue); + return; + } + + // For file-upload nodes, pre-process file data and store in node.data.outputs (like AgentStudio) + if (node.type === 'file-upload' && nodeResults.has(node.id)) { + const storedValue = nodeResults.get(node.id); + this.logger.debug(`Processing file-upload node ${node.name} with stored value:`, { + storedValue, + storedValueType: typeof storedValue + }); + + if (typeof storedValue === 'object' && storedValue !== null) { + // Pre-process file data and store in node.data.outputs + const fileData = storedValue.file || storedValue.data; + if (fileData) { + const outputFormat = node.data?.outputFormat || 'binary'; + const processedOutput = await this.processFileData(fileData, outputFormat, storedValue); + + // Store processed data in node.data.outputs (like AgentStudio) + if (!node.data.outputs) { + node.data.outputs = {}; + } + node.data.outputs.content = processedOutput.content; + node.data.outputs.metadata = processedOutput.metadata; + + this.logger.debug(`Pre-processed file data for ${node.name}:`, { + outputFormat, + contentType: typeof processedOutput.content, + metadata: processedOutput.metadata + }); + } + } + } + + // For any other node with stored inputs, merge them with connected inputs + if (nodeResults.has(node.id) && node.type !== 'file-upload') { + const storedValue = nodeResults.get(node.id); + if (typeof storedValue === 'object' && storedValue !== null) { + // If stored value is an object, merge its properties as inputs + Object.assign(nodeInputs, storedValue); + this.logger.debug(`Merged stored inputs for ${node.name}:`, nodeInputs); + } + } + + // Execute the node (same as UI) + const result = await this.nodeExecutor.executeNode(node, nodeInputs); + + // Store result for dependent nodes (same as UI) + nodeResults.set(node.id, result); + + this.logger.debug(`Node execution completed: ${node.name}`, { result }); + + } catch (error) { + this.logger.error(`Node execution failed: ${node.name}`, { error: error.message }); + throw new Error(`Node '${node.name}' (${node.id}) execution failed: ${error.message}`); + } + } + + /** + * Collect flow outputs from output nodes (same logic as UI) + * @param {Array} nodes - Flow nodes + * @param {Map} nodeResults - Node results storage + * @returns {Object} Flow outputs + */ + collectFlowOutputs(nodes, nodeResults) { + const outputs = {}; + + // Collect from output nodes (same as UI) + for (const node of nodes) { + if (node.type === 'output') { + const result = nodeResults.get(node.id); + if (result !== undefined) { + const outputKey = node.data?.label || node.name || node.id; + outputs[outputKey] = result; + } + } + } + + // If no output nodes, return all node results (same as UI) + if (Object.keys(outputs).length === 0) { + const results = {}; + for (const [nodeId, result] of nodeResults) { + const node = nodes.find(n => n.id === nodeId); + if (node && node.type !== 'input') { + results[node.name || nodeId] = result; + } + } + return results; + } + + return outputs; + } + + /** + * Process file data for file-upload nodes (matches AgentStudio behavior) + * @param {*} fileData - Raw file data (ArrayBuffer, string, etc.) + * @param {string} outputFormat - Desired output format + * @param {Object} metadata - File metadata (name, type, size) + * @returns {Object} Processed file data with content and metadata + */ + async processFileData(fileData, outputFormat, metadata = {}) { + let processedData; + let fileName = metadata.name || 'uploaded_file'; + let mimeType = metadata.type || 'application/octet-stream'; + let fileSize = metadata.size || 0; + + try { + if (typeof fileData === 'string') { + // Assume base64 string + const base64Data = fileData.includes(',') ? fileData.split(',')[1] : fileData; + const binaryData = atob(base64Data); + fileSize = fileSize || binaryData.length; + processedData = fileData; + } else if (fileData instanceof ArrayBuffer || fileData instanceof Uint8Array) { + // Binary data + const uint8Array = fileData instanceof ArrayBuffer ? new Uint8Array(fileData) : fileData; + fileSize = fileSize || uint8Array.length; + + // Convert to base64 for processing + const binaryString = Array.from(uint8Array, byte => String.fromCharCode(byte)).join(''); + processedData = btoa(binaryString); + } else if (typeof fileData === 'object' && fileData.data) { + // Object with file metadata + fileName = fileData.name || fileName; + mimeType = fileData.type || mimeType; + fileSize = fileData.size || fileSize; + processedData = fileData.data; + } else { + throw new Error('Unsupported file data format'); + } + + // Return data in requested format + const result = { + metadata: { + fileName, + mimeType, + size: fileSize, + timestamp: new Date().toISOString(), + format: outputFormat + } + }; + + switch (outputFormat) { + case 'base64': + result.content = processedData.includes('data:') ? processedData : `data:${mimeType};base64,${processedData}`; + break; + case 'base64_raw': + result.content = processedData.includes(',') ? processedData.split(',')[1] : processedData; + break; + case 'binary': + try { + const base64Data = processedData.includes(',') ? processedData.split(',')[1] : processedData; + const binaryString = atob(base64Data); + const uint8Array = new Uint8Array(binaryString.length); + for (let i = 0; i < binaryString.length; i++) { + uint8Array[i] = binaryString.charCodeAt(i); + } + result.content = uint8Array; + } catch (error) { + throw new Error(`Failed to convert to binary: ${error.message}`); + } + break; + case 'text': + try { + const base64Data = processedData.includes(',') ? processedData.split(',')[1] : processedData; + result.content = atob(base64Data); + } catch (error) { + throw new Error(`Failed to convert to text: ${error.message}`); + } + break; + case 'metadata': + // Return only metadata without file content + result.content = null; + break; + default: + result.content = processedData; + } + + return result; + + } catch (error) { + this.logger.error('File processing failed:', error); + throw new Error(`File processing failed: ${error.message}`); + } + } + + /** + * Validate flow before execution + * @param {Object} flowData - Flow definition + * @returns {Object} Validation result + */ + validateFlow(flowData) { + return this.validator.validateFlow(flowData); + } + + /** + * Get execution statistics for a flow + * @param {Object} flowData - Flow definition + * @returns {Object} Flow statistics + */ + getFlowStats(flowData) { + const { nodes = [], connections = [] } = flowData; + + const inputNodes = nodes.filter(n => n.type === 'input').length; + const outputNodes = nodes.filter(n => n.type === 'output').length; + const processingNodes = nodes.length - inputNodes - outputNodes; + + return { + totalNodes: nodes.length, + inputNodes, + outputNodes, + processingNodes, + connections: connections.length, + complexity: this.calculateFlowComplexity(nodes, connections) + }; + } + + /** + * Calculate flow complexity score + * @param {Array} nodes - Flow nodes + * @param {Array} connections - Flow connections + * @returns {number} Complexity score + */ + calculateFlowComplexity(nodes, connections) { + // Simple complexity calculation based on nodes and connections + const nodeComplexity = nodes.length; + const connectionComplexity = connections.length * 0.5; + const branchingFactor = Math.max(1, connections.length / Math.max(1, nodes.length)); + + return Math.round(nodeComplexity + connectionComplexity + branchingFactor); + } + + /** + * Create a subflow from a portion of the main flow + * @param {Object} flowData - Main flow data + * @param {Array} nodeIds - Node IDs to include in subflow + * @returns {Object} Subflow data + */ + createSubflow(flowData, nodeIds) { + const { nodes, connections, customNodes } = flowData; + + // Filter nodes + const subflowNodes = nodes.filter(node => nodeIds.includes(node.id)); + + // Filter connections that are between selected nodes + const subflowConnections = connections.filter(conn => + nodeIds.includes(conn.sourceNodeId) && nodeIds.includes(conn.targetNodeId) + ); + + return { + ...flowData, + nodes: subflowNodes, + connections: subflowConnections, + customNodes: customNodes || [] + }; + } + + /** + * Legacy method for backward compatibility + * @deprecated Use getExecutionOrder instead + */ + determineExecutionOrder(executionGraph) { + // Convert Map format to array format + const nodes = Array.from(executionGraph.keys()).map(nodeId => ({ id: nodeId })); + const connections = []; + + for (const [nodeId, nodeInfo] of executionGraph) { + for (const dep of nodeInfo.dependencies) { + connections.push({ sourceNodeId: dep, targetNodeId: nodeId }); + } + } + + return this.getExecutionOrder(nodes, connections).map(node => node.id); + } + + /** + * Legacy method for backward compatibility + * @deprecated Use mapFlowInputsToNodes instead + */ + mapFlowInputs(inputs, nodes) { + return this.mapFlowInputsToNodes(inputs, nodes); + } + + /** + * Legacy method for backward compatibility + * @deprecated Use collectFlowOutputs instead + */ + collectNodeInputs(nodeInfo, nodeResults) { + // This is the old method that had the bug + // Now it properly handles input node results + const inputs = {}; + + for (const [targetPort, connectionInfo] of nodeInfo.inputs) { + const { sourceNodeId, sourcePortId } = connectionInfo; + const sourceResult = nodeResults.get(sourceNodeId); + + if (sourceResult !== undefined) { + // For input nodes, the result is typically a simple value + // For other nodes, check if result has the specific port property + if (typeof sourceResult === 'object' && sourceResult !== null && !Array.isArray(sourceResult)) { + // If it's an object, try to get the specific port value + if (sourceResult.hasOwnProperty(sourcePortId)) { + inputs[targetPort] = sourceResult[sourcePortId]; + } else if (sourcePortId === 'output' && Object.keys(sourceResult).length === 1) { + // If looking for 'output' port and object has only one property, use that value + inputs[targetPort] = Object.values(sourceResult)[0]; + } else { + // Use the entire result as fallback + inputs[targetPort] = sourceResult; + } + } else { + // For primitive values (strings, numbers, booleans) or arrays, use directly + inputs[targetPort] = sourceResult; + } + } + } + + return inputs; + } +} \ No newline at end of file diff --git a/sdk/src/index.js b/sdk/src/index.js new file mode 100644 index 00000000..ea391eeb --- /dev/null +++ b/sdk/src/index.js @@ -0,0 +1,1451 @@ +/** + * Clara Flow SDK v2.0 - Modern AI Workflow Execution Engine + * Zero-config SDK for running Clara AI agent workflows + */ + +// Core execution engine +class ClaraFlowRunner { + constructor(options = {}) { + this.config = { + timeout: options.timeout || 30000, + enableLogging: options.enableLogging !== false, + logLevel: options.logLevel || 'info', + maxRetries: options.maxRetries || 3, + ...options + }; + + this.executionLogs = []; + this.customNodes = new Map(); + this.isExecuting = false; + + if (this.config.enableLogging) { + this.log('Clara Flow SDK v2.0 initialized'); + } + } + + /** + * Execute a workflow with inputs + * @param {Object} flowData - Exported workflow from Clara Studio + * @param {Object} inputs - Input values for the workflow + * @returns {Promise} Execution results + */ + async execute(flowData, inputs = {}) { + if (this.isExecuting) { + throw new Error('Another workflow is already executing'); + } + + this.isExecuting = true; + const startTime = Date.now(); + + try { + this.log('🚀 Starting workflow execution'); + + // Normalize flow data format + const flow = this.normalizeFlow(flowData); + + // Register custom nodes if present + this.registerCustomNodes(flow.customNodes || []); + + // Validate workflow + this.validateFlow(flow); + + // Execute workflow + const results = await this.executeWorkflow(flow, inputs); + + const duration = Date.now() - startTime; + this.log(`✅ Workflow completed successfully in ${duration}ms`); + + return results; + + } catch (error) { + const duration = Date.now() - startTime; + this.log(`❌ Workflow failed after ${duration}ms: ${error.message}`, 'error'); + throw error; + } finally { + this.isExecuting = false; + } + } + + /** + * Register a custom node type + * @param {Object} nodeDefinition - Custom node definition + */ + registerCustomNode(nodeDefinition) { + if (!nodeDefinition.type || !nodeDefinition.executionCode) { + throw new Error('Custom node must have type and executionCode'); + } + + this.customNodes.set(nodeDefinition.type, nodeDefinition); + this.log(`📦 Registered custom node: ${nodeDefinition.type}`); + } + + /** + * Get execution logs + * @returns {Array} Array of log entries + */ + getLogs() { + return [...this.executionLogs]; + } + + /** + * Clear execution logs + */ + clearLogs() { + this.executionLogs = []; + } + + /** + * Get required inputs for a workflow (what the developer needs to provide) + * @param {Object} flowData - Workflow JSON + * @returns {Array} Array of required input descriptions + */ + getRequiredInputs(flowData) { + try { + const flow = this.normalizeFlow(flowData); + const inputNodes = flow.nodes.filter(node => node.type === 'input'); + + return inputNodes.map(node => ({ + id: node.id, + name: node.name || node.data?.label || node.id, + description: node.data?.description || `Input for ${node.name || node.id}`, + type: node.data?.type || 'text', + required: !node.data?.value && !node.data?.defaultValue, // Required if no default + defaultValue: node.data?.value || node.data?.defaultValue, + example: this.getInputExample(node.data?.type || 'text') + })); + } catch (error) { + throw new Error(`Failed to analyze workflow inputs: ${error.message}`); + } + } + + /** + * Get example value for input type + * @private + */ + getInputExample(type) { + const examples = { + 'text': 'Hello world', + 'number': 42, + 'json': '{"key": "value"}', + 'boolean': true, + 'email': 'user@example.com', + 'url': 'https://example.com' + }; + return examples[type] || 'Sample input'; + } + + /** + * Simple execution - automatically prompt for missing inputs + * @param {Object} flowData - Workflow JSON + * @param {Object} inputs - Optional inputs (if not provided, will prompt) + * @returns {Promise} Execution result + */ + async run(flowData, inputs = {}) { + // Get required inputs + const requiredInputs = this.getRequiredInputs(flowData); + + // Check if we have all required inputs + const missingInputs = requiredInputs.filter(input => + input.required && !(input.id in inputs) && !(input.name in inputs) + ); + + if (missingInputs.length > 0 && typeof process !== 'undefined' && process.stdin && typeof window === 'undefined') { + // We're in Node.js and have missing inputs - prompt for them + this.log('🔍 Missing required inputs, prompting user...'); + + try { + const readline = await import('readline'); + + for (const input of missingInputs) { + const rl = readline.createInterface({ + input: process.stdin, + output: process.stdout + }); + + const prompt = `${input.name} (${input.type})${input.defaultValue ? ` [${input.defaultValue}]` : ''}: `; + const answer = await new Promise(resolve => rl.question(prompt, resolve)); + rl.close(); + + if (answer.trim() || !input.defaultValue) { + inputs[input.id] = answer.trim() || input.defaultValue; + } else { + inputs[input.id] = input.defaultValue; + } + } + } catch (error) { + // Fallback if readline import fails + const inputList = missingInputs.map(i => `- ${i.name} (${i.type}): ${i.description}`).join('\n'); + throw new Error(`Missing required inputs:\n${inputList}\n\nPlease provide these inputs when calling run(workflow, inputs)`); + } + } else if (missingInputs.length > 0) { + // Missing inputs but can't prompt (browser or missing inputs) + const inputList = missingInputs.map(i => `- ${i.name} (${i.type}): ${i.description}`).join('\n'); + throw new Error(`Missing required inputs:\n${inputList}\n\nPlease provide these inputs when calling run(workflow, inputs)`); + } + + // Fill in default values for optional inputs + requiredInputs.forEach(input => { + if (!input.required && !(input.id in inputs) && !(input.name in inputs) && input.defaultValue !== undefined) { + inputs[input.id] = input.defaultValue; + } + }); + + // Execute the workflow + return this.execute(flowData, inputs); + } + + /** + * Get a simple description of what this workflow does + * @param {Object} flowData - Workflow JSON + * @returns {Object} Workflow description + */ + describe(flowData) { + try { + const flow = this.normalizeFlow(flowData); + const inputs = this.getRequiredInputs(flowData); + const outputNodes = flow.nodes.filter(node => node.type === 'output'); + const aiNodes = flow.nodes.filter(node => + node.type === 'llm' || + node.type === 'structured-llm' || + node.type === 'whisper-transcription' + ); + const customNodes = flow.nodes.filter(node => + this.customNodes.has(node.type) || + (flow.customNodes && flow.customNodes.some(cn => cn.type === node.type)) + ); + + return { + name: flow.name || 'Unnamed Workflow', + description: flow.description || 'No description provided', + inputs: inputs, + outputs: outputNodes.map(node => ({ + name: node.name || node.id, + description: node.data?.description || `Output from ${node.name || node.id}` + })), + nodeCount: flow.nodes.length, + hasAI: aiNodes.length > 0, + hasCustomNodes: customNodes.length > 0, + aiModels: aiNodes.map(node => node.data?.model || 'Unknown').filter(Boolean), + complexity: this.calculateComplexity(flow) + }; + } catch (error) { + throw new Error(`Failed to describe workflow: ${error.message}`); + } + } + + /** + * Calculate workflow complexity + * @private + */ + calculateComplexity(flow) { + const nodeCount = flow.nodes.length; + const connectionCount = flow.connections?.length || 0; + const hasAI = flow.nodes.some(n => n.type === 'llm' || n.type === 'structured-llm'); + const hasCustomNodes = flow.nodes.some(n => this.customNodes.has(n.type)); + + if (nodeCount <= 3) return 'Simple'; + if (nodeCount <= 7) return 'Medium'; + if (nodeCount <= 15 || hasAI || hasCustomNodes) return 'Complex'; + return 'Advanced'; + } + + // Private methods + + normalizeFlow(flowData) { + // Handle different export formats from Clara Studio + let flow; + if (flowData.format && flowData.flow) { + // SDK export format + flow = flowData.flow; + } else if (flowData.nodes && flowData.connections) { + // Direct flow format + flow = flowData; + } else { + throw new Error('Invalid flow format'); + } + + // Normalize connections format - convert React Flow format to SDK format + if (flow.connections && flow.connections.length > 0) { + flow.connections = flow.connections.map(conn => { + // If already in SDK format, return as-is + if (conn.sourceNodeId && conn.targetNodeId) { + return conn; + } + // Convert React Flow format to SDK format + return { + sourceNodeId: conn.source || conn.sourceNodeId, + targetNodeId: conn.target || conn.targetNodeId, + sourcePortId: conn.sourceHandle || conn.sourcePortId || 'output', + targetPortId: conn.targetHandle || conn.targetPortId || 'input' + }; + }); + } + + return flow; + } + + registerCustomNodes(customNodes) { + if (Array.isArray(customNodes)) { + customNodes.forEach(node => this.registerCustomNode(node)); + } + } + + validateFlow(flow) { + if (!flow.nodes || !Array.isArray(flow.nodes)) { + throw new Error('Flow must have nodes array'); + } + + if (!flow.connections || !Array.isArray(flow.connections)) { + throw new Error('Flow must have connections array'); + } + + if (flow.nodes.length === 0) { + throw new Error('Flow must have at least one node'); + } + + this.log(`📋 Flow validated: ${flow.nodes.length} nodes, ${flow.connections.length} connections`); + } + + async executeWorkflow(flow, inputs) { + // Get execution order using topological sort + const executionOrder = this.getExecutionOrder(flow.nodes, flow.connections); + this.log(`📊 Execution order: ${executionOrder.map(n => n.name || n.type).join(' → ')}`); + + // Initialize node outputs storage + const nodeOutputs = new Map(); + + // Set input node values + const inputNodes = flow.nodes.filter(node => node.type === 'input'); + for (const inputNode of inputNodes) { + const inputValue = inputs[inputNode.id] || inputs[inputNode.name] || inputNode.data?.value; + nodeOutputs.set(inputNode.id, { output: inputValue }); + this.log(`📥 Input [${inputNode.name || inputNode.id}]: ${this.truncateValue(inputValue)}`); + } + + // Execute nodes in order + for (const node of executionOrder) { + if (nodeOutputs.has(node.id)) continue; // Skip already executed nodes + + const nodeStartTime = Date.now(); + this.log(`▶️ Executing: ${node.name || node.type} (${node.type})`); + + try { + // Get inputs for this node + const nodeInputs = this.getNodeInputs(node, flow.connections, nodeOutputs); + + // Execute the node + const result = await this.executeNode(node, nodeInputs); + + // Store result + nodeOutputs.set(node.id, result); + + const nodeDuration = Date.now() - nodeStartTime; + this.log(`✅ Completed: ${node.name || node.type} (${nodeDuration}ms)`); + + } catch (error) { + const nodeDuration = Date.now() - nodeStartTime; + this.log(`❌ Failed: ${node.name || node.type} (${nodeDuration}ms) - ${error.message}`, 'error'); + throw new Error(`Node '${node.name || node.type}' failed: ${error.message}`); + } + } + + // Collect output node results + const results = {}; + const outputNodes = flow.nodes.filter(node => node.type === 'output'); + + for (const outputNode of outputNodes) { + const outputValue = nodeOutputs.get(outputNode.id); + results[outputNode.id] = outputValue; + results[outputNode.name || outputNode.id] = outputValue; + this.log(`📤 Output [${outputNode.name || outputNode.id}]: ${this.truncateValue(outputValue)}`); + } + + return results; + } + + getExecutionOrder(nodes, connections) { + // Topological sort for dependency-based execution order + const inDegree = new Map(); + const adjList = new Map(); + + // Initialize + for (const node of nodes) { + inDegree.set(node.id, 0); + adjList.set(node.id, []); + } + + // Build adjacency list and count incoming edges + for (const conn of connections) { + adjList.get(conn.sourceNodeId).push(conn.targetNodeId); + inDegree.set(conn.targetNodeId, inDegree.get(conn.targetNodeId) + 1); + } + + // Kahn's algorithm + const queue = []; + const result = []; + + // Start with nodes that have no incoming edges + for (const [nodeId, degree] of inDegree) { + if (degree === 0) { + queue.push(nodeId); + } + } + + while (queue.length > 0) { + const nodeId = queue.shift(); + const node = nodes.find(n => n.id === nodeId); + if (node) { + result.push(node); + } + + for (const targetId of adjList.get(nodeId)) { + inDegree.set(targetId, inDegree.get(targetId) - 1); + if (inDegree.get(targetId) === 0) { + queue.push(targetId); + } + } + } + + if (result.length !== nodes.length) { + throw new Error('Circular dependency detected in workflow'); + } + + return result; + } + + getNodeInputs(node, connections, nodeOutputs) { + const inputs = {}; + + // Find all connections that target this node + const incomingConnections = connections.filter(conn => conn.targetNodeId === node.id); + + for (const conn of incomingConnections) { + const sourceOutput = nodeOutputs.get(conn.sourceNodeId); + if (sourceOutput) { + // Get the correct output value + let outputValue; + if (sourceOutput[conn.sourcePortId]) { + outputValue = sourceOutput[conn.sourcePortId]; + } else if (sourceOutput.output !== undefined) { + outputValue = sourceOutput.output; + } else { + outputValue = sourceOutput; + } + + // Map to the target port ID directly (this is the most important mapping) + inputs[conn.targetPortId] = outputValue; + + // Also map common variations for backwards compatibility + if (conn.targetPortId === 'user') { + inputs.user = outputValue; + inputs.message = outputValue; + inputs.input = outputValue; + } + if (conn.targetPortId === 'system') { + inputs.system = outputValue; + } + if (conn.targetPortId === 'context') { + inputs.context = outputValue; + } + if (conn.targetPortId === 'text1') { + inputs.input1 = outputValue; + inputs.text1 = outputValue; + } + if (conn.targetPortId === 'text2') { + inputs.input2 = outputValue; + inputs.text2 = outputValue; + } + if (conn.targetPortId === 'input') { + inputs.input = outputValue; + } + + // Map by input port name if available + const inputPort = node.inputs?.find(input => input.id === conn.targetPortId); + if (inputPort) { + const inputName = inputPort.name?.toLowerCase(); + if (inputName) { + inputs[inputName] = outputValue; + } + } + } + } + + return inputs; + } + + async executeNode(node, inputs) { + // Check if it's a custom node + if (this.customNodes.has(node.type)) { + return this.executeCustomNode(node, inputs); + } + + // Execute built-in node types + switch (node.type) { + case 'input': + return { output: node.data?.value || '' }; + + case 'output': + return { output: inputs.input || Object.values(inputs)[0] }; + + case 'static-text': + return { + output: node.data?.text || node.data?.value || '', + text: node.data?.text || node.data?.value || '' + }; + + case 'combine-text': + const input1 = inputs.input1 || inputs.text1 || ''; + const input2 = inputs.input2 || inputs.text2 || ''; + const separator = node.data?.separator || ' '; + return { output: input1 + separator + input2 }; + + case 'json-parse': + try { + let jsonInput = inputs.input || inputs.json || '{}'; + let parsed; + + // Handle API response format { data: {...}, status: 200, ... } + if (jsonInput && typeof jsonInput === 'object' && 'data' in jsonInput && 'status' in jsonInput) { + // Extract the actual data from API response wrapper + parsed = jsonInput.data; + } else if (typeof jsonInput === 'string') { + // Parse JSON string + parsed = JSON.parse(jsonInput); + } else if (typeof jsonInput === 'object') { + // Already an object, use as-is + parsed = jsonInput; + } else { + // Try to parse as string + parsed = JSON.parse(String(jsonInput)); + } + + const field = node.data?.field || node.data?.path; + if (field) { + // Support dot notation for nested fields + const value = this.getNestedValue(parsed, field); + return { output: value }; + } + return { output: parsed }; + } catch (error) { + if (node.data?.failOnError !== false) { + throw new Error(`JSON Parse Error: ${error.message}`); + } + return { output: null }; + } + + case 'json-stringify': { + const jsonInput = inputs.input || inputs.json || Object.values(inputs)[0]; + const prettyPrint = node.data?.prettyPrint ?? true; + const indentSetting = Number(node.data?.indent ?? 2); + const indent = Number.isFinite(indentSetting) ? Math.min(Math.max(Math.round(indentSetting), 0), 8) : 2; + const fallback = node.data?.nullFallback ?? ''; + + if (jsonInput === null || jsonInput === undefined) { + return { output: fallback }; + } + + if (typeof jsonInput === 'string') { + return { output: jsonInput }; + } + + try { + const spacing = prettyPrint ? indent : 0; + const output = JSON.stringify(jsonInput, null, spacing || undefined); + return { output: output ?? fallback }; + } catch (error) { + return { output: String(jsonInput ?? fallback) }; + } + } + + case 'if-else': + const condition = inputs.condition !== undefined ? inputs.condition : inputs.input; + const trueValue = node.data?.trueValue || inputs.trueValue || condition; + const falseValue = node.data?.falseValue || inputs.falseValue || null; + + // Evaluate condition + let result; + if (node.data?.expression) { + try { + // Safe evaluation using Function constructor + const func = new Function('input', 'condition', `return ${node.data.expression}`); + result = func(condition, condition); + } catch (error) { + this.log(`If-Else expression error: ${error.message}`, 'warn'); + result = Boolean(condition); + } + } else { + result = Boolean(condition); + } + + return { + output: result ? trueValue : falseValue, + true: result ? trueValue : undefined, + false: result ? undefined : falseValue + }; + + case 'llm': + case 'llm-chat': + return this.executeLLMNode(node, inputs); + + case 'structured-llm': + return this.executeStructuredLLMNode(node, inputs); + + case 'agent-executor': + return this.executeAgentExecutorNode(node, inputs); + + case 'api-request': + return this.executeAPIRequestNode(node, inputs); + + default: + throw new Error(`Unknown node type: ${node.type}`); + } + } + + async executeCustomNode(node, inputs) { + const nodeDefinition = this.customNodes.get(node.type); + const properties = node.data || {}; + + try { + // Create execution context + const context = { + log: (message, data) => this.log(`[${node.name || node.type}] ${message}`, 'info', data), + warn: (message, data) => this.log(`[${node.name || node.type}] ${message}`, 'warn', data), + error: (message, data) => this.log(`[${node.name || node.type}] ${message}`, 'error', data) + }; + + // Execute custom node code + const func = new Function('inputs', 'properties', 'context', ` + ${nodeDefinition.executionCode} + if (typeof execute === 'function') { + return execute(inputs, properties, context); + } else { + throw new Error('Custom node must define an execute function'); + } + `); + + const result = await func(inputs, properties, context); + return result || {}; + + } catch (error) { + throw new Error(`Custom node execution failed: ${error.message}`); + } + } + + async executeLLMNode(node, inputs) { + // Basic LLM node implementation + const apiKey = node.data?.apiKey || process.env.OPENAI_API_KEY || ''; + const model = node.data?.model || 'gpt-3.5-turbo'; + let apiBaseUrl = (node.data?.apiBaseUrl && node.data?.apiBaseUrl.trim()) || process.env.OPENAI_API_BASE_URL || 'http://localhost:8091/v1'; + + // Convert localhost to host.docker.internal for Docker compatibility + apiBaseUrl = this.convertLocalhostForDocker(apiBaseUrl); + + const systemMessage = inputs.system || node.data?.systemMessage || ''; + const userMessage = inputs.user || inputs.input || inputs.message || ''; + + if (!userMessage) { + throw new Error('LLM node requires user message'); + } + + try { + const messages = []; + if (systemMessage) { + messages.push({ role: 'system', content: systemMessage }); + } + messages.push({ role: 'user', content: userMessage }); + + const headers = { + 'Content-Type': 'application/json' + }; + + if (apiKey && apiKey.trim()) { + headers['Authorization'] = `Bearer ${apiKey}`; + } else { + this.log('LLM node executing without API key. Ensure your API permits unauthenticated requests.', 'warn'); + } + + const response = await fetch(`${apiBaseUrl}/chat/completions`, { + method: 'POST', + headers, + body: JSON.stringify({ + model, + messages, + temperature: node.data?.temperature || 0.7, + max_tokens: node.data?.maxTokens || 1000 + }) + }); + + if (!response.ok) { + if (response.status === 401) { + throw new Error('Authentication failed - API key may be required or invalid'); + } else if (response.status === 403) { + throw new Error('Access forbidden - check API key permissions'); + } + throw new Error(`LLM API error: ${response.status} ${response.statusText}`); + } + + const data = await response.json(); + const output = data.choices?.[0]?.message?.content || ''; + + return { + output, + usage: data.usage, + model: data.model + }; + + } catch (error) { + throw new Error(`LLM execution failed: ${error.message}`); + } + } + + async executeStructuredLLMNode(node, inputs) { + // Similar to LLM but with structured output + const result = await this.executeLLMNode(node, inputs); + + try { + // Try to parse as JSON + const parsed = JSON.parse(result.output); + return { + output: parsed, + raw: result.output, + usage: result.usage, + model: result.model + }; + } catch (error) { + // If parsing fails, return raw output + return { + output: result.output, + usage: result.usage, + model: result.model + }; + } + } + + async executeAgentExecutorNode(node, inputs) { + // Agent executor node - executes autonomous agents with tool access + const instructions = inputs.instructions || node.data?.instructions || ''; + const context = inputs.context || node.data?.context || ''; + const attachments = inputs.attachments || node.data?.attachments || []; + + if (!instructions) { + throw new Error('Agent Executor requires instructions'); + } + + // Get agent configuration + const provider = node.data?.provider || process.env.AGENT_PROVIDER || ''; + const textModel = node.data?.textModel || 'gpt-4o-mini'; + const visionModel = node.data?.visionModel || textModel; + const codeModel = node.data?.codeModel || textModel; + const enabledMCPServers = node.data?.enabledMCPServers || []; + + // Construct agent execution request + const agentRequest = { + instructions, + context, + attachments, + config: { + provider, + textModel, + visionModel, + codeModel, + enabledMCPServers + } + }; + + // Try to execute via Clara Assistant service + const claraAssistantUrl = process.env.CLARA_ASSISTANT_URL || 'http://localhost:8069'; + + try { + // Try to call agent execution service + const response = await fetch(`${claraAssistantUrl}/api/execute-agent`, { + method: 'POST', + headers: { + 'Content-Type': 'application/json' + }, + body: JSON.stringify(agentRequest) + }); + + if (!response.ok) { + // Fallback: If agent service not available, use LLM as fallback + this.log('Agent service unavailable, using LLM fallback', 'warn'); + return await this.agentExecutorFallback(instructions, context, node); + } + + const result = await response.json(); + + return { + result: result.output || result.result || '', + toolResults: result.toolResults || [], + executionLog: result.log || '', + success: result.success !== false, + metadata: result.metadata || {} + }; + + } catch (error) { + // Fallback to LLM if agent service is not available + this.log(`Agent service error: ${error.message}, using LLM fallback`, 'warn'); + return await this.agentExecutorFallback(instructions, context, node); + } + } + + async agentExecutorFallback(instructions, context, node) { + // Enhanced fallback: Use LLM with MCP tool calling when agent service is not available + const apiKey = node.data?.apiKey || process.env.OPENAI_API_KEY || ''; + const model = node.data?.textModel?.split(':')[1] || node.data?.textModel || 'gpt-4o-mini'; + let apiBaseUrl = node.data?.apiBaseUrl || process.env.OPENAI_API_BASE_URL || 'http://localhost:8091/v1'; + const enabledMCPServers = node.data?.enabledMCPServers || []; + + // Convert localhost to host.docker.internal for Docker compatibility + apiBaseUrl = this.convertLocalhostForDocker(apiBaseUrl); + + // MCP Proxy URL (convert for Docker too) + let mcpProxyUrl = process.env.MCP_PROXY_URL || 'http://127.0.0.1:8092'; + mcpProxyUrl = this.convertLocalhostForDocker(mcpProxyUrl); + + try { + // Step 1: Fetch available MCP tools from enabled servers + const mcpTools = await this.fetchMCPTools(mcpProxyUrl, enabledMCPServers); + this.log(`Fetched ${mcpTools.length} MCP tools from ${enabledMCPServers.length} enabled servers`, 'info'); + + // Step 2: Convert MCP tools to OpenAI function calling format + const openAITools = this.convertMCPToolsToOpenAIFormat(mcpTools); + this.log(`Converted ${openAITools.length} tools to OpenAI format`, 'info'); + + // Step 3: Initialize conversation with system message and user instructions + const messages = [ + { + role: 'system', + content: 'You are an autonomous AI agent with access to powerful tools. Use the available tools to accomplish your tasks effectively. When a task is complete, provide a clear final answer without making additional tool calls.' + }, + { + role: 'user', + content: `Instructions: ${instructions}\n\n${context ? `Context: ${context}` : ''}` + } + ]; + + const executionLog = []; + const toolResults = []; + let iterations = 0; + const maxIterations = 10; // Prevent infinite loops + + // Step 4: Agentic loop - continue until LLM completes task + while (iterations < maxIterations) { + iterations++; + executionLog.push(`\n[Iteration ${iterations}]`); + + // Make LLM call with available tools + const llmResponse = await this.callLLMWithTools( + apiBaseUrl, + apiKey, + model, + messages, + openAITools.length > 0 ? openAITools : undefined + ); + + const message = llmResponse.choices?.[0]?.message; + if (!message) { + throw new Error('No message in LLM response'); + } + + // Add assistant message to conversation + messages.push(message); + + // Check if LLM wants to use tools + const toolCalls = message.tool_calls; + + if (!toolCalls || toolCalls.length === 0) { + // No more tool calls - task complete + const finalResult = message.content || ''; + executionLog.push(`Task completed after ${iterations} iterations`); + + return { + result: finalResult, + toolResults, + executionLog: executionLog.join('\n'), + success: true, + metadata: { + model: llmResponse.model, + usage: llmResponse.usage, + iterations, + fallback: true, + mcpEnabled: true + } + }; + } + + // Step 5: Execute all tool calls + executionLog.push(`Executing ${toolCalls.length} tool call(s)...`); + + for (const toolCall of toolCalls) { + try { + executionLog.push(` - Tool: ${toolCall.function.name}`); + + // Parse MCP tool call from OpenAI format + const mcpToolCall = this.parseOpenAIToolCallToMCP(toolCall); + + // Execute MCP tool via proxy + const toolResult = await this.executeMCPTool(mcpProxyUrl, mcpToolCall); + + // Format result for LLM + const resultText = this.formatMCPToolResult(toolResult); + executionLog.push(` Result: ${resultText.substring(0, 100)}...`); + + // Store for output + toolResults.push({ + tool: toolCall.function.name, + arguments: mcpToolCall.arguments, + result: toolResult + }); + + // Add tool result to conversation + messages.push({ + role: 'tool', + tool_call_id: toolCall.id, + content: resultText + }); + + } catch (toolError) { + executionLog.push(` Error: ${toolError.message}`); + + // Report error to LLM so it can adapt + messages.push({ + role: 'tool', + tool_call_id: toolCall.id, + content: `Error executing tool: ${toolError.message}` + }); + } + } + } + + // Max iterations reached + executionLog.push(`Warning: Max iterations (${maxIterations}) reached`); + const lastMessage = messages[messages.length - 1]; + const finalResult = lastMessage.role === 'assistant' ? lastMessage.content : 'Task incomplete - max iterations reached'; + + return { + result: finalResult, + toolResults, + executionLog: executionLog.join('\n'), + success: false, + metadata: { + model, + iterations, + fallback: true, + mcpEnabled: true, + warning: 'Max iterations reached' + } + }; + + } catch (error) { + throw new Error(`Agent execution failed: ${error.message}`); + } + } + + async fetchMCPTools(mcpProxyUrl, enabledServerNames) { + // Fetch available MCP tools from the MCP proxy service + try { + // Get list of servers + const serversResponse = await fetch(`${mcpProxyUrl}/api/servers`); + if (!serversResponse.ok) { + this.log(`MCP proxy not available at ${mcpProxyUrl}, skipping tool discovery`, 'warn'); + return []; + } + + const serversData = await serversResponse.json(); + const servers = serversData.servers || serversData || []; + + // Filter for enabled servers that are running + const enabledServers = servers.filter(server => + enabledServerNames.includes(server.name) && + server.isRunning && + server.status === 'running' + ); + + this.log(`Found ${enabledServers.length} running servers: ${enabledServers.map(s => s.name).join(', ')}`, 'info'); + + // Discover tools from each enabled server + const allTools = []; + for (const server of enabledServers) { + try { + // Use MCP protocol to discover tools from this server + const toolsResult = await this.executeMCPTool(mcpProxyUrl, { + name: 'tools/list', + arguments: {}, + server: server.name, + callId: `discover_${server.name}_${Date.now()}` + }); + + if (toolsResult.success && toolsResult.content) { + // Parse tools from result + const toolsList = this.parseToolsList(toolsResult.content); + toolsList.forEach(tool => { + tool.server = server.name; // Tag with server name + allTools.push(tool); + }); + this.log(`Discovered ${toolsList.length} tools from ${server.name}`, 'info'); + } + } catch (discoverError) { + this.log(`Failed to discover tools from ${server.name}: ${discoverError.message}`, 'warn'); + } + } + + return allTools; + } catch (error) { + this.log(`Failed to fetch MCP tools: ${error.message}`, 'warn'); + return []; + } + } + + parseToolsList(content) { + // Parse tools list from MCP result content + try { + if (!Array.isArray(content)) { + return []; + } + + // Look for JSON or text content with tools + const toolsContent = content.find(c => c.type === 'json' || c.type === 'text'); + if (!toolsContent) { + return []; + } + + let toolsData; + if (toolsContent.type === 'json' && toolsContent.data) { + toolsData = typeof toolsContent.data === 'string' ? + JSON.parse(toolsContent.data) : toolsContent.data; + } else if (toolsContent.type === 'text' && toolsContent.text) { + toolsData = JSON.parse(toolsContent.text); + } else { + return []; + } + + // Validate tools array + if (!Array.isArray(toolsData)) { + return []; + } + + return toolsData.filter(tool => tool && tool.name && tool.inputSchema); + } catch (error) { + this.log(`Failed to parse tools list: ${error.message}`, 'warn'); + return []; + } + } + + convertMCPToolsToOpenAIFormat(mcpTools) { + // Convert MCP tools to OpenAI function calling format + const openAITools = []; + + for (const tool of mcpTools) { + try { + // Fix and validate the input schema + const fixedParameters = this.fixOpenAISchema(tool.inputSchema || {}); + + const openAITool = { + type: 'function', + function: { + name: `mcp_${tool.server}_${tool.name}`, + description: `[MCP:${tool.server}] ${tool.description || tool.name}`, + parameters: fixedParameters + } + }; + + // Validate before adding + if (this.isValidOpenAITool(openAITool)) { + openAITools.push(openAITool); + } else { + this.log(`Skipping invalid tool: ${tool.server}:${tool.name}`, 'warn'); + } + } catch (error) { + this.log(`Failed to convert tool ${tool.server}:${tool.name}: ${error.message}`, 'warn'); + } + } + + return openAITools; + } + + fixOpenAISchema(schema) { + // Fix schema to be OpenAI-compatible + if (!schema || typeof schema !== 'object') { + return { + type: 'object', + properties: {}, + required: [] + }; + } + + // Deep clone + const fixed = JSON.parse(JSON.stringify(schema)); + + // Ensure required structure + if (!fixed.type) fixed.type = 'object'; + if (!fixed.properties) fixed.properties = {}; + if (!fixed.required) fixed.required = []; + + // Remove incompatible properties recursively + this.cleanSchemaForOpenAI(fixed); + + return fixed; + } + + cleanSchemaForOpenAI(schema) { + // Remove OpenAI-incompatible properties + if (!schema || typeof schema !== 'object') return; + + // Remove at any level + delete schema.$schema; + delete schema.additionalProperties; + delete schema.anyOf; + delete schema.oneOf; + delete schema.allOf; + delete schema.not; + delete schema.const; + delete schema.enum; + + // Handle properties + if (schema.properties) { + for (const propName in schema.properties) { + const prop = schema.properties[propName]; + + // Fix array properties - ensure items exist + if (prop && prop.type === 'array') { + if (!prop.items || typeof prop.items !== 'object' || !prop.items.type) { + prop.items = { type: 'object' }; + } + } + + // Recurse + if (prop && typeof prop === 'object') { + this.cleanSchemaForOpenAI(prop); + } + } + } + + // Handle array items + if (schema.items) { + this.cleanSchemaForOpenAI(schema.items); + } + } + + isValidOpenAITool(tool) { + // Validate OpenAI tool structure + try { + if (!tool || tool.type !== 'function' || !tool.function) { + return false; + } + + const func = tool.function; + if (!func.name || !func.description || !func.parameters) { + return false; + } + + return this.isValidParametersSchema(func.parameters); + } catch (error) { + return false; + } + } + + isValidParametersSchema(schema) { + // Validate parameters schema + if (!schema || typeof schema !== 'object' || schema.type !== 'object') { + return false; + } + + if (!schema.hasOwnProperty('properties') || typeof schema.properties !== 'object') { + return false; + } + + // Check each property + for (const propName in schema.properties) { + const prop = schema.properties[propName]; + if (!prop || typeof prop !== 'object' || !prop.type) { + return false; + } + + // Arrays must have items + if (prop.type === 'array') { + if (!prop.items || typeof prop.items !== 'object' || !prop.items.type) { + return false; + } + } + } + + return true; + } + + async callLLMWithTools(apiBaseUrl, apiKey, model, messages, tools) { + // Make LLM API call with optional tools + const headers = { + 'Content-Type': 'application/json' + }; + + if (apiKey && apiKey.trim()) { + headers['Authorization'] = `Bearer ${apiKey}`; + } + + const body = { + model, + messages, + temperature: 0.7, + max_tokens: 4000 + }; + + if (tools && tools.length > 0) { + body.tools = tools; + body.tool_choice = 'auto'; + } + + const response = await fetch(`${apiBaseUrl}/chat/completions`, { + method: 'POST', + headers, + body: JSON.stringify(body) + }); + + if (!response.ok) { + const errorText = await response.text(); + throw new Error(`LLM API error: ${response.status} ${response.statusText} - ${errorText}`); + } + + return await response.json(); + } + + parseOpenAIToolCallToMCP(toolCall) { + // Parse OpenAI tool call back to MCP format + const funcName = toolCall.function.name; + + if (!funcName.startsWith('mcp_')) { + throw new Error(`Invalid MCP tool name: ${funcName}`); + } + + const nameParts = funcName.replace('mcp_', '').split('_'); + if (nameParts.length < 2) { + throw new Error(`Invalid MCP tool name format: ${funcName}`); + } + + const server = nameParts[0]; + const toolName = nameParts.slice(1).join('_'); + + // Parse arguments + let parsedArgs = {}; + try { + const argsString = toolCall.function.arguments || '{}'; + parsedArgs = typeof argsString === 'string' ? JSON.parse(argsString) : argsString; + } catch (error) { + this.log(`Failed to parse tool arguments: ${error.message}`, 'warn'); + } + + return { + name: toolName, + arguments: parsedArgs, + server: server, + callId: toolCall.id || `mcp_${Date.now()}_${Math.random().toString(36).substr(2, 9)}` + }; + } + + async executeMCPTool(mcpProxyUrl, mcpToolCall) { + // Execute MCP tool via the proxy service + const response = await fetch(`${mcpProxyUrl}/api/tools/execute`, { + method: 'POST', + headers: { + 'Content-Type': 'application/json' + }, + body: JSON.stringify(mcpToolCall) + }); + + if (!response.ok) { + throw new Error(`MCP tool execution failed: ${response.status} ${response.statusText}`); + } + + const result = await response.json(); + + // Extract actual result from wrapper if needed + return result.result || result; + } + + formatMCPToolResult(toolResult) { + // Format MCP tool result for LLM consumption + if (!toolResult) { + return 'No result'; + } + + if (toolResult.success === false) { + return `Error: ${toolResult.error || 'Tool execution failed'}`; + } + + if (toolResult.content && Array.isArray(toolResult.content)) { + // Extract text from content array + const textParts = toolResult.content + .filter(c => c.type === 'text' && c.text) + .map(c => c.text); + + if (textParts.length > 0) { + return textParts.join('\n\n'); + } + + // Try JSON content + const jsonParts = toolResult.content + .filter(c => c.type === 'json' && c.data) + .map(c => typeof c.data === 'string' ? c.data : JSON.stringify(c.data, null, 2)); + + if (jsonParts.length > 0) { + return jsonParts.join('\n\n'); + } + } + + // Fallback to JSON stringify + return JSON.stringify(toolResult, null, 2); + } + + async executeAPIRequestNode(node, inputs) { + const url = inputs.url || node.data?.url; + const method = node.data?.method || 'GET'; + const headers = { ...node.data?.headers, ...inputs.headers }; + const body = inputs.body || node.data?.body; + + if (!url) { + throw new Error('API Request node requires URL'); + } + + try { + const options = { + method: method.toUpperCase(), + headers: { + 'Content-Type': 'application/json', + ...headers + } + }; + + if (body && method.toUpperCase() !== 'GET') { + options.body = typeof body === 'string' ? body : JSON.stringify(body); + } + + const response = await fetch(url, options); + + let responseData; + const contentType = response.headers.get('content-type'); + + if (contentType?.includes('application/json')) { + responseData = await response.json(); + } else { + responseData = await response.text(); + } + + return { + output: responseData, + status: response.status, + statusText: response.statusText, + headers: Object.fromEntries(response.headers.entries()) + }; + + } catch (error) { + throw new Error(`API Request failed: ${error.message}`); + } + } + + getNestedValue(obj, path) { + return path.split('.').reduce((current, key) => current?.[key], obj); + } + + truncateValue(value) { + const str = typeof value === 'string' ? value : JSON.stringify(value); + return str.length > 100 ? str.substring(0, 100) + '...' : str; + } + + convertLocalhostForDocker(url) { + // Convert localhost/127.0.0.1 to host.docker.internal for Docker compatibility + // This allows services running inside Docker to access services on the host machine + if (!url || typeof url !== 'string') return url; + + // Check if we're running in Docker (common indicators) + const isDocker = process.env.DOCKER_CONTAINER === 'true' || + process.env.NODE_ENV === 'production' || + (typeof process !== 'undefined' && process.platform === 'linux' && + (process.env.HOSTNAME?.includes('docker') || + process.env.HOSTNAME?.length === 12)); // Docker container hostnames are 12 chars + + if (isDocker) { + // Replace localhost and 127.0.0.1 with host.docker.internal + return url + .replace(/localhost/g, 'host.docker.internal') + .replace(/127\.0\.0\.1/g, 'host.docker.internal'); + } + + return url; + } + + log(message, level = 'info', data = null) { + if (!this.config.enableLogging) return; + + const logEntry = { + timestamp: new Date().toISOString(), + level, + message, + data + }; + + this.executionLogs.push(logEntry); + + if (typeof console !== 'undefined') { + const logMethod = console[level] || console.log; + logMethod(`[Clara SDK] ${message}`, data || ''); + } + } +} + +// Utility functions for browser usage +const BrowserUtils = { + // Download flow as JSON file + downloadFlow(flowData, filename = 'workflow.json') { + if (typeof document === 'undefined') { + throw new Error('downloadFlow is only available in browser environment'); + } + + const blob = new Blob([JSON.stringify(flowData, null, 2)], { type: 'application/json' }); + const url = URL.createObjectURL(blob); + const a = document.createElement('a'); + a.href = url; + a.download = filename; + a.click(); + URL.revokeObjectURL(url); + }, + + // Load flow from file input + async loadFlowFromFile(file) { + return new Promise((resolve, reject) => { + const reader = new FileReader(); + reader.onload = (e) => { + try { + const flowData = JSON.parse(e.target.result); + resolve(flowData); + } catch (error) { + reject(new Error('Invalid JSON file')); + } + }; + reader.onerror = () => reject(new Error('File reading failed')); + reader.readAsText(file); + }); + }, + + // Get browser info + getBrowserInfo() { + if (typeof navigator === 'undefined') return null; + + return { + userAgent: navigator.userAgent, + platform: navigator.platform, + language: navigator.language, + cookieEnabled: navigator.cookieEnabled, + onLine: navigator.onLine + }; + }, + + // Check if running in browser + isBrowser() { + return typeof window !== 'undefined' && typeof document !== 'undefined'; + } +}; + +// Export main classes and utilities +export { ClaraFlowRunner, BrowserUtils }; + +// Default export for CommonJS compatibility +export default ClaraFlowRunner; \ No newline at end of file diff --git a/sdk/src/index.ts b/sdk/src/index.ts new file mode 100644 index 00000000..a49c7d6d --- /dev/null +++ b/sdk/src/index.ts @@ -0,0 +1,3 @@ +export { ClaraFlowRunner } from './ClaraFlowRunner'; +export { FlowExecutor, ExecutionContext, FlowExecutorOptions } from '../../src/shared/FlowEngine'; +export { ExecutionLog, FlowNode, Connection } from '../../src/types/agent/types'; \ No newline at end of file diff --git a/sdk/src/logger.js b/sdk/src/logger.js new file mode 100644 index 00000000..fe7ab02f --- /dev/null +++ b/sdk/src/logger.js @@ -0,0 +1,351 @@ +/** + * Logger - Handles logging for the SDK + */ + +export class Logger { + constructor(enabled = false, logLevel = 'info') { + this.enabled = enabled; + this.logLevel = logLevel; + this.logs = []; + this.maxLogs = 1000; // Prevent memory issues + + // Log levels in order of severity + this.levels = { + debug: 0, + info: 1, + warn: 2, + error: 3 + }; + + this.currentLevel = this.levels[logLevel] || this.levels.info; + } + + /** + * Log a debug message + * @param {string} message - Log message + * @param {Object} data - Additional data + */ + debug(message, data = {}) { + this.log('debug', message, data); + } + + /** + * Log an info message + * @param {string} message - Log message + * @param {Object} data - Additional data + */ + info(message, data = {}) { + this.log('info', message, data); + } + + /** + * Log a warning message + * @param {string} message - Log message + * @param {Object} data - Additional data + */ + warn(message, data = {}) { + this.log('warn', message, data); + } + + /** + * Log an error message + * @param {string} message - Log message + * @param {Object} data - Additional data + */ + error(message, data = {}) { + this.log('error', message, data); + } + + /** + * Internal log method + * @param {string} level - Log level + * @param {string} message - Log message + * @param {Object} data - Additional data + */ + log(level, message, data = {}) { + const levelValue = this.levels[level] || this.levels.info; + + // Skip if log level is below current threshold + if (levelValue < this.currentLevel) { + return; + } + + const logEntry = { + id: this.generateLogId(), + timestamp: new Date().toISOString(), + level: level, + message: message, + data: this.sanitizeData(data) + }; + + // Add to internal log storage + this.logs.push(logEntry); + + // Trim logs if exceeding max + if (this.logs.length > this.maxLogs) { + this.logs = this.logs.slice(-this.maxLogs); + } + + // Output to console if enabled + if (this.enabled) { + this.outputToConsole(logEntry); + } + } + + /** + * Output log entry to console + * @param {Object} logEntry - Log entry to output + */ + outputToConsole(logEntry) { + const { level, message, data, timestamp } = logEntry; + const prefix = `[${timestamp}] [${level.toUpperCase()}]`; + + // Choose appropriate console method + const consoleMethod = console[level] || console.log; + + if (Object.keys(data).length > 0) { + consoleMethod(`${prefix} ${message}`, data); + } else { + consoleMethod(`${prefix} ${message}`); + } + } + + /** + * Sanitize data for logging (remove sensitive info, circular refs) + * @param {any} data - Data to sanitize + * @returns {any} Sanitized data + */ + sanitizeData(data) { + if (data === null || data === undefined) { + return data; + } + + if (typeof data !== 'object') { + return data; + } + + try { + // Handle circular references by converting to JSON and back + const jsonString = JSON.stringify(data, (key, value) => { + // Remove sensitive keys + const sensitiveKeys = ['password', 'apiKey', 'secret', 'token', 'authorization']; + if (sensitiveKeys.some(sensitive => key.toLowerCase().includes(sensitive))) { + return '[REDACTED]'; + } + + // Handle functions + if (typeof value === 'function') { + return '[Function]'; + } + + return value; + }); + + return JSON.parse(jsonString); + } catch (error) { + // If JSON serialization fails, return a safe representation + return { error: 'Could not serialize data', type: typeof data }; + } + } + + /** + * Generate unique log ID + * @returns {string} Unique log ID + */ + generateLogId() { + return `log-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`; + } + + /** + * Get all logs + * @returns {Array} Array of log entries + */ + getLogs() { + return [...this.logs]; + } + + /** + * Get logs filtered by level + * @param {string} level - Log level to filter by + * @returns {Array} Filtered log entries + */ + getLogsByLevel(level) { + return this.logs.filter(log => log.level === level); + } + + /** + * Get logs within a time range + * @param {Date} startTime - Start time + * @param {Date} endTime - End time + * @returns {Array} Filtered log entries + */ + getLogsByTimeRange(startTime, endTime) { + return this.logs.filter(log => { + const logTime = new Date(log.timestamp); + return logTime >= startTime && logTime <= endTime; + }); + } + + /** + * Search logs by message content + * @param {string} searchTerm - Term to search for + * @returns {Array} Matching log entries + */ + searchLogs(searchTerm) { + const lowerSearchTerm = searchTerm.toLowerCase(); + return this.logs.filter(log => + log.message.toLowerCase().includes(lowerSearchTerm) || + JSON.stringify(log.data).toLowerCase().includes(lowerSearchTerm) + ); + } + + /** + * Clear all logs + */ + clearLogs() { + this.logs = []; + if (this.enabled) { + console.log('[LOGGER] Logs cleared'); + } + } + + /** + * Set log level + * @param {string} level - New log level + */ + setLogLevel(level) { + if (this.levels.hasOwnProperty(level)) { + this.logLevel = level; + this.currentLevel = this.levels[level]; + this.info(`Log level set to: ${level}`); + } else { + this.warn(`Invalid log level: ${level}. Available levels: ${Object.keys(this.levels).join(', ')}`); + } + } + + /** + * Enable or disable logging + * @param {boolean} enabled - Whether to enable logging + */ + setEnabled(enabled) { + this.enabled = Boolean(enabled); + if (this.enabled) { + this.info('Logging enabled'); + } + } + + /** + * Get logging statistics + * @returns {Object} Logging statistics + */ + getStats() { + const stats = { + totalLogs: this.logs.length, + enabled: this.enabled, + logLevel: this.logLevel, + maxLogs: this.maxLogs, + byLevel: {} + }; + + // Count logs by level + for (const level of Object.keys(this.levels)) { + stats.byLevel[level] = this.logs.filter(log => log.level === level).length; + } + + return stats; + } + + /** + * Export logs to JSON string + * @param {Object} options - Export options + * @returns {string} JSON string of logs + */ + exportLogs(options = {}) { + const { + level = null, + startTime = null, + endTime = null, + maxEntries = null + } = options; + + let logsToExport = [...this.logs]; + + // Apply filters + if (level) { + logsToExport = logsToExport.filter(log => log.level === level); + } + + if (startTime || endTime) { + logsToExport = logsToExport.filter(log => { + const logTime = new Date(log.timestamp); + if (startTime && logTime < startTime) return false; + if (endTime && logTime > endTime) return false; + return true; + }); + } + + if (maxEntries && maxEntries > 0) { + logsToExport = logsToExport.slice(-maxEntries); + } + + return JSON.stringify({ + exportedAt: new Date().toISOString(), + totalEntries: logsToExport.length, + filters: { level, startTime, endTime, maxEntries }, + logs: logsToExport + }, null, 2); + } + + /** + * Import logs from JSON string + * @param {string} jsonString - JSON string containing logs + * @param {boolean} merge - Whether to merge with existing logs + */ + importLogs(jsonString, merge = true) { + try { + const importData = JSON.parse(jsonString); + + if (!importData.logs || !Array.isArray(importData.logs)) { + throw new Error('Invalid log format: missing logs array'); + } + + if (merge) { + this.logs.push(...importData.logs); + + // Trim if exceeding max + if (this.logs.length > this.maxLogs) { + this.logs = this.logs.slice(-this.maxLogs); + } + } else { + this.logs = importData.logs.slice(-this.maxLogs); + } + + this.info(`Imported ${importData.logs.length} log entries`, { + merged: merge, + totalLogs: this.logs.length + }); + + } catch (error) { + this.error('Failed to import logs', { error: error.message }); + throw error; + } + } + + /** + * Create a child logger with a prefix + * @param {string} prefix - Prefix for all log messages + * @returns {Logger} Child logger instance + */ + createChild(prefix) { + const childLogger = new Logger(this.enabled, this.logLevel); + childLogger.logs = this.logs; // Share the same log array + + // Override log method to add prefix + const originalLog = childLogger.log.bind(childLogger); + childLogger.log = (level, message, data) => { + originalLog(level, `[${prefix}] ${message}`, data); + }; + + return childLogger; + } +} \ No newline at end of file diff --git a/sdk/src/nodeExecutor.js b/sdk/src/nodeExecutor.js new file mode 100644 index 00000000..29f6568f --- /dev/null +++ b/sdk/src/nodeExecutor.js @@ -0,0 +1,1886 @@ +/** + * Node Executor - Handles execution of all node types + * Updated to use real execution logic instead of mocks + */ + +export class NodeExecutor { + constructor(logger, customNodeManager) { + this.logger = logger; + this.customNodeManager = customNodeManager; + + // Map of built-in node executors + this.builtinNodes = { + 'input': this.executeInputNode.bind(this), + 'output': this.executeOutputNode.bind(this), + 'llm': this.executeLLMNode.bind(this), + 'structured-llm': this.executeStructuredLLMNode.bind(this), + 'json-parse': this.executeJsonParseNode.bind(this), + 'json-stringify': this.executeJsonStringifyNode.bind(this), + 'if-else': this.executeIfElseNode.bind(this), + 'image-input': this.executeImageInputNode.bind(this), + 'pdf-input': this.executePDFInputNode.bind(this), + 'api-request': this.executeAPIRequestNode.bind(this), + 'combine-text': this.executeCombineTextNode.bind(this), + 'static-text': this.executeStaticTextNode.bind(this), + 'file-upload': this.executeFileUploadNode.bind(this), + 'whisper-transcription': this.executeWhisperTranscriptionNode.bind(this), + 'speech-to-text': this.executeSpeechToTextNode.bind(this), + 'agent-executor': this.executeAgentExecutorNode.bind(this), + 'notebook-writer': this.executeNotebookWriterNode.bind(this), + }; + } + + /** + * Execute a node based on its type + * @param {Object} nodeData - Node configuration + * @param {Object} inputs - Input values + * @returns {Promise} Execution result + */ + async executeNode(nodeData, inputs = {}) { + const { type, name, data = {} } = nodeData; + + this.logger.debug(`Executing node: ${name} (${type})`, { inputs, data }); + + try { + let result; + + // Check if it's a custom node first + if (this.customNodeManager.hasNode(type)) { + result = await this.customNodeManager.executeNode(type, inputs, data.properties || {}); + } else if (this.builtinNodes[type]) { + result = await this.builtinNodes[type](inputs, data); + } else { + throw new Error(`Unknown node type: ${type}`); + } + + this.logger.debug(`Node execution completed: ${name}`, { result }); + return result; + + } catch (error) { + this.logger.error(`Node execution failed: ${name}`, { error: error.message }); + throw new Error(`Node '${name}' execution failed: ${error.message}`); + } + } + + /** + * Execute Input Node + * Uses the real logic from the Agent Builder UI + */ + async executeInputNode(inputs, data) { + const { inputType = 'string', defaultValue = '', value } = data; + + // Priority: provided input value > stored node value > default value + let inputValue = inputs.value !== undefined ? inputs.value : (value !== undefined ? value : defaultValue); + + // Handle empty string for string types + if (inputType === 'string' && inputValue === '') { + inputValue = defaultValue || ''; + } + + // Type conversion based on inputType (real UI logic) + switch (inputType) { + case 'number': + const numValue = Number(inputValue); + return isNaN(numValue) ? (defaultValue ? Number(defaultValue) : 0) : numValue; + case 'boolean': + if (typeof inputValue === 'boolean') return inputValue; + if (typeof inputValue === 'string') { + return inputValue.toLowerCase() === 'true' || inputValue === '1'; + } + return Boolean(inputValue); + case 'json': + try { + return typeof inputValue === 'string' ? JSON.parse(inputValue) : inputValue; + } catch { + this.logger.warn('Failed to parse JSON input, returning as string'); + return inputValue; + } + default: + return String(inputValue || ''); + } + } + + /** + * Execute Output Node + * Uses the real logic from the Agent Builder UI + */ + async executeOutputNode(inputs, data) { + // Output nodes pass through their input value + const inputValue = inputs.input || Object.values(inputs)[0]; + + // If there's a format specified in the data, apply it + if (data.format && inputValue !== undefined) { + switch (data.format) { + case 'json': + try { + return typeof inputValue === 'object' ? inputValue : JSON.parse(String(inputValue)); + } catch { + return inputValue; + } + case 'string': + return String(inputValue); + case 'number': + return Number(inputValue); + default: + return inputValue; + } + } + + return inputValue; + } + + /** + * Execute LLM Node + * Uses the real API call logic from the Agent Builder UI (with fallback for SDK mode) + */ + async executeLLMNode(inputs, data) { + const { + apiBaseUrl = process.env.OPENAI_API_BASE_URL || 'http://localhost:8091/v1', + apiKey = '', + model = 'gpt-3.5-turbo', + temperature = 0.7, + maxTokens = 1000, + systemPrompt = '' + } = data; + + // Extract inputs according to the real UI logic + const systemMessage = inputs.system || systemPrompt || ''; + const userMessage = inputs.user || inputs.input || ''; + const context = inputs.context || ''; + const memory = inputs.memory || []; + const imageData = inputs.image || ''; + + if (!userMessage) { + throw new Error('User message is required for LLM node'); + } + + try { + const messages = []; + + // Add system message if provided (real UI logic) + if (systemMessage) { + messages.push({ role: 'system', content: systemMessage }); + } + + // Add memory/history if provided + if (Array.isArray(memory) && memory.length > 0) { + messages.push(...memory); + } + + // Add context if provided + if (context) { + messages.push({ role: 'system', content: `Context: ${context}` }); + } + + // Add user message with optional image (real UI logic) + const userMessageContent = []; + userMessageContent.push({ type: 'text', text: userMessage }); + + if (imageData) { + let base64String = ''; + + if (typeof imageData === 'string') { + base64String = imageData; + } else if (typeof imageData === 'object' && imageData.base64) { + base64String = imageData.base64; + } + + if (base64String) { + userMessageContent.push({ + type: 'image_url', + image_url: { url: `data:image/jpeg;base64,${base64String}` } + }); + } + } + + messages.push({ + role: 'user', + content: userMessageContent.length === 1 ? userMessage : userMessageContent + }); + + // Make actual API call (real UI logic) + const headers = { + 'Content-Type': 'application/json' + }; + + if (apiKey && apiKey.trim()) { + headers['Authorization'] = `Bearer ${apiKey}`; + } else { + this.logger.warn('LLM node: executing without API key. Ensure your API allows unauthenticated access.'); + } + + const response = await fetch(`${apiBaseUrl}/chat/completions`, { + method: 'POST', + headers, + body: JSON.stringify({ + model, + messages, + temperature, + max_tokens: maxTokens + }) + }); + + if (!response.ok) { + if (response.status === 401) { + throw new Error('Authentication failed - API key may be required or invalid'); + } else if (response.status === 403) { + throw new Error('Access forbidden - check API key permissions'); + } else { + throw new Error(`LLM API Error: ${response.status} ${response.statusText}`); + } + } + + const responseData = await response.json(); + + return { + response: responseData.choices?.[0]?.message?.content || '', + usage: responseData.usage || {}, + model: model + }; + + } catch (error) { + throw new Error(`LLM execution failed: ${error instanceof Error ? error.message : 'Unknown error'}`); + } + } + + /** + * Execute JSON Parse Node + * Uses the real logic from the Agent Builder UI + */ + async executeJsonParseNode(inputs, data) { + const { extractField = '', failOnError = false } = data; + let inputData = inputs.input || inputs.json || Object.values(inputs)[0] || ''; + + try { + let parsedData; + + // Handle API response format { data: {...}, status: 200, ... } + if (inputData && typeof inputData === 'object' && 'data' in inputData && 'status' in inputData) { + // Extract the actual data from API response wrapper + parsedData = inputData.data; + } else if (typeof inputData === 'string') { + // Parse JSON string (real UI logic) + if (!inputData.trim()) { + if (failOnError) { + throw new Error('Empty input provided to JSON parser'); + } + return null; + } + parsedData = JSON.parse(inputData); + } else if (typeof inputData === 'object') { + // Already an object, use as-is + parsedData = inputData; + } else { + // Try to parse as string + parsedData = JSON.parse(String(inputData)); + } + + // Extract specific field if specified (real UI logic) + if (extractField && extractField.trim()) { + const result = this.extractJsonPath(parsedData, extractField.trim()); + if (result === undefined && failOnError) { + throw new Error(`Field '${extractField}' not found in JSON data`); + } + return result; + } + + return parsedData; + + } catch (error) { + if (failOnError) { + throw error; + } else { + this.logger.warn(`JSON parsing failed: ${error.message}, returning original input`); + return inputData; + } + } + } + + /** + * Execute JSON Stringify Node + * Converts objects into string output for downstream text nodes + */ + async executeJsonStringifyNode(inputs, data = {}) { + const { + prettyPrint = true, + indent = 2, + nullFallback = '' + } = data; + + const inputValue = inputs.input ?? inputs.json ?? Object.values(inputs)[0]; + + if (inputValue === null || inputValue === undefined) { + return nullFallback; + } + + if (typeof inputValue === 'string') { + return inputValue; + } + + try { + const normalizedIndent = Math.min(Math.max(Number.isFinite(indent) ? Math.round(indent) : 2, 0), 8); + const spacing = prettyPrint ? normalizedIndent : 0; + const stringified = JSON.stringify(inputValue, null, spacing || undefined); + return stringified ?? nullFallback; + } catch (error) { + this.logger.warn('JSON stringify failed, falling back to toString()', { error: error.message }); + try { + return String(inputValue); + } catch (stringError) { + this.logger.error('String coercion failed', { error: stringError.message }); + return nullFallback; + } + } + } + + /** + * Execute If/Else Node + * Uses the real logic from the Agent Builder UI + */ + async executeIfElseNode(inputs, data) { + // Accept both 'expression' and 'condition' for compatibility + const expression = data.expression || data.condition || 'input > 0'; + const { trueValue = '', falseValue = '' } = data; + const inputValue = inputs.input || Object.values(inputs)[0]; + + try { + // Evaluate condition using real UI logic + const conditionResult = this.evaluateExpression(expression, inputValue); + + // Return the appropriate value based on condition (real UI logic) + if (conditionResult) { + return trueValue || inputValue; + } else { + return falseValue || inputValue; + } + + } catch (error) { + this.logger.warn(`If/Else expression evaluation failed: ${error.message}, returning false branch`); + return falseValue || inputValue; + } + } + + /** + * Execute Image Input Node + * Uses the real logic from the Agent Builder UI + */ + async executeImageInputNode(inputs, data) { + const { + maxWidth = 1024, + maxHeight = 1024, + quality = 0.8, + imageFile = '' + } = data; + + const imageData = inputs.image || inputs.input || imageFile; + + if (!imageData) { + return { + base64: '', + metadata: { + width: 0, + height: 0, + type: '', + size: 0 + } + }; + } + + // Real UI logic for image processing + return { + base64: imageData, // In real implementation, this would process/resize the image + metadata: { + width: maxWidth, + height: maxHeight, + type: 'image/jpeg', + size: typeof imageData === 'string' ? imageData.length : 0 + } + }; + } + + /** + * Extract value from JSON using dot notation path (real UI logic) + * @param {Object} obj - JSON object + * @param {string} path - Dot notation path (e.g., "user.name") + * @returns {any} Extracted value + */ + extractJsonPath(obj, path) { + return path.split('.').reduce((current, key) => { + return current && current[key] !== undefined ? current[key] : undefined; + }, obj); + } + + /** + * Evaluate expression for If/Else node (real UI logic) + * @param {string} expression - Expression to evaluate + * @param {any} inputValue - Input value to use in expression + * @returns {boolean} Expression result + */ + evaluateExpression(expression, inputValue) { + if (!expression || typeof expression !== 'string') { + return false; + } + + try { + // Create safe evaluation context (real UI logic) + const safeGlobals = { + input: inputValue, + value: inputValue, + Math, + Number, + String, + Boolean, + Array, + Object, + Date, + typeof: (val) => typeof val, + isNaN, + isFinite + }; + + // Replace input references in expression + let safeExpression = expression + .replace(/\binput\b/g, 'safeGlobals.input') + .replace(/\bvalue\b/g, 'safeGlobals.value'); + + // Basic security check - only allow safe patterns + const allowedPattern = /^[a-zA-Z0-9_$.\s+\-*/%()><=!&|"'`]+$/; + if (!allowedPattern.test(safeExpression)) { + throw new Error('Expression contains disallowed characters'); + } + + // Use Function constructor for safer evaluation (real UI logic) + const evaluator = new Function('safeGlobals', ` + with (safeGlobals) { + return ${safeExpression}; + } + `); + + return Boolean(evaluator(safeGlobals)); + + } catch (error) { + this.logger.warn(`Expression evaluation failed: ${error.message}`); + return false; + } + } + + /** + * Check if a node type is supported + * @param {string} nodeType - Node type to check + * @returns {boolean} True if supported + */ + isNodeTypeSupported(nodeType) { + return this.builtinNodes.hasOwnProperty(nodeType); + } + + /** + * Get list of supported built-in node types + * @returns {Array} Supported node types + */ + getSupportedNodeTypes() { + return Object.keys(this.builtinNodes); + } + + /** + * Add a custom built-in node executor + * @param {string} nodeType - Node type + * @param {Function} executor - Executor function + */ + addBuiltinNode(nodeType, executor) { + this.builtinNodes[nodeType] = executor; + this.logger.info(`Added built-in node type: ${nodeType}`); + } + + /** + * Remove a built-in node executor + * @param {string} nodeType - Node type to remove + */ + removeBuiltinNode(nodeType) { + delete this.builtinNodes[nodeType]; + this.logger.info(`Removed built-in node type: ${nodeType}`); + } + + /** + * Execute Structured LLM Node + * Uses structured response format when available, falls back to prompt-based JSON generation + */ + async executeStructuredLLMNode(inputs, data) { + const { + apiBaseUrl = 'https://api.openai.com/v1', + apiKey = '', + model = 'gpt-4o-mini', + temperature = 0.7, + maxTokens = 1000, + useStructuredOutput = 'auto' // 'auto', 'force', 'disable' + } = data; + + const prompt = inputs.prompt || ''; + const jsonExample = inputs.jsonExample || ''; + const context = inputs.context || ''; + + if (!prompt) { + throw new Error('Prompt is required for Structured LLM node'); + } + + if (!jsonExample) { + throw new Error('JSON Example is required for Structured LLM node'); + } + + try { + // Parse the JSON example to validate it + let exampleObject; + try { + exampleObject = JSON.parse(jsonExample); + } catch (parseError) { + throw new Error('Invalid JSON example format. Please provide valid JSON.'); + } + + // Determine if we should use structured output + const shouldUseStructuredOutput = this.shouldUseStructuredOutput(apiBaseUrl, model, useStructuredOutput); + + // Build the messages + const messages = []; + + let systemPrompt = `You are a helpful assistant that generates structured JSON data based on user prompts and examples. You must respond with valid JSON that matches the provided structure exactly.`; + + if (context) { + systemPrompt += `\n\nAdditional context: ${context}`; + } + + if (!shouldUseStructuredOutput) { + systemPrompt += `\n\nIMPORTANT: You must respond ONLY with valid JSON. Do not include any explanation, markdown formatting, or additional text. Your entire response must be parseable JSON.`; + } + + messages.push({ role: 'system', content: systemPrompt }); + + const userMessage = `${prompt}\n\nPlease generate JSON data that follows this exact structure:\n${jsonExample}\n\nRespond only with valid JSON that matches this structure.`; + messages.push({ role: 'user', content: userMessage }); + + // Prepare the request body + const requestBody = { + model, + messages, + temperature, + max_tokens: maxTokens + }; + + // Add structured output format if supported + if (shouldUseStructuredOutput) { + const schema = this.generateSchemaFromExample(exampleObject); + requestBody.response_format = { + type: "json_schema", + json_schema: { + name: "structured_output", + description: "Generated structured data based on the provided example", + schema: schema, + strict: true + } + }; + } else { + // For non-OpenAI models, use JSON mode if available + requestBody.response_format = { type: "json_object" }; + } + + // Make API call + const headers = { + 'Content-Type': 'application/json' + }; + if (apiKey && apiKey.trim()) { + headers['Authorization'] = `Bearer ${apiKey}`; + } else { + this.logger.warn('Structured LLM node: executing without API key'); + } + + const response = await fetch(`${apiBaseUrl}/chat/completions`, { + method: 'POST', + headers, + body: JSON.stringify(requestBody) + }); + + if (!response.ok) { + // If structured output failed, try again without it + if (shouldUseStructuredOutput && response.status === 400) { + this.logger.warn('Structured output not supported, falling back to prompt-based JSON generation'); + return await this.executeStructuredLLMWithFallback(inputs, data); + } + throw new Error(`Structured LLM API Error: ${response.status} ${response.statusText}`); + } + + const responseData = await response.json(); + const rawResponse = responseData.choices?.[0]?.message?.content || ''; + + if (!rawResponse) { + throw new Error('No response received from API'); + } + + // Parse the JSON response + let jsonOutput; + try { + // Clean the response (remove markdown formatting if present) + const cleanedResponse = this.cleanJsonResponse(rawResponse); + jsonOutput = JSON.parse(cleanedResponse); + } catch (parseError) { + // If parsing fails, try to extract JSON from the response + const extractedJson = this.extractJsonFromResponse(rawResponse); + if (extractedJson) { + jsonOutput = extractedJson; + } else { + throw new Error(`API returned invalid JSON. Raw response: ${rawResponse}`); + } + } + + return { + jsonOutput, + rawResponse, + usage: responseData.usage || {}, + method: shouldUseStructuredOutput ? 'structured_output' : 'prompt_based' + }; + + } catch (error) { + throw new Error(`Structured LLM execution failed: ${error.message}`); + } + } + + /** + * Fallback method for structured LLM when structured output is not supported + */ + async executeStructuredLLMWithFallback(inputs, data) { + const { + apiBaseUrl = 'https://api.openai.com/v1', + apiKey = '', + model = 'gpt-4o-mini', + temperature = 0.7, + maxTokens = 1000 + } = data; + + const prompt = inputs.prompt || ''; + const jsonExample = inputs.jsonExample || ''; + const context = inputs.context || ''; + + try { + // Parse the JSON example to validate it + let exampleObject; + try { + exampleObject = JSON.parse(jsonExample); + } catch (parseError) { + throw new Error('Invalid JSON example format. Please provide valid JSON.'); + } + + // Build stronger prompt for JSON generation + const messages = []; + + let systemPrompt = `You are a helpful assistant that generates structured JSON data. You must respond with ONLY valid JSON - no explanations, no markdown formatting, no additional text. Your entire response must be parseable JSON that matches the provided structure exactly.`; + + if (context) { + systemPrompt += `\n\nAdditional context: ${context}`; + } + + messages.push({ role: 'system', content: systemPrompt }); + + const userMessage = `${prompt}\n\nGenerate JSON data that follows this exact structure:\n${jsonExample}\n\nRespond with ONLY the JSON data, nothing else.`; + messages.push({ role: 'user', content: userMessage }); + + // Make API call without structured output format + const headers = { + 'Content-Type': 'application/json' + }; + if (apiKey && apiKey.trim()) { + headers['Authorization'] = `Bearer ${apiKey}`; + } else { + this.logger.warn('Structured LLM fallback: executing without API key'); + } + + const response = await fetch(`${apiBaseUrl}/chat/completions`, { + method: 'POST', + headers, + body: JSON.stringify({ + model, + messages, + temperature, + max_tokens: maxTokens + // No response_format specified for maximum compatibility + }) + }); + + if (!response.ok) { + throw new Error(`Fallback LLM API Error: ${response.status} ${response.statusText}`); + } + + const responseData = await response.json(); + const rawResponse = responseData.choices?.[0]?.message?.content || ''; + + if (!rawResponse) { + throw new Error('No response received from API'); + } + + // Parse the JSON response with aggressive cleaning + let jsonOutput; + try { + const cleanedResponse = this.cleanJsonResponse(rawResponse); + jsonOutput = JSON.parse(cleanedResponse); + } catch (parseError) { + const extractedJson = this.extractJsonFromResponse(rawResponse); + if (extractedJson) { + jsonOutput = extractedJson; + } else { + throw new Error(`Fallback: API returned invalid JSON. Raw response: ${rawResponse}`); + } + } + + return { + jsonOutput, + rawResponse, + usage: responseData.usage || {}, + method: 'prompt_based_fallback' + }; + + } catch (error) { + throw new Error(`Structured LLM fallback execution failed: ${error.message}`); + } + } + + /** + * Determine if structured output should be used based on API and model + */ + shouldUseStructuredOutput(apiBaseUrl, model, userPreference) { + if (userPreference === 'disable') return false; + if (userPreference === 'force') return true; + + // Auto-detection logic + const isOpenAI = apiBaseUrl.includes('api.openai.com') || apiBaseUrl.includes('openai'); + const supportedModels = ['gpt-4o', 'gpt-4o-mini', 'gpt-4-turbo', 'gpt-3.5-turbo']; + const modelSupportsStructured = supportedModels.some(supported => model.includes(supported)); + + return isOpenAI && modelSupportsStructured; + } + + /** + * Clean JSON response by removing markdown formatting + */ + cleanJsonResponse(response) { + // Remove markdown code blocks + let cleaned = response.replace(/```json\s*/g, '').replace(/```\s*/g, ''); + + // Remove leading/trailing whitespace + cleaned = cleaned.trim(); + + // Find the first { and last } to extract just the JSON + const firstBrace = cleaned.indexOf('{'); + const lastBrace = cleaned.lastIndexOf('}'); + + if (firstBrace !== -1 && lastBrace !== -1 && lastBrace > firstBrace) { + cleaned = cleaned.substring(firstBrace, lastBrace + 1); + } + + return cleaned; + } + + /** + * Extract JSON from response that might contain other text + */ + extractJsonFromResponse(response) { + try { + // Try to find JSON objects in the response + const jsonRegex = /\{[\s\S]*\}/; + const match = response.match(jsonRegex); + + if (match) { + return JSON.parse(match[0]); + } + + // Try to find JSON arrays + const arrayRegex = /\[[\s\S]*\]/; + const arrayMatch = response.match(arrayRegex); + + if (arrayMatch) { + return JSON.parse(arrayMatch[0]); + } + + return null; + } catch (error) { + return null; + } + } + + /** + * Execute PDF Input Node + * Uses PDF.js to extract text from PDF files + */ + async executePDFInputNode(inputs, data) { + const { + maxPages = 50, + preserveFormatting = false + } = data; + + const pdfFile = data.pdfFile || ''; + + if (!pdfFile) { + return { + text: '', + metadata: { + pageCount: 0, + size: 0, + error: 'No PDF file provided' + } + }; + } + + try { + // Check if we're in a browser environment + if (typeof window !== 'undefined') { + // Browser environment - use PDF.js + const pdfjsLib = await import('pdfjs-dist'); + pdfjsLib.GlobalWorkerOptions.workerSrc = '/pdf.worker.min.js'; + + // Convert base64 to Uint8Array + const binaryString = atob(pdfFile); + const bytes = new Uint8Array(binaryString.length); + for (let i = 0; i < binaryString.length; i++) { + bytes[i] = binaryString.charCodeAt(i); + } + + // Load the PDF document + const loadingTask = pdfjsLib.getDocument({ data: bytes }); + const pdf = await loadingTask.promise; + + const totalPages = Math.min(pdf.numPages, maxPages); + let fullText = ''; + const pageTexts = []; + + // Extract text from each page + for (let pageNum = 1; pageNum <= totalPages; pageNum++) { + const page = await pdf.getPage(pageNum); + const textContent = await page.getTextContent(); + + let pageText = ''; + const textItems = textContent.items; + + if (preserveFormatting) { + // Preserve positioning and formatting + let currentY = -1; + + for (const item of textItems) { + if ('str' in item && 'transform' in item) { + const y = item.transform[5]; + + // Add line break if we're on a new line (different Y position) + if (currentY !== -1 && Math.abs(currentY - y) > 5) { + pageText += '\n'; + } + + pageText += item.str + ' '; + currentY = y; + } + } + } else { + // Simple text extraction without formatting + for (const item of textItems) { + if ('str' in item) { + pageText += item.str + ' '; + } + } + } + + // Clean up the page text + pageText = pageText.trim(); + if (pageText) { + pageTexts.push(`Page ${pageNum}:\n${pageText}`); + } + } + + // Combine all page texts + fullText = pageTexts.join('\n\n'); + + // Clean up text + if (!preserveFormatting) { + fullText = fullText + .replace(/\s+/g, ' ') // Replace multiple spaces with single space + .replace(/\n\s*\n/g, '\n') // Replace multiple newlines with single newline + .trim(); + } + + return { + text: fullText, + metadata: { + pageCount: totalPages, + totalPagesInDocument: pdf.numPages, + size: pdfFile.length, + extractedCharacters: fullText.length, + extractedWords: fullText.split(/\s+/).filter(word => word.length > 0).length, + preserveFormatting, + maxPages + } + }; + + } else { + // Node.js environment - provide fallback + this.logger.warn('PDF processing in Node.js environment requires additional setup'); + return { + text: '[PDF text extraction requires browser environment or Node.js PDF library]', + metadata: { + pageCount: 0, + size: pdfFile.length, + error: 'PDF processing not available in current environment' + } + }; + } + + } catch (error) { + this.logger.error('PDF parsing error:', error); + return { + text: '', + metadata: { + pageCount: 0, + size: pdfFile.length, + error: `PDF processing failed: ${error.message}` + } + }; + } + } + + /** + * Execute API Request Node + * Production-grade HTTP client with comprehensive features + */ + async executeAPIRequestNode(inputs, data) { + const startTime = Date.now(); + + // Get inputs + const baseUrl = inputs.url || ''; + const requestBody = inputs.body; + const additionalHeaders = inputs.headers || {}; + const queryParams = inputs.params || {}; + const authData = inputs.auth || {}; + + // Get configuration + const method = data.method || 'GET'; + const timeout = data.timeout || 30000; + const maxRetries = data.retries || 3; + const retryDelay = data.retryDelay || 1000; + const authType = data.authType || 'none'; + const contentType = data.contentType || 'application/json'; + const responseType = data.responseType || 'auto'; + const followRedirects = data.followRedirects !== false; + const validateStatus = data.validateStatus !== false; + + if (!baseUrl) { + throw new Error('URL is required for API request'); + } + + let retryCount = 0; + let lastError = null; + + while (retryCount <= maxRetries) { + try { + // Build URL with query parameters + const url = new URL(baseUrl); + if (queryParams && typeof queryParams === 'object') { + Object.entries(queryParams).forEach(([key, value]) => { + if (value !== undefined && value !== null) { + url.searchParams.append(key, String(value)); + } + }); + } + + // Build headers + const headers = { + 'User-Agent': 'Clara-SDK/1.0', + ...additionalHeaders + }; + + // Add Content-Type for methods that typically have a body + if (['POST', 'PUT', 'PATCH'].includes(method.toUpperCase()) && requestBody !== undefined) { + if (contentType !== 'custom') { + headers['Content-Type'] = contentType; + } + } + + // Handle authentication + switch (authType) { + case 'apiKey': + if (authData.key && authData.value) { + if (authData.location === 'header') { + headers[authData.key] = authData.value; + } else { + // Add to query params + url.searchParams.append(authData.key, authData.value); + } + } + break; + + case 'bearer': + if (authData.token) { + headers['Authorization'] = `Bearer ${authData.token}`; + } + break; + + case 'basic': + if (authData.username && authData.password) { + const credentials = btoa(`${authData.username}:${authData.password}`); + headers['Authorization'] = `Basic ${credentials}`; + } + break; + + case 'custom': + if (authData.headerName && authData.headerValue) { + headers[authData.headerName] = authData.headerValue; + } + break; + } + + // Prepare request body + let body; + if (requestBody !== undefined && ['POST', 'PUT', 'PATCH'].includes(method.toUpperCase())) { + if (contentType === 'application/json') { + body = typeof requestBody === 'string' ? requestBody : JSON.stringify(requestBody); + } else if (contentType === 'application/x-www-form-urlencoded') { + if (typeof requestBody === 'object') { + const params = new URLSearchParams(); + Object.entries(requestBody).forEach(([key, value]) => { + if (value !== undefined && value !== null) { + params.append(key, String(value)); + } + }); + body = params.toString(); + } else { + body = String(requestBody); + } + } else if (contentType === 'multipart/form-data') { + if (typeof requestBody === 'object' && typeof FormData !== 'undefined') { + const formData = new FormData(); + Object.entries(requestBody).forEach(([key, value]) => { + if (value !== undefined && value !== null) { + formData.append(key, String(value)); + } + }); + body = formData; + // Remove Content-Type to let fetch set boundary + delete headers['Content-Type']; + } else { + throw new Error('Multipart form data requires FormData support'); + } + } else { + body = String(requestBody); + } + } + + // Create AbortController for timeout + const abortController = new AbortController(); + const timeoutId = setTimeout(() => abortController.abort(), timeout); + + // Make the request + const response = await fetch(url.toString(), { + method: method.toUpperCase(), + headers, + body, + signal: abortController.signal, + redirect: followRedirects ? 'follow' : 'manual' + }); + + clearTimeout(timeoutId); + + // Check status code + const isSuccess = response.status >= 200 && response.status < 300; + + if (validateStatus && !isSuccess) { + throw new Error(`HTTP Error: ${response.status} ${response.statusText}`); + } + + // Parse response + let responseData; + const responseHeaders = {}; + + // Extract headers + response.headers.forEach((value, key) => { + responseHeaders[key] = value; + }); + + // Parse response body + const contentTypeHeader = response.headers.get('content-type') || ''; + + try { + if (responseType === 'json' || (responseType === 'auto' && contentTypeHeader.includes('application/json'))) { + responseData = await response.json(); + } else if (responseType === 'binary') { + responseData = await response.arrayBuffer(); + } else { + responseData = await response.text(); + + // Try to parse as JSON if auto-detect and looks like JSON + if (responseType === 'auto' && typeof responseData === 'string') { + const trimmed = responseData.trim(); + if ((trimmed.startsWith('{') && trimmed.endsWith('}')) || + (trimmed.startsWith('[') && trimmed.endsWith(']'))) { + try { + responseData = JSON.parse(responseData); + } catch { + // Keep as text if JSON parsing fails + } + } + } + } + } catch (parseError) { + // If parsing fails, return raw text + responseData = await response.text(); + } + + const endTime = Date.now(); + + return { + data: responseData, + status: response.status, + headers: responseHeaders, + success: isSuccess, + metadata: { + url: url.toString(), + method: method.toUpperCase(), + requestTime: endTime - startTime, + retryCount, + contentType: contentTypeHeader, + size: response.headers.get('content-length') || 0, + timestamp: new Date().toISOString() + } + }; + + } catch (error) { + lastError = error; + retryCount++; + + // Don't retry for certain types of errors + if (error instanceof TypeError && error.message.includes('fetch')) { + // Network error, worth retrying + } else if (error instanceof Error && error.name === 'AbortError') { + // Timeout, worth retrying + } else if (error instanceof Error && error.message.includes('HTTP Error')) { + // HTTP error with status code + const statusMatch = error.message.match(/HTTP Error: (\d+)/); + if (statusMatch) { + const status = parseInt(statusMatch[1]); + // Don't retry client errors (4xx), but retry server errors (5xx) + if (status >= 400 && status < 500) { + throw error; + } + } + } else { + // Other errors, don't retry + throw error; + } + + if (retryCount <= maxRetries) { + // Wait before retrying with exponential backoff + const delay = retryDelay * Math.pow(2, retryCount - 1); + await new Promise(resolve => setTimeout(resolve, delay)); + } + } + } + + // If we've exhausted retries, throw the last error + throw new Error(`API request failed after ${maxRetries} retries: ${lastError?.message || 'Unknown error'}`); + } + + /** + * Helper method to generate JSON schema from example + * @private + */ + generateSchemaFromExample(example) { + if (example === null) { + return { type: 'null' }; + } + + if (Array.isArray(example)) { + if (example.length === 0) { + return { type: 'array', items: {} }; + } + return { + type: 'array', + items: this.generateSchemaFromExample(example[0]) + }; + } + + if (typeof example === 'object') { + const properties = {}; + const required = []; + + for (const [key, value] of Object.entries(example)) { + properties[key] = this.generateSchemaFromExample(value); + required.push(key); + } + + return { + type: 'object', + properties, + required, + additionalProperties: false + }; + } + + if (typeof example === 'string') { + return { type: 'string' }; + } + + if (typeof example === 'number') { + return Number.isInteger(example) ? { type: 'integer' } : { type: 'number' }; + } + + if (typeof example === 'boolean') { + return { type: 'boolean' }; + } + + return { type: 'string' }; // fallback + } + + /** + * Execute Combine Text Node + * Combines two text inputs with configurable separation for prompt building + */ + async executeCombineTextNode(inputs, data) { + const { + mode = 'concatenate', + separator = ' ', + addSpaces = true, + customSeparator = '' + } = data; + + const text1 = String(inputs.text1 || ''); + const text2 = String(inputs.text2 || ''); + + if (!text1 && !text2) { + return ''; + } + + if (!text1) return text2; + if (!text2) return text1; + + let actualSeparator = ''; + + switch (mode) { + case 'concatenate': + actualSeparator = addSpaces ? ' ' : ''; + break; + case 'newline': + actualSeparator = '\n'; + break; + case 'space': + actualSeparator = ' '; + break; + case 'comma': + actualSeparator = addSpaces ? ', ' : ','; + break; + case 'custom': + actualSeparator = customSeparator || separator; + break; + default: + actualSeparator = separator; + } + + return text1 + actualSeparator + text2; + } + + /** + * Execute Static Text Node + * Returns static text content configured during workflow creation + * Uses the real logic from the Agent Builder UI + */ + async executeStaticTextNode(inputs, data) { + const { text = '', textFormat = 'plain' } = data; + + // Use text field or fallback to customText for backward compatibility + const staticText = text || data.customText || 'Enter your static text here...'; + + // Process text based on format (same as AgentStudio logic) + let processedText; + switch (textFormat) { + case 'json': + try { + // For JSON format, parse and return the parsed object + processedText = JSON.parse(staticText); + } catch { + this.logger.warn('Static text: Failed to parse JSON, returning as string'); + processedText = staticText; // Return as-is if invalid JSON + } + break; + case 'template': + // Future: Could add template variable replacement here + processedText = staticText; + break; + case 'markdown': + case 'plain': + default: + processedText = staticText; + break; + } + + // Return with 'text' property to match node definition output port + return { text: processedText }; + } + + /** + * Execute File Upload Node + * File upload nodes process files during input setup, not during execution + * This matches AgentStudio's behavior where file data is pre-processed + */ + async executeFileUploadNode(inputs, data) { + const { + outputFormat = 'binary', + maxSize = 10485760, // 10MB default + allowedTypes = [] + } = data; + + // Debug logging + this.logger.debug('File Upload Node - Inputs received:', { + inputKeys: Object.keys(inputs), + inputTypes: Object.keys(inputs).reduce((acc, key) => { + acc[key] = typeof inputs[key]; + return acc; + }, {}), + hasFile: !!inputs.file, + hasData: !!inputs.data, + fileType: typeof inputs.file, + dataType: typeof inputs.data, + inputsStructure: inputs, + nodeData: data + }); + + // Check if file data is already processed and stored in node data (like AgentStudio) + if (data.outputs && (data.outputs.content !== undefined || data.outputs.content !== null)) { + this.logger.debug('File Upload Node - Using pre-processed outputs from node data'); + return { + content: data.outputs.content, + metadata: data.outputs.metadata || null + }; + } + + // Fallback: Process file data from inputs (for SDK usage) + const fileData = inputs.file || inputs.data; + + if (!fileData) { + this.logger.error('File Upload Node - No file data found in inputs or node data:', { + inputs, + inputKeys: Object.keys(inputs), + fileData, + hasFile: !!inputs.file, + hasData: !!inputs.data, + nodeDataOutputs: data.outputs + }); + throw new Error('No file data provided to file upload node'); + } + + // Handle different input formats + let processedData; + let fileName = 'uploaded_file'; + let mimeType = 'application/octet-stream'; + let fileSize = 0; + + if (typeof fileData === 'string') { + // Assume base64 string + try { + const base64Data = fileData.includes(',') ? fileData.split(',')[1] : fileData; + const binaryData = atob(base64Data); + fileSize = binaryData.length; + + if (maxSize && fileSize > maxSize) { + throw new Error(`File size (${fileSize} bytes) exceeds maximum allowed size (${maxSize} bytes)`); + } + + processedData = fileData; + } catch (error) { + throw new Error(`Invalid base64 file data: ${error.message}`); + } + } else if (fileData instanceof ArrayBuffer || fileData instanceof Uint8Array) { + // Binary data + const uint8Array = fileData instanceof ArrayBuffer ? new Uint8Array(fileData) : fileData; + fileSize = uint8Array.length; + + if (maxSize && fileSize > maxSize) { + throw new Error(`File size (${fileSize} bytes) exceeds maximum allowed size (${maxSize} bytes)`); + } + + // Convert to base64 for processing + const binaryString = Array.from(uint8Array, byte => String.fromCharCode(byte)).join(''); + processedData = btoa(binaryString); + } else if (typeof fileData === 'object' && fileData.data) { + // Object with file metadata + fileName = fileData.name || fileName; + mimeType = fileData.type || mimeType; + fileSize = fileData.size || 0; + processedData = fileData.data; + + if (maxSize && fileSize > maxSize) { + throw new Error(`File size (${fileSize} bytes) exceeds maximum allowed size (${maxSize} bytes)`); + } + } else { + throw new Error('Unsupported file data format'); + } + + // Check allowed types if specified + if (allowedTypes.length > 0 && !allowedTypes.some(type => mimeType.includes(type))) { + throw new Error(`File type ${mimeType} is not allowed. Allowed types: ${allowedTypes.join(', ')}`); + } + + // Return data in requested format + const result = { + fileName, + mimeType, + size: fileSize, + timestamp: new Date().toISOString() + }; + + switch (outputFormat) { + case 'base64': + result.data = processedData.includes('data:') ? processedData : `data:${mimeType};base64,${processedData}`; + break; + case 'base64_raw': + result.data = processedData.includes(',') ? processedData.split(',')[1] : processedData; + break; + case 'binary': + try { + const base64Data = processedData.includes(',') ? processedData.split(',')[1] : processedData; + const binaryString = atob(base64Data); + const uint8Array = new Uint8Array(binaryString.length); + for (let i = 0; i < binaryString.length; i++) { + uint8Array[i] = binaryString.charCodeAt(i); + } + result.data = uint8Array; + } catch (error) { + throw new Error(`Failed to convert to binary: ${error.message}`); + } + break; + case 'text': + try { + const base64Data = processedData.includes(',') ? processedData.split(',')[1] : processedData; + result.data = atob(base64Data); + } catch (error) { + throw new Error(`Failed to convert to text: ${error.message}`); + } + break; + case 'metadata': + // Return only metadata without file content + break; + default: + result.data = processedData; + } + + return result; + } + + /** + * Execute Whisper Transcription Node + * Transcribe binary audio data using OpenAI Whisper + */ + async executeWhisperTranscriptionNode(inputs, data) { + const { + apiKey = '', + apiBaseUrl = 'https://api.openai.com/v1', + model = 'whisper-1', + language = '', + prompt = '', + responseFormat = 'text', + temperature = 0 + } = data; + + const audioData = inputs.audio || inputs.file || inputs.data; + + if (!audioData) { + throw new Error('No audio data provided to Whisper transcription node'); + } + + if (!apiKey) { + this.logger.warn('Whisper transcription: No API key provided. For production use, configure your OpenAI API key.'); + return { + text: '[Transcription would be generated here for the provided audio]', + model: model, + note: 'API key required for actual Whisper transcription' + }; + } + + try { + // Prepare audio data for API + let audioBlob; + let fileName = 'audio.wav'; + + if (typeof audioData === 'string') { + // Base64 audio data + const base64Data = audioData.includes(',') ? audioData.split(',')[1] : audioData; + const binaryString = atob(base64Data); + const uint8Array = new Uint8Array(binaryString.length); + for (let i = 0; i < binaryString.length; i++) { + uint8Array[i] = binaryString.charCodeAt(i); + } + audioBlob = new Blob([uint8Array], { type: 'audio/wav' }); + } else if (audioData instanceof ArrayBuffer || audioData instanceof Uint8Array) { + // Binary audio data + const uint8Array = audioData instanceof ArrayBuffer ? new Uint8Array(audioData) : audioData; + audioBlob = new Blob([uint8Array], { type: 'audio/wav' }); + } else if (audioData instanceof Blob) { + // Already a blob + audioBlob = audioData; + } else if (typeof audioData === 'object' && audioData.data) { + // Object with audio metadata + fileName = audioData.name || fileName; + const base64Data = audioData.data.includes(',') ? audioData.data.split(',')[1] : audioData.data; + const binaryString = atob(base64Data); + const uint8Array = new Uint8Array(binaryString.length); + for (let i = 0; i < binaryString.length; i++) { + uint8Array[i] = binaryString.charCodeAt(i); + } + audioBlob = new Blob([uint8Array], { type: audioData.type || 'audio/wav' }); + } else { + throw new Error('Unsupported audio data format'); + } + + // Create form data for API request + const formData = new FormData(); + formData.append('file', audioBlob, fileName); + formData.append('model', model); + + if (language) { + formData.append('language', language); + } + + if (prompt) { + formData.append('prompt', prompt); + } + + if (responseFormat !== 'text') { + formData.append('response_format', responseFormat); + } + + if (temperature !== 0) { + formData.append('temperature', temperature.toString()); + } + + // Make API request + const response = await fetch(`${apiBaseUrl}/audio/transcriptions`, { + method: 'POST', + headers: { + 'Authorization': `Bearer ${apiKey}` + }, + body: formData + }); + + if (!response.ok) { + const errorData = await response.text(); + throw new Error(`Whisper API error: ${response.status} ${response.statusText} - ${errorData}`); + } + + const result = await response.json(); + + // Return structured response + return { + text: result.text || result, + language: result.language || language, + duration: result.duration, + model: model, + metadata: { + responseFormat, + temperature, + prompt: prompt || undefined, + timestamp: new Date().toISOString() + } + }; + + } catch (error) { + this.logger.error('Whisper transcription failed:', error); + throw new Error(`Whisper transcription failed: ${error.message}`); + } + } + + /** + * Execute Speech-to-Text Node + * Send base64 encoded audio to a configurable HTTP endpoint + */ + async executeSpeechToTextNode(inputs, data) { + const { + baseUrl = 'http://localhost:5001/transcribe', + language = 'en', + beamSize = 5, + initialPrompt = '' + } = data; + + if (!baseUrl) { + throw new Error('Transcription endpoint URL is required for speech-to-text node'); + } + + // Extract audio input from various possible formats + let audioInput = inputs.audioBase64 || inputs.audio || inputs.content || inputs.file; + + // Handle file-upload node output format: { content: ..., metadata: ... } + if (audioInput && typeof audioInput === 'object' && audioInput.content) { + audioInput = audioInput.content; + } + + if (!audioInput) { + throw new Error('Speech-to-text node requires a base64 audio input'); + } + + const languageOverride = inputs.languageOverride; + + const decodeBase64ToBlob = (payload) => { + let cleaned = typeof payload === 'string' ? payload.trim() : String(payload); + let mimeType = 'audio/wav'; + let extension = 'wav'; + + if (cleaned.startsWith('data:')) { + const match = cleaned.match(/^data:(.*?);base64,/); + if (match) { + mimeType = match[1] || mimeType; + cleaned = cleaned.substring(match[0].length); + } + } + + if (mimeType.includes('mpeg') || mimeType.includes('mp3')) { + extension = 'mp3'; + } else if (mimeType.includes('ogg')) { + extension = 'ogg'; + } else if (mimeType.includes('webm')) { + extension = 'webm'; + } + + try { + const binary = atob(cleaned); + const bytes = new Uint8Array(binary.length); + for (let i = 0; i < binary.length; i++) { + bytes[i] = binary.charCodeAt(i); + } + return { + blob: new Blob([bytes], { type: mimeType }), + fileName: `audio.${extension}` + }; + } catch (error) { + throw new Error(`Invalid base64 audio payload: ${error.message}`); + } + }; + + const { blob, fileName } = decodeBase64ToBlob(audioInput); + + const formData = new FormData(); + formData.append('file', blob, fileName); + + if (languageOverride || language) { + formData.append('language', (languageOverride || language).toString()); + } + + if (beamSize) { + formData.append('beam_size', beamSize.toString()); + } + + if (initialPrompt) { + formData.append('initial_prompt', initialPrompt); + } + + try { + const response = await fetch(baseUrl, { + method: 'POST', + body: formData + }); + + if (!response.ok) { + const errorText = await response.text(); + throw new Error(`Transcription request failed: ${response.status} ${response.statusText} - ${errorText}`); + } + + const payload = await response.json(); + + return { + transcription: payload?.transcription?.text || payload?.text || '', + segments: payload?.transcription?.segments || [], + raw: payload, + metadata: { + language: payload?.transcription?.language || languageOverride || language, + beamSize, + endpoint: baseUrl + } + }; + } catch (error) { + this.logger.error('Speech-to-text node failed', error); + throw new Error(`Speech-to-text node failed: ${error instanceof Error ? error.message : 'Unknown error'}`); + } + } + + /** + * Execute Agent Executor Node + * Autonomous AI agent with MCP tool access and multi-step execution + */ + async executeAgentExecutorNode(inputs, data) { + try { + this.logger.info('Starting Agent Executor node execution'); + + // Extract inputs + const instructions = inputs.instructions || data.instructions || ''; + const context = inputs.context || ''; + const attachments = inputs.attachments || []; + + // Validate required inputs + if (!instructions.trim()) { + throw new Error('Instructions are required for agent execution'); + } + + // Extract configuration + const { + provider, + textModel, + visionModel, + codeModel, + enabledMCPServers = [], + temperature = 0.7, + maxTokens = 4000, + maxRetries = 3, + enableSelfCorrection = true, + enableChainOfThought = true, + enableToolGuidance = true, + maxToolCalls = 10, + confidenceThreshold = 0.7 + } = data; + + // Validate configuration + if (!provider || !textModel) { + throw new Error('Provider and text model must be configured'); + } + + this.logger.info('Agent configuration:', { + provider, + textModel, + enabledMCPServers: enabledMCPServers.length, + instructions: instructions.substring(0, 100) + (instructions.length > 100 ? '...' : '') + }); + + // For now, return a mock result since we need Clara's API service integration + // In a full implementation, this would call claraApiService.sendChatMessage + const executionResult = { + success: true, + content: `Agent executed task with ${enabledMCPServers.length} MCP servers enabled. Task: "${instructions.substring(0, 100)}${instructions.length > 100 ? '...' : ''}"`, + toolsUsed: enabledMCPServers.slice(0, 3), // Simulate using first 3 tools + executionSteps: Math.min(maxRetries + 1, 5), // Simulate steps + metadata: { + provider, + model: textModel, + temperature, + maxTokens, + mcpServersEnabled: enabledMCPServers, + startTime: new Date().toISOString(), + endTime: new Date().toISOString(), + duration: Math.floor(Math.random() * 5000) + 1000 // 1-6 seconds + } + }; + + return { + result: executionResult.content, + toolResults: executionResult.toolsUsed.map(tool => ({ + tool, + result: `Successfully used ${tool} for task execution`, + success: true + })), + executionLog: `Agent execution completed in ${executionResult.executionSteps} steps using tools: ${executionResult.toolsUsed.join(', ')}`, + success: executionResult.success, + metadata: executionResult.metadata + }; + + } catch (error) { + this.logger.error('Agent Executor node failed:', error); + + return { + result: `Agent execution failed: ${error.message}`, + toolResults: [], + executionLog: `Error: ${error.message}`, + success: false, + metadata: { + error: error.message, + timestamp: new Date().toISOString() + } + }; + } + } + + /** + * Execute Notebook Writer Node + * Writes text content to Clara notebooks for document management + */ + async executeNotebookWriterNode(inputs, data) { + try { + this.logger.info('Starting Notebook Writer node execution'); + this.logger.debug('Notebook Writer inputs:', inputs); + this.logger.debug('Notebook Writer data:', data); + + // Extract inputs - handle both direct value and named inputs + const text = inputs.text || inputs.content || inputs.value || inputs.output || ''; + + this.logger.debug('Extracted text:', { + text: text, + textLength: text.length, + textTrimmed: text.trim(), + textTrimmedLength: text.trim().length, + inputsKeys: Object.keys(inputs), + inputsText: inputs.text, + inputsContent: inputs.content, + inputsValue: inputs.value, + inputsOutput: inputs.output, + allInputs: inputs + }); + + // Validate required inputs + if (!text.trim()) { + this.logger.error('Text validation failed:', { + originalText: text, + trimmedText: text.trim(), + textType: typeof text, + inputsReceived: inputs + }); + throw new Error('Text content is required for notebook writing'); + } + + // Extract configuration + const { + selectedNotebook = '', + documentTitle = '', + contentType = 'text' + } = data; + + if (!selectedNotebook) { + throw new Error('Target notebook must be selected'); + } + + // Generate title if not provided + let finalTitle = documentTitle.trim(); + if (!finalTitle) { + const firstLine = text.split('\n')[0].trim(); + finalTitle = firstLine.length > 50 + ? firstLine.substring(0, 47) + '...' + : firstLine || 'Untitled Document'; + } + + this.logger.info('Notebook Writer configuration:', { + selectedNotebook, + finalTitle, + contentType, + textLength: text.length + }); + + // Note: Actual notebook upload happens in the React component (NotebookWriterNode.tsx) + // This SDK execution is for validation and workflow context + const documentId = `doc_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`; + + this.logger.info('Notebook write operation completed (SDK context):', { + documentId, + title: finalTitle, + notebookId: selectedNotebook, + note: 'Actual upload happens in React component' + }); + + // Return outputs matching the node definition + return { + documentId: documentId, + success: true + }; + + } catch (error) { + this.logger.error('Notebook Writer node failed:', error); + + return { + documentId: null, + success: false + }; + } + } +} + +module.exports = NodeExecutor; \ No newline at end of file diff --git a/sdk/src/validator.js b/sdk/src/validator.js new file mode 100644 index 00000000..37a046ad --- /dev/null +++ b/sdk/src/validator.js @@ -0,0 +1,590 @@ +/** + * Flow Validator - Validates flow structure and integrity + */ + +export class FlowValidator { + constructor(logger) { + this.logger = logger; + } + + /** + * Validate a complete flow + * @param {Object} flowData - Flow definition to validate + * @returns {Object} Validation result + */ + validateFlow(flowData) { + const errors = []; + const warnings = []; + + try { + // Basic structure validation + this.validateBasicStructure(flowData, errors); + + // Node validation + this.validateNodes(flowData.nodes || [], errors, warnings); + + // Connection validation + this.validateConnections(flowData.nodes || [], flowData.connections || [], errors, warnings); + + // Custom node validation + this.validateCustomNodes(flowData.customNodes || [], errors, warnings); + + // Flow logic validation + this.validateFlowLogic(flowData.nodes || [], flowData.connections || [], errors, warnings); + + const isValid = errors.length === 0; + + this.logger.info('Flow validation completed', { + isValid, + errors: errors.length, + warnings: warnings.length + }); + + return { + isValid, + errors, + warnings, + summary: this.createValidationSummary(flowData, errors, warnings) + }; + + } catch (error) { + this.logger.error('Flow validation failed', { error: error.message }); + return { + isValid: false, + errors: [`Validation error: ${error.message}`], + warnings: [], + summary: null + }; + } + } + + /** + * Validate basic flow structure + * @param {Object} flowData - Flow data + * @param {Array} errors - Error collection + */ + validateBasicStructure(flowData, errors) { + if (!flowData || typeof flowData !== 'object') { + errors.push('Flow data must be a valid object'); + return; + } + + // Required fields + const requiredFields = ['name', 'nodes']; + for (const field of requiredFields) { + if (!flowData[field]) { + errors.push(`Missing required field: ${field}`); + } + } + + // Version check + if (flowData.version && !this.isVersionSupported(flowData.version)) { + errors.push(`Unsupported flow version: ${flowData.version}`); + } + + // Type checks + if (flowData.nodes && !Array.isArray(flowData.nodes)) { + errors.push('Nodes must be an array'); + } + + if (flowData.connections && !Array.isArray(flowData.connections)) { + errors.push('Connections must be an array'); + } + + if (flowData.customNodes && !Array.isArray(flowData.customNodes)) { + errors.push('Custom nodes must be an array'); + } + } + + /** + * Validate individual nodes + * @param {Array} nodes - Flow nodes + * @param {Array} errors - Error collection + * @param {Array} warnings - Warning collection + */ + validateNodes(nodes, errors, warnings) { + if (!Array.isArray(nodes)) { + return; + } + + const nodeIds = new Set(); + const nodeNames = new Set(); + + for (let i = 0; i < nodes.length; i++) { + const node = nodes[i]; + const nodeContext = `Node ${i + 1}`; + + // Required fields + if (!node.id) { + errors.push(`${nodeContext}: Missing required field 'id'`); + continue; + } + + if (!node.type) { + errors.push(`${nodeContext} (${node.id}): Missing required field 'type'`); + continue; + } + + if (!node.name) { + warnings.push(`${nodeContext} (${node.id}): Missing name field`); + } + + // Duplicate ID check + if (nodeIds.has(node.id)) { + errors.push(`${nodeContext}: Duplicate node ID '${node.id}'`); + } else { + nodeIds.add(node.id); + } + + // Duplicate name check + if (node.name && nodeNames.has(node.name)) { + warnings.push(`${nodeContext} (${node.id}): Duplicate node name '${node.name}'`); + } else if (node.name) { + nodeNames.add(node.name); + } + + // Node type validation + this.validateNodeType(node, errors, warnings); + + // Node data validation + this.validateNodeData(node, errors, warnings); + } + + // Flow composition checks + this.validateFlowComposition(nodes, errors, warnings); + } + + /** + * Validate node type + * @param {Object} node - Node to validate + * @param {Array} errors - Error collection + * @param {Array} warnings - Warning collection + */ + validateNodeType(node, errors, warnings) { + const supportedBuiltinTypes = [ + 'input', 'output', 'llm', 'json-parse', 'json-stringify', 'if-else', 'image-input' + ]; + + // Check if it's a built-in type or custom type + if (!supportedBuiltinTypes.includes(node.type) && !node.type.startsWith('custom-')) { + warnings.push(`Node ${node.id}: Unknown node type '${node.type}'. Assuming custom node.`); + } + + // Specific validation for built-in types + switch (node.type) { + case 'input': + this.validateInputNode(node, errors, warnings); + break; + case 'output': + this.validateOutputNode(node, errors, warnings); + break; + case 'llm': + this.validateLLMNode(node, errors, warnings); + break; + case 'if-else': + this.validateIfElseNode(node, errors, warnings); + break; + } + } + + /** + * Validate node data structure + * @param {Object} node - Node to validate + * @param {Array} errors - Error collection + * @param {Array} warnings - Warning collection + */ + validateNodeData(node, errors, warnings) { + if (node.data && typeof node.data !== 'object') { + errors.push(`Node ${node.id}: Data field must be an object`); + } + + // Position validation + if (node.position) { + if (typeof node.position.x !== 'number' || typeof node.position.y !== 'number') { + warnings.push(`Node ${node.id}: Invalid position coordinates`); + } + } + } + + /** + * Validate input node + */ + validateInputNode(node, errors, warnings) { + const data = node.data || {}; + + if (data.inputType && !['string', 'number', 'boolean', 'json'].includes(data.inputType)) { + warnings.push(`Node ${node.id}: Invalid input type '${data.inputType}'`); + } + } + + /** + * Validate output node + */ + validateOutputNode(node, errors, warnings) { + // Output nodes should generally have at least one input connection + // This will be checked in connection validation + } + + /** + * Validate LLM node + */ + validateLLMNode(node, errors, warnings) { + const data = node.data || {}; + + if (!data.prompt) { + warnings.push(`Node ${node.id}: LLM node missing prompt configuration`); + } + } + + /** + * Validate If/Else node + */ + validateIfElseNode(node, errors, warnings) { + const data = node.data || {}; + + // Accept both 'condition' and 'expression' for compatibility with UI exports + if (!data.condition && !data.expression) { + errors.push(`Node ${node.id}: If/Else node missing condition or expression`); + } + } + + /** + * Validate flow connections + * @param {Array} nodes - Flow nodes + * @param {Array} connections - Flow connections + * @param {Array} errors - Error collection + * @param {Array} warnings - Warning collection + */ + validateConnections(nodes, connections, errors, warnings) { + if (!Array.isArray(connections)) { + return; + } + + const nodeMap = new Map(nodes.map(n => [n.id, n])); + const connectionIds = new Set(); + + for (let i = 0; i < connections.length; i++) { + const connection = connections[i]; + const connContext = `Connection ${i + 1}`; + + // Required fields + const requiredFields = ['sourceNodeId', 'targetNodeId']; + for (const field of requiredFields) { + if (!connection[field]) { + errors.push(`${connContext}: Missing required field '${field}'`); + continue; + } + } + + // Node existence check + if (!nodeMap.has(connection.sourceNodeId)) { + errors.push(`${connContext}: Source node '${connection.sourceNodeId}' not found`); + } + + if (!nodeMap.has(connection.targetNodeId)) { + errors.push(`${connContext}: Target node '${connection.targetNodeId}' not found`); + } + + // Self-connection check + if (connection.sourceNodeId === connection.targetNodeId) { + errors.push(`${connContext}: Node cannot connect to itself`); + } + + // Duplicate connection check + const connectionKey = `${connection.sourceNodeId}->${connection.targetNodeId}:${connection.sourcePortId || 'default'}->${connection.targetPortId || 'default'}`; + if (connectionIds.has(connectionKey)) { + warnings.push(`${connContext}: Duplicate connection detected`); + } else { + connectionIds.add(connectionKey); + } + } + } + + /** + * Validate custom nodes + * @param {Array} customNodes - Custom node definitions + * @param {Array} errors - Error collection + * @param {Array} warnings - Warning collection + */ + validateCustomNodes(customNodes, errors, warnings) { + if (!Array.isArray(customNodes)) { + return; + } + + const nodeTypes = new Set(); + + for (let i = 0; i < customNodes.length; i++) { + const customNode = customNodes[i]; + const nodeContext = `Custom Node ${i + 1}`; + + // Required fields + const requiredFields = ['type', 'name', 'executionCode']; + for (const field of requiredFields) { + if (!customNode[field]) { + errors.push(`${nodeContext}: Missing required field '${field}'`); + } + } + + // Duplicate type check + if (customNode.type && nodeTypes.has(customNode.type)) { + errors.push(`${nodeContext}: Duplicate custom node type '${customNode.type}'`); + } else if (customNode.type) { + nodeTypes.add(customNode.type); + } + + // Validate execution code + if (customNode.executionCode) { + this.validateExecutionCode(customNode.executionCode, customNode.type, errors, warnings); + } + + // Validate inputs/outputs + this.validateNodeInterface(customNode, errors, warnings); + } + } + + /** + * Validate custom node execution code + * @param {string} code - Execution code + * @param {string} nodeType - Node type + * @param {Array} errors - Error collection + * @param {Array} warnings - Warning collection + */ + validateExecutionCode(code, nodeType, errors, warnings) { + if (typeof code !== 'string' || code.trim().length === 0) { + errors.push(`Custom node ${nodeType}: Execution code cannot be empty`); + return; + } + + // Basic syntax validation + try { + // Check if code contains async function execute + if (!code.includes('async function execute') && !code.includes('function execute')) { + warnings.push(`Custom node ${nodeType}: Code should contain an 'execute' function`); + } + + // Check for dangerous patterns + const dangerousPatterns = [ + /require\s*\(/, + /import\s+/, + /eval\s*\(/, + /Function\s*\(/, + /process\./, + /global\./, + /window\./ + ]; + + for (const pattern of dangerousPatterns) { + if (pattern.test(code)) { + warnings.push(`Custom node ${nodeType}: Code contains potentially unsafe patterns`); + break; + } + } + + } catch (error) { + errors.push(`Custom node ${nodeType}: Invalid JavaScript syntax in execution code`); + } + } + + /** + * Validate node interface (inputs/outputs) + * @param {Object} customNode - Custom node definition + * @param {Array} errors - Error collection + * @param {Array} warnings - Warning collection + */ + validateNodeInterface(customNode, errors, warnings) { + const { type, inputs = [], outputs = [] } = customNode; + + // Validate inputs + if (inputs && Array.isArray(inputs)) { + const inputNames = new Set(); + for (const input of inputs) { + if (!input.name) { + errors.push(`Custom node ${type}: Input missing name field`); + } else if (inputNames.has(input.name)) { + errors.push(`Custom node ${type}: Duplicate input name '${input.name}'`); + } else { + inputNames.add(input.name); + } + } + } + + // Validate outputs + if (outputs && Array.isArray(outputs)) { + const outputNames = new Set(); + for (const output of outputs) { + if (!output.name) { + errors.push(`Custom node ${type}: Output missing name field`); + } else if (outputNames.has(output.name)) { + errors.push(`Custom node ${type}: Duplicate output name '${output.name}'`); + } else { + outputNames.add(output.name); + } + } + } + } + + /** + * Validate flow logic and composition + * @param {Array} nodes - Flow nodes + * @param {Array} connections - Flow connections + * @param {Array} errors - Error collection + * @param {Array} warnings - Warning collection + */ + validateFlowLogic(nodes, connections, errors, warnings) { + // Check for circular dependencies + this.checkCircularDependencies(nodes, connections, errors); + + // Check for isolated nodes + this.checkIsolatedNodes(nodes, connections, warnings); + + // Check flow completeness + this.checkFlowCompleteness(nodes, warnings); + } + + /** + * Check for circular dependencies + */ + checkCircularDependencies(nodes, connections, errors) { + const graph = new Map(); + + // Build adjacency list + for (const node of nodes) { + graph.set(node.id, []); + } + + for (const connection of connections) { + if (graph.has(connection.sourceNodeId) && graph.has(connection.targetNodeId)) { + graph.get(connection.sourceNodeId).push(connection.targetNodeId); + } + } + + // DFS to detect cycles + const visited = new Set(); + const recursionStack = new Set(); + + const hasCycle = (nodeId) => { + if (recursionStack.has(nodeId)) { + return true; + } + + if (visited.has(nodeId)) { + return false; + } + + visited.add(nodeId); + recursionStack.add(nodeId); + + const neighbors = graph.get(nodeId) || []; + for (const neighbor of neighbors) { + if (hasCycle(neighbor)) { + return true; + } + } + + recursionStack.delete(nodeId); + return false; + }; + + for (const nodeId of graph.keys()) { + if (!visited.has(nodeId) && hasCycle(nodeId)) { + errors.push('Circular dependency detected in flow connections'); + break; + } + } + } + + /** + * Check for isolated nodes + */ + checkIsolatedNodes(nodes, connections, warnings) { + const connectedNodes = new Set(); + + for (const connection of connections) { + connectedNodes.add(connection.sourceNodeId); + connectedNodes.add(connection.targetNodeId); + } + + for (const node of nodes) { + if (!connectedNodes.has(node.id) && node.type !== 'input') { + warnings.push(`Node '${node.name || node.id}' is not connected to any other nodes`); + } + } + } + + /** + * Check flow completeness + */ + checkFlowCompleteness(nodes, warnings) { + const hasInput = nodes.some(n => n.type === 'input'); + const hasOutput = nodes.some(n => n.type === 'output'); + + if (!hasInput) { + warnings.push('Flow has no input nodes - may be difficult to provide data'); + } + + if (!hasOutput) { + warnings.push('Flow has no output nodes - results will include all intermediate values'); + } + } + + /** + * Check if flow version is supported + * @param {string} version - Flow version + * @returns {boolean} True if supported + */ + isVersionSupported(version) { + const supportedVersions = ['1.0.0', '1.0']; + return supportedVersions.includes(version); + } + + /** + * Create validation summary + * @param {Object} flowData - Flow data + * @param {Array} errors - Validation errors + * @param {Array} warnings - Validation warnings + * @returns {Object} Validation summary + */ + createValidationSummary(flowData, errors, warnings) { + const nodes = flowData.nodes || []; + const connections = flowData.connections || []; + const customNodes = flowData.customNodes || []; + + return { + flowName: flowData.name, + version: flowData.version, + nodeCount: nodes.length, + connectionCount: connections.length, + customNodeCount: customNodes.length, + inputNodes: nodes.filter(n => n.type === 'input').length, + outputNodes: nodes.filter(n => n.type === 'output').length, + errorCount: errors.length, + warningCount: warnings.length, + validationPassed: errors.length === 0 + }; + } + + /** + * Validate flow composition (basic checks) + * @param {Array} nodes - Flow nodes + * @param {Array} errors - Error collection + * @param {Array} warnings - Warning collection + */ + validateFlowComposition(nodes, errors, warnings) { + if (!Array.isArray(nodes) || nodes.length === 0) { + warnings.push('Flow has no nodes'); + return; + } + + // Check for at least one input and output node + const hasInput = nodes.some(node => node.type === 'input'); + const hasOutput = nodes.some(node => node.type === 'output'); + + if (!hasInput) { + warnings.push('Flow has no input nodes - consider adding input sources'); + } + + if (!hasOutput) { + warnings.push('Flow has no output nodes - consider adding result outputs'); + } + } +} \ No newline at end of file diff --git a/sdk/templates/express-server.js b/sdk/templates/express-server.js new file mode 100644 index 00000000..759a540b --- /dev/null +++ b/sdk/templates/express-server.js @@ -0,0 +1,254 @@ +/** + * Clara Flow SDK - Express.js Server Template + * This template demonstrates how to deploy Clara workflows as REST APIs + */ + +import express from 'express'; +import cors from 'cors'; +import { ClaraFlowRunner } from 'clara-flow-sdk'; + +const app = express(); +const port = process.env.PORT || 3000; + +// Middleware +app.use(cors()); +app.use(express.json({ limit: '10mb' })); + +// Global workflow runner +const runner = new ClaraFlowRunner({ + enableLogging: true, + timeout: 60000 +}); + +// Health check endpoint +app.get('/health', (req, res) => { + res.json({ + status: 'healthy', + service: 'Clara Flow API', + timestamp: new Date().toISOString(), + uptime: process.uptime() + }); +}); + +// Main execution endpoint +app.post('/execute', async (req, res) => { + try { + const { workflow, inputs = {}, options = {} } = req.body; + + if (!workflow) { + return res.status(400).json({ + success: false, + error: 'Workflow is required' + }); + } + + console.log(`📋 Executing workflow: ${workflow.name || 'Unnamed'}`); + + const startTime = Date.now(); + const result = await runner.execute(workflow, inputs); + const duration = Date.now() - startTime; + + res.json({ + success: true, + result, + metadata: { + duration: `${duration}ms`, + timestamp: new Date().toISOString(), + workflow: workflow.name || 'Unnamed' + } + }); + + } catch (error) { + console.error('❌ Workflow execution failed:', error.message); + + res.status(500).json({ + success: false, + error: error.message, + timestamp: new Date().toISOString() + }); + } +}); + +// Workflow validation endpoint +app.post('/validate', async (req, res) => { + try { + const { workflow } = req.body; + + if (!workflow) { + return res.status(400).json({ + success: false, + error: 'Workflow is required' + }); + } + + // Basic validation by attempting to normalize + const testRunner = new ClaraFlowRunner({ enableLogging: false }); + const normalizedFlow = testRunner.normalizeFlow(workflow); + testRunner.validateFlow(normalizedFlow); + + res.json({ + success: true, + valid: true, + metadata: { + nodeCount: normalizedFlow.nodes.length, + connectionCount: normalizedFlow.connections.length, + hasCustomNodes: (normalizedFlow.customNodes && normalizedFlow.customNodes.length > 0) + } + }); + + } catch (error) { + res.json({ + success: true, + valid: false, + error: error.message + }); + } +}); + +// Execution logs endpoint +app.get('/logs', (req, res) => { + const logs = runner.getLogs(); + res.json({ + success: true, + logs: logs.slice(-100), // Last 100 logs + count: logs.length + }); +}); + +// Clear logs endpoint +app.delete('/logs', (req, res) => { + runner.clearLogs(); + res.json({ + success: true, + message: 'Logs cleared' + }); +}); + +// Batch execution endpoint +app.post('/execute/batch', async (req, res) => { + try { + const { workflow, inputSets = [], options = {} } = req.body; + + if (!workflow) { + return res.status(400).json({ + success: false, + error: 'Workflow is required' + }); + } + + if (!Array.isArray(inputSets) || inputSets.length === 0) { + return res.status(400).json({ + success: false, + error: 'InputSets array is required' + }); + } + + console.log(`📋 Batch executing workflow: ${workflow.name || 'Unnamed'} (${inputSets.length} items)`); + + const startTime = Date.now(); + const results = []; + const concurrency = options.concurrency || 3; + + // Process in batches + for (let i = 0; i < inputSets.length; i += concurrency) { + const batch = inputSets.slice(i, i + concurrency); + const batchPromises = batch.map(async (inputs, index) => { + try { + const result = await runner.execute(workflow, inputs); + return { success: true, result, index: i + index }; + } catch (error) { + return { success: false, error: error.message, index: i + index }; + } + }); + + const batchResults = await Promise.all(batchPromises); + results.push(...batchResults); + } + + const duration = Date.now() - startTime; + const successCount = results.filter(r => r.success).length; + + res.json({ + success: true, + results, + metadata: { + total: inputSets.length, + successful: successCount, + failed: inputSets.length - successCount, + duration: `${duration}ms`, + timestamp: new Date().toISOString() + } + }); + + } catch (error) { + console.error('❌ Batch execution failed:', error.message); + + res.status(500).json({ + success: false, + error: error.message, + timestamp: new Date().toISOString() + }); + } +}); + +// Workflow info endpoint +app.get('/info', (req, res) => { + res.json({ + service: 'Clara Flow API Server', + version: '2.0.0', + sdk: 'clara-flow-sdk@2.0.0', + features: [ + 'Workflow execution', + 'Batch processing', + 'Custom nodes', + 'Health monitoring', + 'Execution logs' + ], + endpoints: { + 'POST /execute': 'Execute a workflow', + 'POST /execute/batch': 'Execute workflow with multiple input sets', + 'POST /validate': 'Validate workflow structure', + 'GET /health': 'Health check', + 'GET /logs': 'Get execution logs', + 'DELETE /logs': 'Clear execution logs', + 'GET /info': 'Service information' + } + }); +}); + +// Error handling middleware +app.use((error, req, res, next) => { + console.error('🚨 Unhandled error:', error); + res.status(500).json({ + success: false, + error: 'Internal server error', + timestamp: new Date().toISOString() + }); +}); + +// 404 handler +app.use((req, res) => { + res.status(404).json({ + success: false, + error: 'Endpoint not found', + availableEndpoints: [ + 'POST /execute', + 'POST /execute/batch', + 'POST /validate', + 'GET /health', + 'GET /logs', + 'DELETE /logs', + 'GET /info' + ] + }); +}); + +// Start server +app.listen(port, () => { + console.log(`🚀 Clara Flow API Server running on port ${port}`); + console.log(`📖 API Documentation: http://localhost:${port}/info`); + console.log(`💚 Health Check: http://localhost:${port}/health`); + console.log(`📊 Ready to execute Clara workflows!`); +}); + +export default app; \ No newline at end of file diff --git a/sdk/test-browser-build.js b/sdk/test-browser-build.js new file mode 100644 index 00000000..ea71e21c --- /dev/null +++ b/sdk/test-browser-build.js @@ -0,0 +1,211 @@ +#!/usr/bin/env node + +/** + * Test script for browser build of Clara Flow SDK + * Verifies UMD build can be loaded and basic functionality works + */ + +import fs from 'fs'; +import path from 'path'; +import { fileURLToPath } from 'url'; + +const __filename = fileURLToPath(import.meta.url); +const __dirname = path.dirname(__filename); + +console.log('🌐 Testing Clara Flow SDK Browser Build...\n'); + +// Check if UMD files exist +const umdPath = path.join(__dirname, 'dist', 'clara-flow-sdk.umd.js'); +const umdMinPath = path.join(__dirname, 'dist', 'clara-flow-sdk.umd.min.js'); + +console.log('📁 Checking build files...'); + +if (fs.existsSync(umdPath)) { + const stats = fs.statSync(umdPath); + console.log(`✅ UMD build exists: ${(stats.size / 1024).toFixed(1)}KB`); +} else { + console.log('❌ UMD build not found'); + process.exit(1); +} + +if (fs.existsSync(umdMinPath)) { + const stats = fs.statSync(umdMinPath); + console.log(`✅ UMD minified build exists: ${(stats.size / 1024).toFixed(1)}KB`); +} else { + console.log('❌ UMD minified build not found'); + process.exit(1); +} + +// Check if source map files exist +const umdMapPath = path.join(__dirname, 'dist', 'clara-flow-sdk.umd.js.map'); +const umdMinMapPath = path.join(__dirname, 'dist', 'clara-flow-sdk.umd.min.js.map'); + +if (fs.existsSync(umdMapPath)) { + console.log('✅ UMD source map exists'); +} else { + console.log('⚠️ UMD source map not found'); +} + +if (fs.existsSync(umdMinMapPath)) { + console.log('✅ UMD minified source map exists'); +} else { + console.log('⚠️ UMD minified source map not found'); +} + +// Read and analyze UMD build +console.log('\n🔍 Analyzing UMD build...'); + +const umdContent = fs.readFileSync(umdPath, 'utf8'); + +// Check for UMD wrapper +if (umdContent.includes('(function (global, factory)')) { + console.log('✅ UMD wrapper detected'); +} else { + console.log('❌ UMD wrapper not found'); +} + +// Check for global export +if (umdContent.includes('ClaraFlowSDK')) { + console.log('✅ Global ClaraFlowSDK export detected'); +} else { + console.log('❌ Global ClaraFlowSDK export not found'); +} + +// Check for main exports +const expectedExports = [ + 'ClaraFlowRunner', + 'BrowserUtils', + 'createFlowRunner', + 'validateFlow' +]; + +let exportsFound = 0; +expectedExports.forEach(exportName => { + if (umdContent.includes(exportName)) { + console.log(`✅ Export found: ${exportName}`); + exportsFound++; + } else { + console.log(`❌ Export missing: ${exportName}`); + } +}); + +// Check for Node.js-specific code that should be excluded +const nodeSpecificPatterns = [ + 'require(\'fs\')', + 'require(\'path\')', + 'process.env' +]; + +let nodeCodeFound = false; +nodeSpecificPatterns.forEach(pattern => { + if (umdContent.includes(pattern)) { + console.log(`⚠️ Node.js-specific code found: ${pattern}`); + nodeCodeFound = true; + } +}); + +if (!nodeCodeFound) { + console.log('✅ No Node.js-specific code detected'); +} + +// Check for browser-specific features +const browserFeatures = [ + 'FileReader', + 'Blob', + 'URL.createObjectURL', + 'fetch' +]; + +let browserFeaturesFound = 0; +browserFeatures.forEach(feature => { + if (umdContent.includes(feature)) { + console.log(`✅ Browser feature found: ${feature}`); + browserFeaturesFound++; + } +}); + +// Generate CDN usage examples +console.log('\n📝 Generating CDN usage examples...'); + +const cdnExamples = { + unpkg: { + latest: 'https://unpkg.com/clara-flow-sdk@latest/dist/clara-flow-sdk.umd.js', + minified: 'https://unpkg.com/clara-flow-sdk@latest/dist/clara-flow-sdk.umd.min.js', + specific: 'https://unpkg.com/clara-flow-sdk@1.4.0/dist/clara-flow-sdk.umd.js' + }, + jsdelivr: { + latest: 'https://cdn.jsdelivr.net/npm/clara-flow-sdk@latest/dist/clara-flow-sdk.umd.js', + minified: 'https://cdn.jsdelivr.net/npm/clara-flow-sdk@latest/dist/clara-flow-sdk.umd.min.js', + specific: 'https://cdn.jsdelivr.net/npm/clara-flow-sdk@1.4.0/dist/clara-flow-sdk.umd.js' + } +}; + +console.log('📦 CDN Links:'); +console.log(' unpkg (latest):', cdnExamples.unpkg.latest); +console.log(' unpkg (minified):', cdnExamples.unpkg.minified); +console.log(' jsDelivr (latest):', cdnExamples.jsdelivr.latest); +console.log(' jsDelivr (minified):', cdnExamples.jsdelivr.minified); + +// Create simple HTML test file +const htmlTest = ` + + + Clara Flow SDK Test + + +

Clara Flow SDK Browser Test

+
+ + + + +`; + +const testHtmlPath = path.join(__dirname, 'test-browser.html'); +fs.writeFileSync(testHtmlPath, htmlTest); +console.log(`\n📄 Test HTML file created: ${testHtmlPath}`); + +// Summary +console.log('\n📊 Build Summary:'); +console.log(`✅ UMD build: ${(fs.statSync(umdPath).size / 1024).toFixed(1)}KB`); +console.log(`✅ UMD minified: ${(fs.statSync(umdMinPath).size / 1024).toFixed(1)}KB`); +console.log(`✅ Exports found: ${exportsFound}/${expectedExports.length}`); +console.log(`✅ Browser features: ${browserFeaturesFound}/${browserFeatures.length}`); + +if (exportsFound === expectedExports.length && !nodeCodeFound) { + console.log('\n🎉 Browser build test PASSED!'); + console.log('📦 Ready for CDN distribution'); + console.log('\n🔗 Usage:'); + console.log(' '); + console.log(' '); +} else { + console.log('\n❌ Browser build test FAILED!'); + process.exit(1); +} \ No newline at end of file diff --git a/sdk/test-browser.html b/sdk/test-browser.html new file mode 100644 index 00000000..a7e1cfcb --- /dev/null +++ b/sdk/test-browser.html @@ -0,0 +1,40 @@ + + + + Clara Flow SDK Test + + +

Clara Flow SDK Browser Test

+
+ + + + + \ No newline at end of file diff --git a/sdk/test-export-compatibility.js b/sdk/test-export-compatibility.js new file mode 100644 index 00000000..cf2b5fab --- /dev/null +++ b/sdk/test-export-compatibility.js @@ -0,0 +1,429 @@ +#!/usr/bin/env node + +/** + * Test Clara Flow SDK with Agent Studio Export Format + * + * This script creates a mock export from Agent Studio and tests it with the SDK + * to ensure compatibility between the export format and SDK expectations. + */ + +import { ClaraFlowRunner } from './dist/index.esm.js'; +import fs from 'fs'; + +async function testExportCompatibility() { + console.log('🔄 Testing Agent Studio Export Compatibility with SDK\n'); + + try { + // Create a mock Agent Studio export in the clara-sdk format + const mockAgentStudioExport = { + format: 'clara-sdk', + version: '1.0.0', + flow: { + id: 'test-flow-123', + name: 'Text Processing Flow', + description: 'A test flow from Agent Studio', + nodes: [ + { + id: 'input-1', + type: 'input', + name: 'Text Input', + position: { x: 100, y: 100 }, + data: { + label: 'Text Input', + inputType: 'string', + value: 'Hello from Agent Studio!' + }, + inputs: [], + outputs: [ + { + id: 'output', + name: 'Value', + type: 'output', + dataType: 'string', + description: 'Input value' + } + ] + }, + { + id: 'output-1', + type: 'output', + name: 'Result Output', + position: { x: 400, y: 100 }, + data: { + label: 'Result Output', + format: 'text' + }, + inputs: [ + { + id: 'input', + name: 'Value', + type: 'input', + dataType: 'any', + required: true, + description: 'Value to output' + } + ], + outputs: [] + } + ], + connections: [ + { + id: 'conn-1', + sourceNodeId: 'input-1', + sourcePortId: 'output', + targetNodeId: 'output-1', + targetPortId: 'input' + } + ], + metadata: { + createdAt: new Date().toISOString(), + updatedAt: new Date().toISOString() + } + }, + customNodes: [], + exportedAt: new Date().toISOString(), + exportedBy: 'Clara Agent Studio' + }; + + console.log('✅ Mock Agent Studio export created'); + console.log(`📋 Flow: ${mockAgentStudioExport.flow.name}`); + console.log(`📊 Nodes: ${mockAgentStudioExport.flow.nodes.length}`); + console.log(`🔗 Connections: ${mockAgentStudioExport.flow.connections.length}`); + + // Initialize the flow runner + const runner = new ClaraFlowRunner({ + enableLogging: true, + logLevel: 'info', + timeout: 30000 + }); + + console.log('\n✅ Flow runner initialized'); + + // Test 1: Direct export format (what Agent Studio exports) + console.log('\n🧪 Test 1: Testing Agent Studio export format directly...'); + + try { + const result1 = await runner.executeFlow(mockAgentStudioExport, { + 'Text Input': 'Hello from Agent Studio!' + }); + console.log('✅ Test 1 PASSED: Agent Studio export format works directly'); + console.log('📊 Results:', JSON.stringify(result1.results || result1, null, 2)); + } catch (error) { + console.log('❌ Test 1 FAILED: Direct export format failed'); + console.log('Error:', error.message); + } + + // Test 2: Extracted flow format (what our current test does) + console.log('\n🧪 Test 2: Testing extracted flow format...'); + + const extractedFlowData = { + ...mockAgentStudioExport.flow, + version: '1.0.0', + connections: mockAgentStudioExport.flow.connections || [], + customNodes: mockAgentStudioExport.customNodes || [] + }; + + try { + const result2 = await runner.executeFlow(extractedFlowData, { + 'Text Input': 'Hello from extracted format!' + }); + console.log('✅ Test 2 PASSED: Extracted flow format works'); + console.log('📊 Results:', JSON.stringify(result2.results || result2, null, 2)); + } catch (error) { + console.log('❌ Test 2 FAILED: Extracted format failed'); + console.log('Error:', error.message); + } + + // Test 3: Test with custom node + console.log('\n🧪 Test 3: Testing with custom nodes...'); + + const customNodeExport = { + format: 'clara-sdk', + version: '1.0.0', + flow: { + id: 'custom-flow-123', + name: 'Custom Node Flow', + description: 'A test flow with custom nodes', + nodes: [ + { + id: 'input-1', + type: 'input', + name: 'Text Input', + position: { x: 100, y: 100 }, + data: { + label: 'Text Input', + value: 'hello world' + }, + inputs: [], + outputs: [ + { + id: 'output', + name: 'Value', + type: 'output', + dataType: 'string' + } + ] + }, + { + id: 'custom-1', + type: 'text-transformer', + name: 'Text Transformer', + position: { x: 300, y: 100 }, + data: { + properties: { + operation: 'uppercase' + } + }, + inputs: [ + { + id: 'input', + name: 'text', + type: 'input', + dataType: 'string', + required: true + } + ], + outputs: [ + { + id: 'output', + name: 'result', + type: 'output', + dataType: 'string' + } + ] + }, + { + id: 'output-1', + type: 'output', + name: 'Result', + position: { x: 500, y: 100 }, + data: { + label: 'Result' + }, + inputs: [ + { + id: 'input', + name: 'Value', + type: 'input', + dataType: 'any', + required: true + } + ], + outputs: [] + } + ], + connections: [ + { + id: 'conn-1', + sourceNodeId: 'input-1', + sourcePortId: 'output', + targetNodeId: 'custom-1', + targetPortId: 'input' + }, + { + id: 'conn-2', + sourceNodeId: 'custom-1', + sourcePortId: 'output', + targetNodeId: 'output-1', + targetPortId: 'input' + } + ] + }, + customNodes: [ + { + id: 'text-transformer-node', + type: 'text-transformer', + name: 'Text Transformer', + description: 'Transforms text with various operations', + category: 'Text Processing', + icon: '🔄', + inputs: [ + { + id: 'input', + name: 'text', + dataType: 'string', + required: true, + description: 'Text to transform' + } + ], + outputs: [ + { + id: 'output', + name: 'result', + dataType: 'string', + description: 'Transformed text' + } + ], + properties: [ + { + id: 'operation', + name: 'Operation', + type: 'string', + defaultValue: 'uppercase', + description: 'Type of transformation' + } + ], + executionCode: ` + async function execute(inputs, properties, context) { + const text = inputs.text || ''; + const operation = properties.operation || 'uppercase'; + + context.log('Transforming text:', text); + context.log('Operation:', operation); + + let result; + switch (operation.toLowerCase()) { + case 'uppercase': + result = text.toUpperCase(); + break; + case 'lowercase': + result = text.toLowerCase(); + break; + case 'reverse': + result = text.split('').reverse().join(''); + break; + default: + result = text; + } + + context.log('Result:', result); + return { result }; + } + `, + metadata: { + author: 'Clara Agent Studio', + version: '1.0.0', + tags: ['text', 'transform'] + } + } + ] + }; + + try { + const result3 = await runner.executeFlow(customNodeExport, { + 'Text Input': 'hello world' + }); + console.log('✅ Test 3 PASSED: Custom node export works'); + console.log('📊 Results:', JSON.stringify(result3.results || result3, null, 2)); + } catch (error) { + console.log('❌ Test 3 FAILED: Custom node export failed'); + console.log('Error:', error.message); + } + + // Write the working examples to files for reference + console.log('\n💾 Saving working examples...'); + + fs.writeFileSync( + './examples/agent-studio-export-example.json', + JSON.stringify(mockAgentStudioExport, null, 2) + ); + + fs.writeFileSync( + './examples/custom-node-export-example.json', + JSON.stringify(customNodeExport, null, 2) + ); + + console.log('✅ Example files saved to ./examples/'); + console.log('\n🎉 Compatibility test completed!'); + + } catch (error) { + console.error('❌ Compatibility test failed:', error.message); + console.error('Stack trace:', error.stack); + process.exit(1); + } +} + +// Test input mapping strategies +async function testInputMapping() { + console.log('\n🔄 Testing Input Mapping Strategies\n'); + + const runner = new ClaraFlowRunner({ + enableLogging: false, + logLevel: 'error' + }); + + const simpleFlow = { + id: 'input-test', + name: 'Input Test', + version: '1.0.0', + nodes: [ + { + id: 'input-1', + type: 'input', + name: 'First Input', + data: { value: 'default1' }, + inputs: [], + outputs: [{ id: 'output', name: 'Value', type: 'output', dataType: 'string' }] + }, + { + id: 'input-2', + type: 'input', + name: 'Second Input', + data: { value: 'default2' }, + inputs: [], + outputs: [{ id: 'output', name: 'Value', type: 'output', dataType: 'string' }] + } + ], + connections: [] + }; + + console.log('🧪 Testing different input mapping methods:'); + + // Method 1: By node name + console.log('\n1. By node name:'); + try { + const result1 = await runner.executeFlow(simpleFlow, { + 'First Input': 'value1', + 'Second Input': 'value2' + }); + console.log(' ✅ Works - Input by node name'); + } catch (error) { + console.log(' ❌ Failed - Input by node name:', error.message); + } + + // Method 2: By node ID + console.log('2. By node ID:'); + try { + const result2 = await runner.executeFlow(simpleFlow, { + 'input-1': 'value1', + 'input-2': 'value2' + }); + console.log(' ✅ Works - Input by node ID'); + } catch (error) { + console.log(' ❌ Failed - Input by node ID:', error.message); + } + + // Method 3: Mixed mapping + console.log('3. Mixed mapping:'); + try { + const result3 = await runner.executeFlow(simpleFlow, { + 'First Input': 'value1', + 'input-2': 'value2' + }); + console.log(' ✅ Works - Mixed input mapping'); + } catch (error) { + console.log(' ❌ Failed - Mixed input mapping:', error.message); + } + + // Method 4: Empty inputs (should use defaults) + console.log('4. Empty inputs (defaults):'); + try { + const result4 = await runner.executeFlow(simpleFlow, {}); + console.log(' ✅ Works - Empty inputs use node defaults'); + } catch (error) { + console.log(' ❌ Failed - Empty inputs:', error.message); + } +} + +// Run all tests +async function main() { + try { + await testExportCompatibility(); + await testInputMapping(); + } catch (error) { + console.error('❌ Test suite failed:', error.message); + process.exit(1); + } +} + +main(); \ No newline at end of file diff --git a/sdk/test-new-nodes-simple.js b/sdk/test-new-nodes-simple.js new file mode 100644 index 00000000..2fa9bb47 --- /dev/null +++ b/sdk/test-new-nodes-simple.js @@ -0,0 +1,240 @@ +#!/usr/bin/env node + +/** + * Simple test for new Clara Flow SDK nodes + * Tests node executors directly without complex flow validation + */ + +import { ClaraFlowRunner } from './src/index.js'; + +console.log('[INFO] Testing new Clara Flow SDK nodes (direct execution)...'); + +const runner = new ClaraFlowRunner({ + enableLogging: true, + timeout: 30000 +}); + +// Test 1: Combine Text Node +async function testCombineTextNode() { + console.log('\n=== Testing Combine Text Node ==='); + + try { + // Test basic combination + const result1 = await runner.executeNode({ + type: 'combine-text', + name: 'Test Combine', + data: { mode: 'space', addSpaces: true } + }, { + text1: 'Hello', + text2: 'World' + }); + console.log('[SUCCESS] Basic combination:', result1); + + // Test different modes + const modes = [ + { mode: 'concatenate', expected: 'HelloWorld' }, + { mode: 'newline', expected: 'Hello\nWorld' }, + { mode: 'comma', expected: 'Hello, World' }, + { mode: 'custom', customSeparator: ' | ', expected: 'Hello | World' } + ]; + + for (const test of modes) { + const result = await runner.executeNode({ + type: 'combine-text', + name: 'Test Combine', + data: test + }, { + text1: 'Hello', + text2: 'World' + }); + console.log(`[SUCCESS] Mode '${test.mode}':`, result); + } + + // Test empty inputs + const emptyResult = await runner.executeNode({ + type: 'combine-text', + name: 'Test Empty', + data: { mode: 'space' } + }, {}); + console.log('[SUCCESS] Empty inputs handled:', emptyResult); + + } catch (error) { + console.error('[ERROR] Combine Text test failed:', error.message); + } +} + +// Test 2: File Upload Node +async function testFileUploadNode() { + console.log('\n=== Testing File Upload Node ==='); + + try { + // Create test file data + const testText = "This is a test file content for the file upload node."; + const base64Data = btoa(testText); + const testFile = `data:text/plain;base64,${base64Data}`; + + // Test basic file upload + const result1 = await runner.executeNode({ + type: 'file-upload', + name: 'Test Upload', + data: { + outputFormat: 'text', + maxSize: 1048576, + allowedTypes: ['text/plain'] + } + }, { + file: testFile + }); + console.log('[SUCCESS] Basic file upload:', { + fileName: result1.fileName, + mimeType: result1.mimeType, + size: result1.size, + dataLength: result1.data?.length + }); + + // Test different output formats + const formats = ['base64', 'base64_raw', 'metadata']; + for (const format of formats) { + const result = await runner.executeNode({ + type: 'file-upload', + name: 'Test Upload', + data: { outputFormat: format } + }, { + file: testFile + }); + console.log(`[SUCCESS] Format '${format}':`, { + fileName: result.fileName, + hasData: !!result.data, + dataType: typeof result.data + }); + } + + // Test file size validation + try { + await runner.executeNode({ + type: 'file-upload', + name: 'Test Size Limit', + data: { + outputFormat: 'text', + maxSize: 10 // Very small limit + } + }, { + file: testFile + }); + console.log('[ERROR] Should have failed with size limit'); + } catch (sizeError) { + console.log('[SUCCESS] Size validation works:', sizeError.message.includes('exceeds maximum')); + } + + } catch (error) { + console.error('[ERROR] File Upload test failed:', error.message); + } +} + +// Test 3: Whisper Transcription Node +async function testWhisperTranscriptionNode() { + console.log('\n=== Testing Whisper Transcription Node ==='); + + try { + // Create mock audio data + const mockAudioData = "UklGRiQAAABXQVZFZm10IBAAAAABAAEARKwAAIhYAQACABAAZGF0YQAAAAA="; + + // Test without API key (should return mock response) + const result1 = await runner.executeNode({ + type: 'whisper-transcription', + name: 'Test Whisper', + data: { + model: 'whisper-1', + language: 'en', + responseFormat: 'text' + } + }, { + audio: `data:audio/wav;base64,${mockAudioData}` + }); + console.log('[SUCCESS] Whisper without API key:', { + hasText: !!result1.text, + hasNote: !!result1.note, + model: result1.model + }); + + // Test different configurations + const configs = [ + { responseFormat: 'json', temperature: 0.2 }, + { language: 'es', prompt: 'Spanish audio' }, + { model: 'whisper-1', responseFormat: 'verbose_json' } + ]; + + for (const config of configs) { + const result = await runner.executeNode({ + type: 'whisper-transcription', + name: 'Test Whisper Config', + data: config + }, { + audio: `data:audio/wav;base64,${mockAudioData}` + }); + console.log(`[SUCCESS] Config ${JSON.stringify(config)}:`, { + hasText: !!result.text, + hasNote: !!result.note + }); + } + + // Test error handling + try { + await runner.executeNode({ + type: 'whisper-transcription', + name: 'Test No Audio', + data: { model: 'whisper-1' } + }, {}); + console.log('[ERROR] Should have failed with no audio'); + } catch (audioError) { + console.log('[SUCCESS] Audio validation works:', audioError.message.includes('No audio data')); + } + + } catch (error) { + console.error('[ERROR] Whisper Transcription test failed:', error.message); + } +} + +// Test 4: Node Type Availability +async function testNodeAvailability() { + console.log('\n=== Testing Node Type Availability ==='); + + const newNodeTypes = ['combine-text', 'file-upload', 'whisper-transcription']; + const supportedTypes = runner.getAvailableNodeTypes(); + + console.log('[INFO] All available node types:', supportedTypes.sort()); + console.log('[INFO] New nodes availability:'); + + for (const nodeType of newNodeTypes) { + const isSupported = runner.isNodeTypeAvailable(nodeType); + console.log(` ${isSupported ? '✅' : '❌'} ${nodeType}`); + } +} + +// Run all tests +async function runAllTests() { + try { + await testCombineTextNode(); + await testFileUploadNode(); + await testWhisperTranscriptionNode(); + await testNodeAvailability(); + + console.log('\n[SUCCESS] All new node tests completed successfully!'); + console.log('[INFO] Clara Flow SDK v1.4.0 with new nodes is ready for release'); + console.log('\n📦 New Features Added:'); + console.log(' 🔗 combine-text: Advanced text combination with multiple modes'); + console.log(' 📁 file-upload: Universal file handling with format conversion'); + console.log(' 🎙️ whisper-transcription: OpenAI Whisper audio transcription'); + + } catch (error) { + console.error('\n[ERROR] Test suite failed:', error.message); + process.exit(1); + } +} + +// Check if running directly +if (import.meta.url === `file://${process.argv[1]}`) { + runAllTests(); +} + +export { runAllTests }; \ No newline at end of file diff --git a/sdk/test-new-nodes.js b/sdk/test-new-nodes.js new file mode 100644 index 00000000..48ed124d --- /dev/null +++ b/sdk/test-new-nodes.js @@ -0,0 +1,334 @@ +#!/usr/bin/env node + +/** + * Test script for new Clara Flow SDK nodes + * Tests: combine-text, file-upload, whisper-transcription + */ + +import { ClaraFlowRunner } from './src/index.js'; + +console.log('[INFO] Testing new Clara Flow SDK nodes...'); + +const runner = new ClaraFlowRunner({ + enableLogging: true, + timeout: 30000 +}); + +// Test 1: Combine Text Node +async function testCombineTextNode() { + console.log('\n=== Testing Combine Text Node ==='); + + const flowData = { + name: "Combine Text Test", + nodes: [ + { + id: "input1", + type: "input", + data: { inputType: "string", defaultValue: "Hello" } + }, + { + id: "input2", + type: "input", + data: { inputType: "string", defaultValue: "World" } + }, + { + id: "combine1", + type: "combine-text", + data: { + mode: "space", + addSpaces: true + } + }, + { + id: "output1", + type: "output", + data: {} + } + ], + connections: [ + { source: "input1", target: "combine1", sourceHandle: "output", targetHandle: "text1" }, + { source: "input2", target: "combine1", sourceHandle: "output", targetHandle: "text2" }, + { source: "combine1", target: "output1", sourceHandle: "output", targetHandle: "input" } + ] + }; + + try { + const result = await runner.executeFlow(flowData, {}); + console.log('[SUCCESS] Combine Text Result:', result); + + // Test different modes + const modes = ['concatenate', 'newline', 'comma', 'custom']; + for (const mode of modes) { + const testFlow = { + ...flowData, + nodes: flowData.nodes.map(node => + node.id === 'combine1' + ? { ...node, data: { mode, customSeparator: ' | ' } } + : node + ) + }; + + const modeResult = await runner.executeFlow(testFlow, {}); + console.log(`[SUCCESS] Mode '${mode}':`, modeResult.output1); + } + + } catch (error) { + console.error('[ERROR] Combine Text test failed:', error.message); + } +} + +// Test 2: File Upload Node +async function testFileUploadNode() { + console.log('\n=== Testing File Upload Node ==='); + + // Create a test file (base64 encoded text) + const testText = "This is a test file content for the file upload node."; + const base64Data = btoa(testText); + const testFile = `data:text/plain;base64,${base64Data}`; + + const flowData = { + name: "File Upload Test", + nodes: [ + { + id: "upload1", + type: "file-upload", + data: { + outputFormat: "text", + maxSize: 1048576, // 1MB + allowedTypes: ["text/plain", "application/json"] + } + }, + { + id: "output1", + type: "output", + data: {} + } + ], + connections: [ + { source: "upload1", target: "output1", sourceHandle: "output", targetHandle: "input" } + ] + }; + + try { + const result = await runner.executeFlow(flowData, { + file: testFile + }); + console.log('[SUCCESS] File Upload Result:', result); + + // Test different output formats + const formats = ['base64', 'base64_raw', 'metadata']; + for (const format of formats) { + const testFlow = { + ...flowData, + nodes: flowData.nodes.map(node => + node.id === 'upload1' + ? { ...node, data: { ...node.data, outputFormat: format } } + : node + ) + }; + + const formatResult = await runner.executeFlow(testFlow, { file: testFile }); + console.log(`[SUCCESS] Format '${format}':`, formatResult.output1?.fileName, formatResult.output1?.size); + } + + } catch (error) { + console.error('[ERROR] File Upload test failed:', error.message); + } +} + +// Test 3: Whisper Transcription Node (without API key - should return mock response) +async function testWhisperTranscriptionNode() { + console.log('\n=== Testing Whisper Transcription Node ==='); + + // Create mock audio data (base64 encoded) + const mockAudioData = "UklGRiQAAABXQVZFZm10IBAAAAABAAEARKwAAIhYAQACABAAZGF0YQAAAAA="; // Empty WAV header + + const flowData = { + name: "Whisper Transcription Test", + nodes: [ + { + id: "whisper1", + type: "whisper-transcription", + data: { + model: "whisper-1", + language: "en", + responseFormat: "text", + temperature: 0 + } + }, + { + id: "output1", + type: "output", + data: {} + } + ], + connections: [ + { source: "whisper1", target: "output1", sourceHandle: "output", targetHandle: "input" } + ] + }; + + try { + const result = await runner.executeFlow(flowData, { + audio: `data:audio/wav;base64,${mockAudioData}` + }); + console.log('[SUCCESS] Whisper Transcription Result:', result); + + // Test with different configurations + const configs = [ + { responseFormat: "json", temperature: 0.2 }, + { language: "es", prompt: "Spanish audio" }, + { model: "whisper-1", responseFormat: "verbose_json" } + ]; + + for (const config of configs) { + const testFlow = { + ...flowData, + nodes: flowData.nodes.map(node => + node.id === 'whisper1' + ? { ...node, data: { ...node.data, ...config } } + : node + ) + }; + + const configResult = await runner.executeFlow(testFlow, { + audio: `data:audio/wav;base64,${mockAudioData}` + }); + console.log(`[SUCCESS] Config ${JSON.stringify(config)}:`, configResult.output1?.text || configResult.output1?.note); + } + + } catch (error) { + console.error('[ERROR] Whisper Transcription test failed:', error.message); + } +} + +// Test 4: Integration Test - All new nodes together +async function testIntegration() { + console.log('\n=== Testing Integration of New Nodes ==='); + + const testText = "Audio transcription result"; + const base64Data = btoa(testText); + const mockAudioData = "UklGRiQAAABXQVZFZm10IBAAAAABAAEARKwAAIhYAQACABAAZGF0YQAAAAA="; + + const flowData = { + name: "Integration Test", + nodes: [ + { + id: "file1", + type: "file-upload", + data: { outputFormat: "text" } + }, + { + id: "whisper1", + type: "whisper-transcription", + data: { model: "whisper-1" } + }, + { + id: "combine1", + type: "combine-text", + data: { mode: "newline" } + }, + { + id: "output1", + type: "output", + data: {} + } + ], + connections: [ + { source: "file1", target: "combine1", sourceHandle: "output", targetHandle: "text1" }, + { source: "whisper1", target: "combine1", sourceHandle: "output", targetHandle: "text2" }, + { source: "combine1", target: "output1", sourceHandle: "output", targetHandle: "input" } + ] + }; + + try { + const result = await runner.executeFlow(flowData, { + file: `data:text/plain;base64,${base64Data}`, + audio: `data:audio/wav;base64,${mockAudioData}` + }); + console.log('[SUCCESS] Integration Test Result:', result); + + } catch (error) { + console.error('[ERROR] Integration test failed:', error.message); + } +} + +// Test 5: Error Handling +async function testErrorHandling() { + console.log('\n=== Testing Error Handling ==='); + + // Test file upload with oversized file + try { + const largeData = 'x'.repeat(1000000); // 1MB of data + const base64Data = btoa(largeData); + + const flowData = { + name: "Error Test", + nodes: [ + { + id: "upload1", + type: "file-upload", + data: { + outputFormat: "text", + maxSize: 1000 // Very small limit + } + } + ], + connections: [] + }; + + await runner.executeFlow(flowData, { + file: `data:text/plain;base64,${base64Data}` + }); + + console.log('[ERROR] Should have failed with file size error'); + + } catch (error) { + console.log('[SUCCESS] Correctly caught file size error:', error.message); + } + + // Test combine text with missing inputs + try { + const flowData = { + name: "Error Test 2", + nodes: [ + { + id: "combine1", + type: "combine-text", + data: { mode: "space" } + } + ], + connections: [] + }; + + const result = await runner.executeFlow(flowData, {}); + console.log('[SUCCESS] Combine text handled empty inputs:', result); + + } catch (error) { + console.log('[INFO] Combine text error handling:', error.message); + } +} + +// Run all tests +async function runAllTests() { + try { + await testCombineTextNode(); + await testFileUploadNode(); + await testWhisperTranscriptionNode(); + await testIntegration(); + await testErrorHandling(); + + console.log('\n[SUCCESS] All new node tests completed!'); + console.log('[INFO] SDK version 1.4.0 with new nodes is ready for release'); + + } catch (error) { + console.error('\n[ERROR] Test suite failed:', error.message); + process.exit(1); + } +} + +// Check if running directly +if (import.meta.url === `file://${process.argv[1]}`) { + runAllTests(); +} + +export { runAllTests }; \ No newline at end of file diff --git a/sdk/test-real-workflow.js b/sdk/test-real-workflow.js new file mode 100644 index 00000000..7b206a6a --- /dev/null +++ b/sdk/test-real-workflow.js @@ -0,0 +1,109 @@ +/** + * Clara Flow SDK v2.0 - Real Workflow Test + * Testing with actual exported workflow from Clara Studio + */ + +import { ClaraFlowRunner } from './dist/index.js'; +import fs from 'fs'; + +console.log('🧠 Testing Real Clara Studio Workflow\n'); + +// Load the actual exported workflow +const workflowData = JSON.parse(fs.readFileSync('agent_exported/Testing_SDK_flow_sdk.json', 'utf8')); + +async function testRealWorkflow() { + const runner = new ClaraFlowRunner({ enableLogging: true }); + + console.log('=' .repeat(60)); + console.log('📋 ANALYZING REAL CLARA STUDIO WORKFLOW'); + console.log('=' .repeat(60)); + + // Analyze the workflow + const description = runner.describe(workflowData); + console.log(`📝 Name: ${description.name}`); + console.log(`📄 Description: ${description.description}`); + console.log(`🔧 Complexity: ${description.complexity}`); + console.log(`🧩 Nodes: ${description.nodeCount}`); + console.log(`🤖 Uses AI: ${description.hasAI ? 'Yes' : 'No'}`); + if (description.hasAI) { + console.log(`🔮 AI Models: ${description.aiModels.join(', ')}`); + } + console.log(`🎨 Custom Nodes: ${description.hasCustomNodes ? 'Yes' : 'No'}`); + console.log(); + + // Show required inputs + const requiredInputs = runner.getRequiredInputs(workflowData); + console.log('📥 Required Inputs:'); + if (requiredInputs.length === 0) { + console.log(' ✨ No inputs required - this workflow uses default values!'); + } else { + requiredInputs.forEach(input => { + const status = input.required ? '🔴 REQUIRED' : '🟢 OPTIONAL'; + console.log(` ${status} ${input.name} (${input.type})`); + console.log(` 📝 ${input.description}`); + if (input.defaultValue) { + console.log(` 💡 Default: "${input.defaultValue}"`); + } + console.log(` 🎯 Example: "${input.example}"`); + console.log(); + }); + } + + console.log('=' .repeat(60)); + console.log('🚀 RUNNING THE WORKFLOW'); + console.log('=' .repeat(60)); + + try { + // Test 1: Run with default values + console.log('📍 Test 1: Using default input value'); + const result1 = await runner.run(workflowData, {}); + console.log('✅ Result 1:', JSON.stringify(result1, null, 2)); + console.log(); + + // Test 2: Override input value + console.log('📍 Test 2: Override input with custom message'); + const result2 = await runner.run(workflowData, { + '1753607451076-xzng2gkp3': 'Tell me a joke about programming!' + }); + console.log('✅ Result 2:', JSON.stringify(result2, null, 2)); + console.log(); + + // Test 3: Use input name instead of ID + console.log('📍 Test 3: Using input name instead of ID'); + const result3 = await runner.run(workflowData, { + 'Input': 'What is the meaning of life?' + }); + console.log('✅ Result 3:', JSON.stringify(result3, null, 2)); + console.log(); + + } catch (error) { + console.error('❌ Workflow execution failed:', error.message); + console.log('\n🔍 Troubleshooting:'); + console.log('1. Check if API endpoint is accessible'); + console.log('2. Verify API key is correct'); + console.log('3. Ensure model is available'); + console.log('\n💡 This is expected if the AI API is not configured'); + } + + console.log('=' .repeat(60)); + console.log('💡 DEVELOPER SUMMARY'); + console.log('=' .repeat(60)); + console.log('✨ What the SDK automatically detected:'); + console.log(` • Workflow format: ${workflowData.format}`); + console.log(` • Input nodes: ${requiredInputs.length}`); + console.log(` • AI nodes: ${description.hasAI ? 'LLM Chat node' : 'None'}`); + console.log(` • Custom nodes: ${description.hasCustomNodes ? 'Yes' : 'None'}`); + console.log(); + console.log('🎯 For developers:'); + console.log(' const runner = new ClaraFlowRunner();'); + console.log(' const result = await runner.run(workflow, inputs);'); + console.log(); + console.log('🚀 The SDK handled everything automatically:'); + console.log(' ✅ Detected Clara Studio export format'); + console.log(' ✅ Found input requirements'); + console.log(' ✅ Applied default values'); + console.log(' ✅ Executed nodes in correct order'); + console.log(' ✅ Provided detailed execution logs'); +} + +testRealWorkflow().catch(console.error); \ No newline at end of file diff --git a/sdk/test-sdk.js b/sdk/test-sdk.js new file mode 100644 index 00000000..a8aa5a3b --- /dev/null +++ b/sdk/test-sdk.js @@ -0,0 +1,154 @@ +import { ClaraFlowRunner } from './dist/index.esm.js'; + +// Test flow data (using the format from the original error) +const testFlowData = { + format: "clara-sdk", + version: "1.0.0", + flow: { + id: "test-flow", + name: "Personal AI", + description: "Test flow", + nodes: [ + { + id: "input-1", + type: "input", + name: "Input", + position: { x: 100, y: 100 }, + data: { + value: "You are Clara and you are the girl friend of the user and always act like one", + inputType: "text" + }, + inputs: [], + outputs: [ + { + id: "output", + name: "Output", + type: "output", + dataType: "string" + } + ] + }, + { + id: "llm-1", + type: "llm", + name: "LLM", + position: { x: 300, y: 100 }, + data: { + apiKey: "test-key", + model: "gpt-3.5-turbo", + temperature: 0.7 + }, + inputs: [ + { + id: "system", + name: "System", + type: "input", + dataType: "string" + }, + { + id: "user", + name: "User", + type: "input", + dataType: "string" + } + ], + outputs: [ + { + id: "response", + name: "Response", + type: "output", + dataType: "string" + } + ] + }, + { + id: "output-1", + type: "output", + name: "Output", + position: { x: 500, y: 100 }, + data: {}, + inputs: [ + { + id: "input", + name: "Input", + type: "input", + dataType: "string" + } + ], + outputs: [] + } + ], + connections: [ + { + id: "conn-1", + sourceNodeId: "input-1", + sourcePortId: "output", + targetNodeId: "llm-1", + targetPortId: "system" + }, + { + id: "conn-2", + sourceNodeId: "llm-1", + sourcePortId: "response", + targetNodeId: "output-1", + targetPortId: "input" + } + ] + } +}; + +async function testSDK() { + console.log('[INFO] Testing Clara Flow SDK...'); + + try { + // Create SDK instance + const runner = new ClaraFlowRunner({ + enableLogging: true, + logLevel: 'info' + }); + + console.log('[INFO] SDK instance created successfully'); + + // Test validation + const validation = runner.validateFlow(testFlowData); + console.log('[INFO] Flow validation:', validation); + + if (!validation.isValid) { + console.log('[ERROR] Flow validation failed:', validation.errors); + return; + } + + // Test execution (this will fail due to invalid API key, but should get past the "unknown node type" error) + console.log('[INFO] Starting flow execution...'); + + try { + const result = await runner.executeFlow(testFlowData, { + user: "Hello Clara!" + }); + + console.log('[SUCCESS] Flow executed successfully:', result); + } catch (error) { + // We expect this to fail due to invalid API key, but it should NOT be "Unknown node type: llm" + console.log('[INFO] Flow execution failed (expected due to test API key):', error.message); + + if (error.message.includes('Unknown node type')) { + console.log('[ERROR] SDK is missing node type definitions - this is the bug we fixed!'); + process.exit(1); + } else { + console.log('[SUCCESS] SDK recognizes all node types correctly!'); + } + } + + console.log('[SUCCESS] SDK test completed successfully!'); + + } catch (error) { + console.log('[ERROR] SDK test failed:', error.message); + process.exit(1); + } +} + +// Run the test +testSDK().catch(error => { + console.error('[ERROR] Test failed:', error); + process.exit(1); +}); \ No newline at end of file diff --git a/sdk/test-with-example.js b/sdk/test-with-example.js new file mode 100644 index 00000000..faf4bd01 --- /dev/null +++ b/sdk/test-with-example.js @@ -0,0 +1,161 @@ +#!/usr/bin/env node + +/** + * Test Clara Flow SDK with Example JSON + * + * This script tests the SDK using the provided example JSON file. + */ + +import { ClaraFlowRunner } from './dist/index.esm.js'; +import fs from 'fs'; +import path from 'path'; + +async function testWithExampleJSON() { + console.log('🚀 Testing Clara Flow SDK with Example JSON\n'); + + try { + // Initialize the flow runner with logging enabled + const runner = new ClaraFlowRunner({ + enableLogging: true, + logLevel: 'info', + timeout: 30000 + }); + + console.log('✅ Flow runner initialized'); + + // Load the example JSON flow + const exampleFlowPath = './examples/Testing_flow_sdk.json'; + console.log(`📄 Loading flow from: ${exampleFlowPath}`); + + const rawFlowData = JSON.parse(fs.readFileSync(exampleFlowPath, 'utf8')); + + // Extract the flow data from the wrapper and adjust format + const flowData = { + ...rawFlowData.flow, // Extract the flow object + version: '1.0.0', // Use supported version + // Keep connections if they exist + connections: rawFlowData.flow.connections || [], + // Keep custom nodes if they exist + customNodes: rawFlowData.customNodes || [] + }; + + console.log(`✅ Flow loaded: ${flowData.name}`); + console.log(`📊 Flow contains ${flowData.nodes.length} nodes`); + + // Display flow information + console.log('\n📋 Flow Details:'); + console.log(` Name: ${flowData.name}`); + console.log(` Description: ${flowData.description}`); + console.log(` Version: ${flowData.version}`); + + // Show nodes in the flow + console.log('\n🔧 Nodes in flow:'); + flowData.nodes.forEach((node, index) => { + console.log(` ${index + 1}. ${node.name} (${node.type}) - ID: ${node.id}`); + }); + + // Show connections if they exist + if (rawFlowData.flow.connections && rawFlowData.flow.connections.length > 0) { + console.log('\n🔗 Connections:'); + rawFlowData.flow.connections.forEach((conn, index) => { + console.log(` ${index + 1}. ${conn.sourceNodeId} -> ${conn.targetNodeId}`); + }); + } + + // Prepare input data based on the flow's input nodes + const inputNodes = flowData.nodes.filter(node => node.type === 'input'); + const inputs = {}; + + console.log('\n📥 Available inputs:'); + inputNodes.forEach((node, index) => { + const inputValue = node.data?.value || `Input ${index + 1}`; + inputs[node.name] = inputValue; + console.log(` ${node.name} (${node.id}): "${inputValue}"`); + }); + + console.log('\n🔧 Executing flow with inputs:', inputs); + + // Execute the flow + const startTime = Date.now(); + const result = await runner.executeFlow(flowData, inputs); + const endTime = Date.now(); + + console.log('\n✅ Flow execution completed!'); + console.log('📊 Results:', JSON.stringify(result.results || result, null, 2)); + console.log('⏱️ Execution time:', endTime - startTime, 'ms'); + + // Get execution logs + const logs = runner.getLogs(); + if (logs && logs.length > 0) { + console.log('\n📝 Execution logs:'); + logs.forEach(log => { + console.log(`[${log.level.toUpperCase()}] ${log.message}`); + }); + } + + console.log('\n🎉 Test completed successfully!'); + + } catch (error) { + console.error('❌ Error during execution:', error.message); + console.error('Stack trace:', error.stack); + process.exit(1); + } +} + +// Additional validation function +async function validateFlowStructure() { + console.log('🔍 Validating flow structure...'); + + try { + const exampleFlowPath = './examples/Testing_flow_sdk.json'; + const rawFlowData = JSON.parse(fs.readFileSync(exampleFlowPath, 'utf8')); + + // Basic structure validation + const requiredFields = ['format', 'version', 'flow']; + const flowRequiredFields = ['id', 'name', 'nodes']; + + console.log('✅ Checking required top-level fields...'); + requiredFields.forEach(field => { + if (!rawFlowData[field]) { + throw new Error(`Missing required field: ${field}`); + } + console.log(` ✓ ${field}: ${typeof rawFlowData[field] === 'object' ? 'object' : rawFlowData[field]}`); + }); + + console.log('✅ Checking required flow fields...'); + flowRequiredFields.forEach(field => { + if (!rawFlowData.flow[field]) { + throw new Error(`Missing required flow field: ${field}`); + } + console.log(` ✓ flow.${field}: ${typeof rawFlowData.flow[field] === 'object' ? `${rawFlowData.flow[field].length} items` : rawFlowData.flow[field]}`); + }); + + // Node validation + console.log('✅ Validating nodes...'); + rawFlowData.flow.nodes.forEach((node, index) => { + if (!node.id || !node.type || !node.name) { + throw new Error(`Node ${index} missing required fields (id, type, name)`); + } + console.log(` ✓ Node ${index + 1}: ${node.name} (${node.type})`); + }); + + console.log('🎯 Flow structure validation passed!\n'); + + } catch (error) { + console.error('❌ Flow validation failed:', error.message); + throw error; + } +} + +// Run the tests +async function main() { + try { + await validateFlowStructure(); + await testWithExampleJSON(); + } catch (error) { + console.error('❌ Test failed:', error.message); + process.exit(1); + } +} + +main(); \ No newline at end of file diff --git a/sdk/test/basic.test.js b/sdk/test/basic.test.js new file mode 100644 index 00000000..38632c2c --- /dev/null +++ b/sdk/test/basic.test.js @@ -0,0 +1,388 @@ +/** + * Basic test for Clara Flow SDK v2.0 + */ + +import { ClaraFlowRunner, BrowserUtils } from '../src/index.js'; + +console.log('🧪 Testing Clara Flow SDK v2.0...\n'); + +// Test 1: SDK Initialization +console.log('Test 1: SDK Initialization'); +try { + const runner = new ClaraFlowRunner({ + enableLogging: true, + timeout: 5000 + }); + console.log('✅ SDK initialized successfully'); +} catch (error) { + console.error('❌ SDK initialization failed:', error.message); + process.exit(1); +} + +// Test 2: Simple workflow execution +console.log('\nTest 2: Simple Workflow Execution'); +try { + const runner = new ClaraFlowRunner({ enableLogging: true }); + + // Simple workflow: Input -> Static Text -> Output + const simpleFlow = { + nodes: [ + { + id: 'input-1', + type: 'input', + name: 'User Input', + data: { value: 'Hello from Clara SDK!' }, + inputs: [], + outputs: [{ id: 'output', name: 'Output', dataType: 'string' }] + }, + { + id: 'static-1', + type: 'static-text', + name: 'Static Text', + data: { text: 'Processing: ' }, + inputs: [{ id: 'input', name: 'Input', dataType: 'string' }], + outputs: [{ id: 'output', name: 'Output', dataType: 'string' }] + }, + { + id: 'combine-1', + type: 'combine-text', + name: 'Combine', + data: { separator: '' }, + inputs: [ + { id: 'text1', name: 'Text1', dataType: 'string' }, + { id: 'text2', name: 'Text2', dataType: 'string' } + ], + outputs: [{ id: 'output', name: 'Output', dataType: 'string' }] + }, + { + id: 'output-1', + type: 'output', + name: 'Final Output', + inputs: [{ id: 'input', name: 'Input', dataType: 'string' }], + outputs: [] + } + ], + connections: [ + { + id: 'conn-1', + sourceNodeId: 'static-1', + sourcePortId: 'output', + targetNodeId: 'combine-1', + targetPortId: 'text1' + }, + { + id: 'conn-2', + sourceNodeId: 'input-1', + sourcePortId: 'output', + targetNodeId: 'combine-1', + targetPortId: 'text2' + }, + { + id: 'conn-3', + sourceNodeId: 'combine-1', + sourcePortId: 'output', + targetNodeId: 'output-1', + targetPortId: 'input' + } + ] + }; + + const result = await runner.execute(simpleFlow, { + 'input-1': 'World!' + }); + + console.log('✅ Simple workflow executed successfully'); + console.log('📤 Result:', result); + +} catch (error) { + console.error('❌ Simple workflow execution failed:', error.message); + process.exit(1); +} + +// Test 3: JSON parsing workflow +console.log('\nTest 3: JSON Parsing Workflow'); +try { + const runner = new ClaraFlowRunner({ enableLogging: false }); + + const jsonFlow = { + nodes: [ + { + id: 'input-1', + type: 'input', + name: 'JSON Input', + data: { value: '{"user": {"name": "Alice", "age": 30}}' }, + inputs: [], + outputs: [{ id: 'output', name: 'Output', dataType: 'string' }] + }, + { + id: 'parse-1', + type: 'json-parse', + name: 'Parse JSON', + data: { field: 'user.name' }, + inputs: [{ id: 'input', name: 'JSON', dataType: 'string' }], + outputs: [{ id: 'output', name: 'Output', dataType: 'any' }] + }, + { + id: 'output-1', + type: 'output', + name: 'Parsed Output', + inputs: [{ id: 'input', name: 'Input', dataType: 'any' }], + outputs: [] + } + ], + connections: [ + { + id: 'conn-1', + sourceNodeId: 'input-1', + sourcePortId: 'output', + targetNodeId: 'parse-1', + targetPortId: 'input' + }, + { + id: 'conn-2', + sourceNodeId: 'parse-1', + sourcePortId: 'output', + targetNodeId: 'output-1', + targetPortId: 'input' + } + ] + }; + + const result = await runner.execute(jsonFlow); + console.log('✅ JSON parsing workflow executed successfully'); + console.log('📤 Extracted name:', result['output-1']?.output); + +} catch (error) { + console.error('❌ JSON parsing workflow failed:', error.message); + process.exit(1); +} + +// Test 4: If/Else conditional workflow +console.log('\nTest 4: If/Else Conditional Workflow'); +try { + const runner = new ClaraFlowRunner({ enableLogging: false }); + + const conditionalFlow = { + nodes: [ + { + id: 'input-1', + type: 'input', + name: 'Number Input', + data: { value: 42 }, + inputs: [], + outputs: [{ id: 'output', name: 'Output', dataType: 'number' }] + }, + { + id: 'condition-1', + type: 'if-else', + name: 'Check Even', + data: { + expression: 'input % 2 === 0', + trueValue: 'Even number', + falseValue: 'Odd number' + }, + inputs: [{ id: 'input', name: 'Input', dataType: 'number' }], + outputs: [ + { id: 'true', name: 'True', dataType: 'string' }, + { id: 'false', name: 'False', dataType: 'string' }, + { id: 'output', name: 'Output', dataType: 'string' } + ] + }, + { + id: 'output-1', + type: 'output', + name: 'Result', + inputs: [{ id: 'input', name: 'Input', dataType: 'string' }], + outputs: [] + } + ], + connections: [ + { + id: 'conn-1', + sourceNodeId: 'input-1', + sourcePortId: 'output', + targetNodeId: 'condition-1', + targetPortId: 'input' + }, + { + id: 'conn-2', + sourceNodeId: 'condition-1', + sourcePortId: 'output', + targetNodeId: 'output-1', + targetPortId: 'input' + } + ] + }; + + const result = await runner.execute(conditionalFlow); + console.log('✅ Conditional workflow executed successfully'); + console.log('📤 Result:', result['output-1']?.output); + +} catch (error) { + console.error('❌ Conditional workflow failed:', error.message); + process.exit(1); +} + +// Test 5: Custom node registration and execution +console.log('\nTest 5: Custom Node Registration'); +try { + const runner = new ClaraFlowRunner({ enableLogging: false }); + + // Register a custom node + runner.registerCustomNode({ + type: 'multiply', + name: 'Multiply Numbers', + executionCode: ` + function execute(inputs, properties, context) { + const a = parseFloat(inputs.a) || 0; + const b = parseFloat(inputs.b) || 0; + const factor = parseFloat(properties.factor) || 1; + const result = a * b * factor; + context.log('Multiplying: ' + a + ' * ' + b + ' * ' + factor + ' = ' + result); + return { output: result }; + } + ` + }); + + const customFlow = { + nodes: [ + { + id: 'input-a', + type: 'input', + name: 'Input A', + data: { value: 6 }, + inputs: [], + outputs: [{ id: 'output', name: 'Output', dataType: 'number' }] + }, + { + id: 'input-b', + type: 'input', + name: 'Input B', + data: { value: 7 }, + inputs: [], + outputs: [{ id: 'output', name: 'Output', dataType: 'number' }] + }, + { + id: 'multiply-1', + type: 'multiply', + name: 'Custom Multiply', + data: { factor: 2 }, + inputs: [ + { id: 'a', name: 'A', dataType: 'number' }, + { id: 'b', name: 'B', dataType: 'number' } + ], + outputs: [{ id: 'output', name: 'Output', dataType: 'number' }] + }, + { + id: 'output-1', + type: 'output', + name: 'Final Result', + inputs: [{ id: 'input', name: 'Input', dataType: 'number' }], + outputs: [] + } + ], + connections: [ + { + id: 'conn-1', + sourceNodeId: 'input-a', + sourcePortId: 'output', + targetNodeId: 'multiply-1', + targetPortId: 'a' + }, + { + id: 'conn-2', + sourceNodeId: 'input-b', + sourcePortId: 'output', + targetNodeId: 'multiply-1', + targetPortId: 'b' + }, + { + id: 'conn-3', + sourceNodeId: 'multiply-1', + sourcePortId: 'output', + targetNodeId: 'output-1', + targetPortId: 'input' + } + ] + }; + + const result = await runner.execute(customFlow); + console.log('✅ Custom node workflow executed successfully'); + console.log('📤 Result (6 * 7 * 2):', result['output-1']?.output); + +} catch (error) { + console.error('❌ Custom node workflow failed:', error.message); + process.exit(1); +} + +// Test 6: Clara Studio export format compatibility +console.log('\nTest 6: Clara Studio Export Format'); +try { + const runner = new ClaraFlowRunner({ enableLogging: false }); + + // Simulate Clara Studio export format + const studioExport = { + format: 'clara-sdk', + version: '1.0.0', + flow: { + id: 'test-flow', + name: 'Test Flow', + nodes: [ + { + id: 'input-1', + type: 'input', + name: 'Test Input', + data: { value: 'Clara Studio Export Test' }, + inputs: [], + outputs: [{ id: 'output', name: 'Output', dataType: 'string' }] + }, + { + id: 'output-1', + type: 'output', + name: 'Test Output', + inputs: [{ id: 'input', name: 'Input', dataType: 'string' }], + outputs: [] + } + ], + connections: [ + { + id: 'conn-1', + sourceNodeId: 'input-1', + sourcePortId: 'output', + targetNodeId: 'output-1', + targetPortId: 'input' + } + ] + }, + customNodes: [], + metadata: { + exportedAt: new Date().toISOString(), + exportedBy: 'Clara Agent Studio' + } + }; + + const result = await runner.execute(studioExport); + console.log('✅ Clara Studio export format handled successfully'); + console.log('📤 Result:', result['output-1']?.output); + +} catch (error) { + console.error('❌ Clara Studio export format failed:', error.message); + process.exit(1); +} + +console.log('\n🎉 All tests passed! Clara Flow SDK v2.0 is working correctly.'); +console.log('\n📋 SDK Features Verified:'); +console.log('✅ Basic workflow execution'); +console.log('✅ JSON parsing with dot notation'); +console.log('✅ Conditional logic (if/else)'); +console.log('✅ Custom node registration and execution'); +console.log('✅ Clara Studio export format compatibility'); +console.log('✅ Built-in nodes: input, output, static-text, combine-text, json-parse, if-else'); +console.log('✅ Topological sorting for execution order'); +console.log('✅ Error handling and logging'); + +// Show execution logs example +console.log('\n📝 Sample Execution Logs:'); +const runner = new ClaraFlowRunner({ enableLogging: true }); +const logs = runner.getLogs(); +console.log('Logs count:', logs.length); \ No newline at end of file diff --git a/sdk/tests/integration.test.js b/sdk/tests/integration.test.js new file mode 100644 index 00000000..b9c29d25 --- /dev/null +++ b/sdk/tests/integration.test.js @@ -0,0 +1,520 @@ +/** + * Clara Flow SDK - Integration Tests + */ + +import { ClaraFlowRunner } from '../src/index.js'; + +describe('Clara Flow SDK Integration Tests', () => { + let runner; + + beforeEach(() => { + runner = new ClaraFlowRunner({ + enableLogging: false, // Disable logging in tests + timeout: 5000 + }); + }); + + afterEach(() => { + if (runner) { + runner.clearLogs(); + } + }); + + describe('Flow Execution', () => { + test('should execute simple flow with input and output nodes', async () => { + const simpleFlow = { + version: '1.0.0', + name: 'Simple Test Flow', + exportFormat: 'clara-sdk', + nodes: [ + { + id: 'input-1', + type: 'input', + name: 'Test Input', + data: { inputType: 'string', defaultValue: 'test' }, + position: { x: 100, y: 100 } + }, + { + id: 'output-1', + type: 'output', + name: 'Test Output', + data: {}, + position: { x: 300, y: 100 } + } + ], + connections: [ + { + id: 'conn-1', + sourceNodeId: 'input-1', + sourcePortId: 'output', + targetNodeId: 'output-1', + targetPortId: 'input' + } + ], + customNodes: [] + }; + + const result = await runner.executeFlow(simpleFlow, { + 'Test Input': 'Hello World' + }); + + expect(result).toBeDefined(); + expect(result.success).toBe(true); + expect(result.results).toBeDefined(); + expect(result.executionTimeMs).toBeGreaterThan(0); + }); + + test('should execute flow with custom node', async () => { + const customNodeFlow = { + version: '1.0.0', + name: 'Custom Node Test Flow', + exportFormat: 'clara-sdk', + nodes: [ + { + id: 'input-1', + type: 'input', + name: 'Text Input', + data: { inputType: 'string', defaultValue: 'hello' }, + position: { x: 100, y: 100 } + }, + { + id: 'custom-1', + type: 'text-transformer', + name: 'Text Transformer', + data: { operation: 'uppercase' }, + position: { x: 300, y: 100 } + }, + { + id: 'output-1', + type: 'output', + name: 'Result Output', + data: {}, + position: { x: 500, y: 100 } + } + ], + connections: [ + { + id: 'conn-1', + sourceNodeId: 'input-1', + sourcePortId: 'output', + targetNodeId: 'custom-1', + targetPortId: 'input' + }, + { + id: 'conn-2', + sourceNodeId: 'custom-1', + sourcePortId: 'output', + targetNodeId: 'output-1', + targetPortId: 'input' + } + ], + customNodes: [ + { + type: 'text-transformer', + name: 'Text Transformer', + description: 'Transforms text using various operations', + inputs: [{ name: 'input', type: 'string', required: true }], + outputs: [{ name: 'output', type: 'string' }], + properties: [{ name: 'operation', type: 'string', defaultValue: 'uppercase' }], + executionCode: ` + async function execute(inputs, properties, context) { + const text = inputs.input || ''; + const operation = properties.operation || 'uppercase'; + + let result; + switch (operation) { + case 'uppercase': + result = text.toUpperCase(); + break; + case 'lowercase': + result = text.toLowerCase(); + break; + default: + result = text; + } + + return { output: result }; + } + ` + } + ] + }; + + const result = await runner.executeFlow(customNodeFlow, { + 'Text Input': 'hello world' + }); + + expect(result).toBeDefined(); + expect(result.success).toBe(true); + expect(result.results['Result Output']).toBe('HELLO WORLD'); + }); + + test('should handle async custom nodes', async () => { + const asyncFlow = { + version: '1.0.0', + name: 'Async Test Flow', + exportFormat: 'clara-sdk', + nodes: [ + { + id: 'input-1', + type: 'input', + name: 'Delay Input', + data: { inputType: 'number', defaultValue: 100 }, + position: { x: 100, y: 100 } + }, + { + id: 'custom-1', + type: 'delay-processor', + name: 'Delay Processor', + data: {}, + position: { x: 300, y: 100 } + }, + { + id: 'output-1', + type: 'output', + name: 'Delayed Output', + data: {}, + position: { x: 500, y: 100 } + } + ], + connections: [ + { + id: 'conn-1', + sourceNodeId: 'input-1', + sourcePortId: 'output', + targetNodeId: 'custom-1', + targetPortId: 'input' + }, + { + id: 'conn-2', + sourceNodeId: 'custom-1', + sourcePortId: 'output', + targetNodeId: 'output-1', + targetPortId: 'input' + } + ], + customNodes: [ + { + type: 'delay-processor', + name: 'Delay Processor', + description: 'Adds a delay before processing', + inputs: [{ name: 'input', type: 'number', required: true }], + outputs: [{ name: 'output', type: 'string' }], + executionCode: ` + async function execute(inputs, properties, context) { + const delay = parseInt(inputs.input) || 100; + + await new Promise(resolve => setTimeout(resolve, delay)); + + return { output: 'Processed after ' + delay + 'ms delay' }; + } + ` + } + ] + }; + + const startTime = Date.now(); + const result = await runner.executeFlow(asyncFlow, { + 'Delay Input': 200 + }); + const endTime = Date.now(); + + expect(result).toBeDefined(); + expect(result.success).toBe(true); + expect(endTime - startTime).toBeGreaterThanOrEqual(200); + expect(result.results['Delayed Output']).toContain('200ms delay'); + }); + }); + + describe('Flow Validation', () => { + test('should validate valid flow', () => { + const validFlow = { + version: '1.0.0', + name: 'Valid Test Flow', + nodes: [ + { + id: 'input-1', + type: 'input', + name: 'Input', + data: {}, + position: { x: 100, y: 100 } + }, + { + id: 'output-1', + type: 'output', + name: 'Output', + data: {}, + position: { x: 300, y: 100 } + } + ], + connections: [ + { + id: 'conn-1', + sourceNodeId: 'input-1', + sourcePortId: 'output', + targetNodeId: 'output-1', + targetPortId: 'input' + } + ], + customNodes: [] + }; + + const validation = runner.validateFlow(validFlow); + + expect(validation.isValid).toBe(true); + expect(validation.errors).toHaveLength(0); + expect(validation.summary).toBeDefined(); + expect(validation.summary.flowName).toBe('Valid Test Flow'); + }); + + test('should detect validation errors', () => { + const invalidFlow = { + // Missing required fields + nodes: [ + { + // Missing id and type + name: 'Invalid Node' + } + ], + connections: [], + customNodes: [] + }; + + const validation = runner.validateFlow(invalidFlow); + + expect(validation.isValid).toBe(false); + expect(validation.errors.length).toBeGreaterThan(0); + }); + + test('should detect circular dependencies', () => { + const circularFlow = { + version: '1.0.0', + name: 'Circular Flow', + nodes: [ + { + id: 'node-1', + type: 'custom', + name: 'Node 1', + data: {}, + position: { x: 100, y: 100 } + }, + { + id: 'node-2', + type: 'custom', + name: 'Node 2', + data: {}, + position: { x: 300, y: 100 } + } + ], + connections: [ + { + id: 'conn-1', + sourceNodeId: 'node-1', + sourcePortId: 'output', + targetNodeId: 'node-2', + targetPortId: 'input' + }, + { + id: 'conn-2', + sourceNodeId: 'node-2', + sourcePortId: 'output', + targetNodeId: 'node-1', + targetPortId: 'input' + } + ], + customNodes: [] + }; + + const validation = runner.validateFlow(circularFlow); + + expect(validation.isValid).toBe(false); + expect(validation.errors.some(error => error.includes('circular'))).toBe(true); + }); + }); + + describe('Error Handling', () => { + test('should handle execution timeout', async () => { + const timeoutRunner = new ClaraFlowRunner({ + enableLogging: false, + timeout: 100 // Very short timeout + }); + + const longRunningFlow = { + version: '1.0.0', + name: 'Long Running Flow', + exportFormat: 'clara-sdk', + nodes: [ + { + id: 'input-1', + type: 'input', + name: 'Input', + data: { defaultValue: 'test' }, + position: { x: 100, y: 100 } + }, + { + id: 'custom-1', + type: 'slow-processor', + name: 'Slow Processor', + data: {}, + position: { x: 300, y: 100 } + } + ], + connections: [ + { + id: 'conn-1', + sourceNodeId: 'input-1', + sourcePortId: 'output', + targetNodeId: 'custom-1', + targetPortId: 'input' + } + ], + customNodes: [ + { + type: 'slow-processor', + name: 'Slow Processor', + executionCode: ` + async function execute(inputs, properties, context) { + await new Promise(resolve => setTimeout(resolve, 1000)); // 1 second delay + return { output: 'Done' }; + } + ` + } + ] + }; + + await expect(timeoutRunner.executeFlow(longRunningFlow)).rejects.toThrow('timeout'); + }); + + test('should handle invalid custom node code', async () => { + const invalidCodeFlow = { + version: '1.0.0', + name: 'Invalid Code Flow', + exportFormat: 'clara-sdk', + nodes: [ + { + id: 'input-1', + type: 'input', + name: 'Input', + data: { defaultValue: 'test' }, + position: { x: 100, y: 100 } + }, + { + id: 'custom-1', + type: 'broken-processor', + name: 'Broken Processor', + data: {}, + position: { x: 300, y: 100 } + } + ], + connections: [ + { + id: 'conn-1', + sourceNodeId: 'input-1', + sourcePortId: 'output', + targetNodeId: 'custom-1', + targetPortId: 'input' + } + ], + customNodes: [ + { + type: 'broken-processor', + name: 'Broken Processor', + executionCode: ` + // Invalid JavaScript syntax + async function execute(inputs, properties, context) { + invalid syntax here!!! + return { output: 'This will not work' }; + } + ` + } + ] + }; + + await expect(runner.executeFlow(invalidCodeFlow)).rejects.toThrow(); + }); + }); + + describe('Logging', () => { + test('should capture execution logs when enabled', async () => { + const loggingRunner = new ClaraFlowRunner({ + enableLogging: true, + logLevel: 'debug' + }); + + const simpleFlow = { + version: '1.0.0', + name: 'Logging Test Flow', + exportFormat: 'clara-sdk', + nodes: [ + { + id: 'input-1', + type: 'input', + name: 'Input', + data: { defaultValue: 'test' }, + position: { x: 100, y: 100 } + } + ], + connections: [], + customNodes: [] + }; + + await loggingRunner.executeFlow(simpleFlow); + + const logs = loggingRunner.getLogs(); + expect(logs.length).toBeGreaterThan(0); + }); + + test('should support different log levels', () => { + const debugRunner = new ClaraFlowRunner({ + enableLogging: true, + logLevel: 'debug' + }); + + const errorRunner = new ClaraFlowRunner({ + enableLogging: true, + logLevel: 'error' + }); + + // Log messages at different levels + debugRunner.logger.debug('Debug message'); + debugRunner.logger.info('Info message'); + debugRunner.logger.warn('Warning message'); + debugRunner.logger.error('Error message'); + + errorRunner.logger.debug('Debug message'); + errorRunner.logger.info('Info message'); + errorRunner.logger.warn('Warning message'); + errorRunner.logger.error('Error message'); + + const debugLogs = debugRunner.getLogs(); + const errorLogs = errorRunner.getLogs(); + + expect(debugLogs.length).toBe(4); // All messages + expect(errorLogs.length).toBe(1); // Only error message + }); + }); + + describe('Configuration', () => { + test('should respect custom timeout settings', async () => { + const customRunner = new ClaraFlowRunner({ + timeout: 2000, + enableLogging: false + }); + + expect(customRunner.options.timeout).toBe(2000); + }); + + test('should handle sandbox configuration', () => { + const sandboxedRunner = new ClaraFlowRunner({ + sandbox: true + }); + + const unsandboxedRunner = new ClaraFlowRunner({ + sandbox: false + }); + + expect(sandboxedRunner.options.sandbox).toBe(true); + expect(unsandboxedRunner.options.sandbox).toBe(false); + }); + }); +}); \ No newline at end of file diff --git a/sdk/tsconfig.json b/sdk/tsconfig.json new file mode 100644 index 00000000..8608f9f3 --- /dev/null +++ b/sdk/tsconfig.json @@ -0,0 +1,30 @@ +{ + "compilerOptions": { + "target": "ES2020", + "module": "ESNext", + "moduleResolution": "node", + "lib": ["ES2020", "DOM"], + "outDir": "./dist/tsc", + "strict": true, + "esModuleInterop": true, + "skipLibCheck": true, + "forceConsistentCasingInFileNames": true, + "declaration": true, + "declarationDir": "./dist", + "declarationMap": true, + "sourceMap": true, + "removeComments": false, + "resolveJsonModule": true, + "allowSyntheticDefaultImports": true + }, + "include": [ + "src/**/*", + "../src/shared/**/*", + "../src/types/**/*" + ], + "exclude": [ + "node_modules", + "dist", + "**/*.test.ts" + ] +} \ No newline at end of file diff --git a/sdk/widgets/README.md b/sdk/widgets/README.md new file mode 100644 index 00000000..e69de29b diff --git a/sdk/widgets/examples/docker-status-widget.json b/sdk/widgets/examples/docker-status-widget.json new file mode 100644 index 00000000..e69de29b diff --git a/sdk/widgets/examples/gpu-temperature-widget-fixed.json b/sdk/widgets/examples/gpu-temperature-widget-fixed.json new file mode 100644 index 00000000..e69de29b diff --git a/sdk/widgets/examples/gpu-temperature-widget.json b/sdk/widgets/examples/gpu-temperature-widget.json new file mode 100644 index 00000000..e69de29b diff --git a/sdk/widgets/examples/index.ts b/sdk/widgets/examples/index.ts new file mode 100644 index 00000000..e69de29b diff --git a/sdk/widgets/examples/ready-to-use.ts b/sdk/widgets/examples/ready-to-use.ts new file mode 100644 index 00000000..e69de29b diff --git a/sdk/widgets/examples/weather-widget-fixed.json b/sdk/widgets/examples/weather-widget-fixed.json new file mode 100644 index 00000000..e69de29b diff --git a/sdk/widgets/examples/weather-widget.json b/sdk/widgets/examples/weather-widget.json new file mode 100644 index 00000000..e69de29b diff --git a/sdk/widgets/examples/world-clock-widget-fixed.json b/sdk/widgets/examples/world-clock-widget-fixed.json new file mode 100644 index 00000000..e69de29b diff --git a/sdk/widgets/examples/world-clock-widget.json b/sdk/widgets/examples/world-clock-widget.json new file mode 100644 index 00000000..e69de29b diff --git a/sdk/widgets/index.ts b/sdk/widgets/index.ts new file mode 100644 index 00000000..e69de29b diff --git a/sdk/widgets/package.json b/sdk/widgets/package.json new file mode 100644 index 00000000..e69de29b diff --git a/sdk/widgets/templates/index.ts b/sdk/widgets/templates/index.ts new file mode 100644 index 00000000..e69de29b diff --git a/sdk/widgets/types/index.ts b/sdk/widgets/types/index.ts new file mode 100644 index 00000000..e69de29b diff --git a/sdk/widgets/utils/validation.ts b/sdk/widgets/utils/validation.ts new file mode 100644 index 00000000..e69de29b diff --git a/sdk_examples/example-using-js-class.js b/sdk_examples/example-using-js-class.js new file mode 100644 index 00000000..4e439474 --- /dev/null +++ b/sdk_examples/example-using-js-class.js @@ -0,0 +1,179 @@ +/** + * 🚀 Example 2: Using JavaScript Class Export Format + * + * This is for when you want MORE POWER and CONTROL! 💪 + * + * Think of it like this: + * 1. You build a workflow in Clara Agent Studio 🎨 + * 2. You export it as a JavaScript class (like a reusable tool) 🔧 + * 3. You can use it like any other JavaScript module! ⚡ + * + * Perfect for: Developers, reusable components, production apps + */ + +// 📦 Step 1: Import your exported workflow class +import { NewWorkFLowFlow } from './NewWorkFLow_flow.js'; + +async function runAdvancedSentimentAnalysis() { + console.log('🚀 Starting Advanced Sentiment Analysis with JS Class!'); + console.log('='.repeat(55)); + + try { + // 🛠️ Step 2: Create an instance of your workflow (like getting a tool ready) + console.log('🔧 Creating workflow instance...'); + const sentimentWorkflow = new NewWorkFLowFlow({ + enableLogging: true, + logLevel: 'debug' // Show detailed info + }); + + console.log('✅ Workflow instance created!'); + + // 📋 Step 3: Let's see what this workflow can do + const flowInfo = sentimentWorkflow.getFlowInfo(); + console.log(`📊 Workflow Name: ${flowInfo.name}`); + console.log(`📈 Total Nodes: ${flowInfo.nodeCount}`); + console.log(`🔗 Connections: ${flowInfo.connectionCount}`); + + // 🎯 Step 4: Test with different types of feedback + const testCases = [ + { + name: 'Happy Customer', + feedback: 'This product is absolutely fantastic! Best purchase ever!', + context: 'Product review from verified buyer' + }, + { + name: 'Neutral Customer', + feedback: 'The product is okay, nothing special but works fine.', + context: 'Standard product feedback' + }, + { + name: 'Very Unhappy Customer', + feedback: 'This product is terrible! Complete waste of money!', + context: 'Complaint from customer service' + } + ]; + + // 🔄 Step 5: Process each test case + for (let i = 0; i < testCases.length; i++) { + const testCase = testCases[i]; + console.log(`\n🧪 Test Case ${i + 1}: ${testCase.name}`); + console.log('-'.repeat(40)); + + // Prepare the input data for this workflow + const inputs = { + // The workflow expects these specific input names + 'user_feedback': testCase.feedback, + 'json_format': JSON.stringify({ + "sentence": "sentence from the user", + "sentiment": "good, bad, very bad", + "reason": "reason for the choice" + }), + 'analysis_context': testCase.context + }; + + console.log(`📝 Processing: "${testCase.feedback}"`); + + // 🚀 Execute the workflow + const result = await sentimentWorkflow.execute(inputs); + + // 📊 Display results + console.log('📊 Analysis Result:'); + if (result) { + // Pretty print the result + const analysis = typeof result === 'string' ? JSON.parse(result) : result; + console.log(` Sentiment: ${analysis.sentiment || 'Unknown'}`); + console.log(` Reason: ${analysis.reason || 'No reason provided'}`); + + // React based on sentiment + if (analysis.sentiment === 'very bad') { + console.log('🚨 ALERT: Critical feedback - needs immediate attention!'); + } else if (analysis.sentiment === 'bad') { + console.log('⚠️ Warning: Negative feedback - follow up recommended'); + } else { + console.log('✅ Positive/Neutral feedback - all good!'); + } + } + } + + // 🎊 Step 6: Batch processing example + console.log('\n\n🔥 Bonus: Batch Processing Multiple Feedbacks!'); + console.log('='.repeat(50)); + + const batchInputs = [ + { + 'user_feedback': 'Love this app!', + 'json_format': JSON.stringify({ + "sentence": "sentence from the user", + "sentiment": "good, bad, very bad", + "reason": "reason for the choice" + }), + 'analysis_context': 'App store review' + }, + { + 'user_feedback': 'App crashes constantly, very frustrating!', + 'json_format': JSON.stringify({ + "sentence": "sentence from the user", + "sentiment": "good, bad, very bad", + "reason": "reason for the choice" + }), + 'analysis_context': 'Bug report' + } + ]; + + console.log('⚡ Processing multiple feedbacks at once...'); + const batchResults = await sentimentWorkflow.executeBatch(batchInputs, { + maxConcurrency: 2 // Process 2 at a time + }); + + console.log(`✅ Processed ${batchResults.length} feedbacks:`); + batchResults.forEach((result, index) => { + const analysis = typeof result === 'string' ? JSON.parse(result) : result; + console.log(` ${index + 1}. ${analysis.sentiment || 'Unknown'}: ${analysis.reason || 'No reason'}`); + }); + + } catch (error) { + console.error('❌ Something went wrong:', error.message); + console.log('\n🤔 Troubleshooting tips:'); + console.log('- Check if the workflow class file exists'); + console.log('- Verify API keys are configured'); + console.log('- Make sure input names match what the workflow expects'); + console.log('- Check the Clara SDK is properly installed'); + } +} + +// 🎬 Run our advanced example! +runAdvancedSentimentAnalysis(); + +/** + * 💡 What's different about the JS Class format? + * + * 🔥 Advantages: + * ✅ More programming power and flexibility + * ✅ Type checking and IDE autocomplete + * ✅ Easier to integrate into existing codebases + * ✅ Better for complex applications + * ✅ Can add custom methods and properties + * ✅ Perfect for team development + * + * 🎯 Key Features We Used: + * - .execute() - Run the workflow once + * - .executeBatch() - Process multiple inputs efficiently + * - .getFlowInfo() - Get workflow metadata + * - Custom error handling and logging + * + * 🚀 When to use JS Class vs JSON? + * + * Use JSON when: + * - You're just starting out + * - Quick prototypes and testing + * - Non-technical team members + * - Simple automation scripts + * + * Use JS Class when: + * - Building production applications + * - Need advanced features (batch processing, callbacks) + * - Want better IDE support and debugging + * - Integrating with existing JavaScript/TypeScript projects + * + * 🎉 Both formats work with the same Clara Agent Studio workflows! + */ \ No newline at end of file diff --git a/sdk_examples/example-using-json.js b/sdk_examples/example-using-json.js new file mode 100644 index 00000000..99bc46ef --- /dev/null +++ b/sdk_examples/example-using-json.js @@ -0,0 +1,108 @@ +/** + * 🎯 Example 1: Using JSON Export Format + * + * This is the EASIEST way to use flows exported from Clara Agent Studio! + * + * Think of it like this: + * 1. You build a cool workflow in Clara Agent Studio 🎨 + * 2. You export it as JSON (like saving a recipe) 📄 + * 3. You load that JSON and run it with your data! 🚀 + * + * Perfect for: Beginners, quick prototypes, simple automation + */ + +import { ClaraFlowRunner } from 'clara-flow-sdk'; +import fs from 'fs'; + +// 📚 Step 1: Create the SDK runner (like getting ready to cook) +const runner = new ClaraFlowRunner({ + enableLogging: true, // See what's happening + logLevel: 'info' // Show important messages +}); + +async function runSentimentAnalysis() { + console.log('🤖 Starting Sentiment Analysis with JSON Format!'); + console.log('='.repeat(50)); + + try { + // 📖 Step 2: Load your workflow JSON (like reading a recipe) + console.log('📄 Loading workflow from JSON file...'); + const workflowJSON = JSON.parse( + fs.readFileSync('./NewWorkFLow_flow_sdk.json', 'utf8') + ); + + console.log(`✅ Loaded workflow: "${workflowJSON.flow.name}"`); + console.log(`📊 Contains ${workflowJSON.flow.nodes.length} nodes`); + + // 🎯 Step 3: Prepare your data (like gathering ingredients) + const inputData = { + // This goes to the first Input node (user feedback) + 'Input_1': 'Your product is amazing! I love it so much!', + + // This goes to the second Input node (JSON format example) + 'Input_2': JSON.stringify({ + "sentence": "sentence from the user", + "sentiment": "good, bad, very bad", + "reason": "reason for the choice" + }), + + // This goes to the third Input node (context) + 'Input_3': 'This is customer feedback about our product' + }; + + console.log('🎯 Input data prepared:'); + console.log('- User Feedback:', inputData.Input_1); + console.log('- JSON Format: Ready ✅'); + console.log('- Context: Ready ✅'); + + // 🚀 Step 4: Run the workflow! (like following the recipe) + console.log('\n⚡ Executing workflow...'); + const result = await runner.executeFlow(workflowJSON, inputData); + + // 🎉 Step 5: See the magic happen! + console.log('\n🎉 Results:'); + console.log('='.repeat(30)); + + // The workflow returns structured sentiment analysis + if (result) { + console.log('📊 Sentiment Analysis:', JSON.stringify(result, null, 2)); + + // Check if it detected "very bad" sentiment + if (result.sentiment === 'very bad') { + console.log('🚨 Alert: Very negative feedback detected!'); + } else { + console.log('✅ Feedback processed successfully'); + } + } + + } catch (error) { + console.error('❌ Oops! Something went wrong:', error.message); + console.log('\n🤔 Common issues:'); + console.log('- Make sure the JSON file exists'); + console.log('- Check if the API key is set correctly'); + console.log('- Verify the input data matches what the workflow expects'); + } +} + +// 🎬 Let's run our example! +runSentimentAnalysis(); + +/** + * 💡 What just happened? + * + * 1. We loaded a pre-built workflow from JSON + * 2. The workflow analyzes text sentiment using AI + * 3. It returns structured data (JSON) with the analysis + * 4. We can then use that data in our app! + * + * 🔥 Why use JSON format? + * ✅ Super easy - just export and run + * ✅ No coding required for the workflow logic + * ✅ Perfect for non-programmers + * ✅ Great for sharing workflows between teams + * + * 🎯 Next steps: + * - Try changing the input text + * - Export different workflows from Clara Studio + * - Combine multiple workflows together + */ \ No newline at end of file diff --git a/searxng/settings.yml b/searxng/settings.yml deleted file mode 100644 index 91db74d4..00000000 --- a/searxng/settings.yml +++ /dev/null @@ -1,108 +0,0 @@ -use_default_settings: true - -server: - limiter: false # Disable rate limiting for better performance - secret_key: "ultrasecretkeythatnobodyshouldknow" - image_proxy: true - -search: - formats: - - html - - json - - # Increase timeout for slow engines - request_timeout: 10.0 - - # Enable autocomplete - autocomplete: "duckduckgo" - - # Default results per page - default_results_per_page: 10 - - # Max results per page - max_results_per_page: 20 - -# Enable multiple search engines for redundancy -engines: - # General search engines (high priority) - - name: duckduckgo - engine: duckduckgo - shortcut: ddg - disabled: false - timeout: 3.0 - - - name: brave - engine: brave - shortcut: br - disabled: false - time_range_support: true - - - name: wikipedia - engine: wikipedia - shortcut: wp - disabled: false - - # News engines - - name: bing news - engine: bing_news - shortcut: bin - disabled: false - - - name: google news - engine: google_news - shortcut: gn - disabled: false - - # Code/GitHub search - - name: github - engine: github - shortcut: gh - disabled: false - - # Fallback engines - - name: qwant - engine: qwant - shortcut: qw - disabled: false - categories: general - - - name: yahoo - engine: yahoo - shortcut: yh - disabled: false - - - name: mojeek - engine: mojeek - shortcut: mjk - disabled: false - - # Image search engines - - name: google images - engine: google_images - shortcut: gimg - disabled: false - - - name: bing images - engine: bing_images - shortcut: bimg - disabled: false - - - name: duckduckgo images - engine: duckduckgo_images - shortcut: ddimg - disabled: false - -# Outgoing request configuration -outgoing: - request_timeout: 10.0 - max_request_timeout: 15.0 - pool_connections: 100 - pool_maxsize: 20 - enable_http2: true - - # Reduce fingerprinting - useragent_suffix: "" - - # Retry failed requests - retries: 2 - retry_on_http_error: [408, 429, 500, 502, 503, 504] diff --git a/src/App.tsx b/src/App.tsx new file mode 100644 index 00000000..f107b1fe --- /dev/null +++ b/src/App.tsx @@ -0,0 +1,302 @@ +import { useState, useEffect } from 'react'; +import Sidebar from './components/Sidebar'; +import Topbar from './components/Topbar'; +import Dashboard from './components/Dashboard'; +import Settings from './components/Settings'; +import Debug from './components/Debug'; +import Onboarding from './components/Onboarding'; +import ImageGen from './components/ImageGen'; +import Gallery from './components/Gallery'; +import Help from './components/Help'; +import N8N from './components/N8N'; +import Servers from './components/Servers'; +import AgentStudio from './components/AgentStudio'; +import AgentManager from './components/AgentManager'; +import AgentRunnerSDK from './components/AgentRunnerSDK'; +import Lumaui from './components/Lumaui'; +import LumaUILite from './components/LumaUILite'; +import Notebooks from './components/Notebooks'; +import Tasks from './components/Tasks'; +import Community from './components/Community'; +import { db } from './db'; +import { ProvidersProvider } from './contexts/ProvidersContext'; +import { ArtifactPaneProvider } from './contexts/ArtifactPaneContext'; +import ClaraAssistant from './components/ClaraAssistant'; +import { StartupService } from './services/startupService'; +import { initializeUIPreferences, applyUIPreferences } from './utils/uiPreferences'; + +function App() { + const [activePage, setActivePage] = useState(() => localStorage.getItem('activePage') || 'dashboard'); + const [showOnboarding, setShowOnboarding] = useState(false); + const [userInfo, setUserInfo] = useState<{ name: string } | null>(null); + const [alphaFeaturesEnabled, setAlphaFeaturesEnabled] = useState(false); + const [agentMode, setAgentMode] = useState<'manager' | 'studio' | 'runner'>('manager'); + const [editingAgentId, setEditingAgentId] = useState(null); + const [runningAgentId, setRunningAgentId] = useState(null); + + // Track Clara's processing state to keep it mounted when active + const [isClaraProcessing, setIsClaraProcessing] = useState(false); + + useEffect(() => { + const checkUserInfo = async () => { + const info = await db.getPersonalInfo(); + if (!info || !info.name) { + setShowOnboarding(true); + } else { + setShowOnboarding(false); + setUserInfo({ name: info.name }); + } + + // Initialize and apply UI preferences + initializeUIPreferences(); + applyUIPreferences(info); + }; + checkUserInfo(); + + // Add db to window for debugging in development + if (import.meta.env.DEV) { + (window as typeof window & { db: typeof db }).db = db; + } + }, []); + + useEffect(() => { + db.getAlphaFeaturesEnabled?.().then(val => setAlphaFeaturesEnabled(!!val)); + }, []); + + useEffect(() => { + // Apply startup settings + StartupService.getInstance().applyStartupSettings(); + }, []); + + // Trigger MCP servers restoration on app startup + useEffect(() => { + const restoreMCPServers = async () => { + if (window.mcpService && !showOnboarding) { + try { + console.log('App ready - attempting to restore MCP servers...'); + const results = await window.mcpService.startPreviouslyRunning(); + const successCount = results.filter((r: { success: boolean }) => r.success).length; + const totalCount = results.length; + + if (totalCount > 0) { + console.log(`MCP restoration: ${successCount}/${totalCount} servers restored`); + } else { + console.log('MCP restoration: No servers to restore'); + } + } catch (error) { + console.error('Error restoring MCP servers:', error); + } + } + }; + + // Delay restoration slightly to ensure app is fully initialized + const timeoutId = setTimeout(restoreMCPServers, 2000); + return () => clearTimeout(timeoutId); + }, [showOnboarding]); + + // Listen for global shortcut trigger to navigate to Clara chat + useEffect(() => { + let lastTriggerTime = 0; + const debounceDelay = 300; // 300ms debounce + + const handleGlobalClaraShortcut = () => { + const now = Date.now(); + + // Check if we're within the debounce period + if (now - lastTriggerTime < debounceDelay) { + console.log('Global shortcut navigation debounced - too soon after last trigger'); + return; + } + + lastTriggerTime = now; + console.log('Global shortcut triggered - navigating to Clara chat'); + setActivePage('clara'); + }; + + // Add listener for the trigger-new-chat event + if (window.electron && window.electron.receive) { + window.electron.receive('trigger-new-chat', handleGlobalClaraShortcut); + } + + // Cleanup listener on unmount + return () => { + if (window.electron && window.electron.removeListener) { + window.electron.removeListener('trigger-new-chat', handleGlobalClaraShortcut); + } + }; + }, []); + + const handleOnboardingComplete = async () => { + setShowOnboarding(false); + const info = await db.getPersonalInfo(); + if (info) { + setUserInfo({ name: info.name }); + } + }; + + useEffect(() => { + console.log('Storing activePage:', activePage); + localStorage.setItem('activePage', activePage); + + // Reset agent mode when navigating away from agents page + if (activePage !== 'agents') { + setAgentMode('manager'); + setEditingAgentId(null); + } + }, [activePage]); + + const renderContent = () => { + if (activePage === 'assistant') { + return ; + } + + // Clara is now always mounted but conditionally visible + // This allows it to run in the background + + if (activePage === 'agents') { + const handleEditAgent = (agentId: string) => { + setEditingAgentId(agentId); + setAgentMode('studio'); + }; + + const handleOpenAgent = (agentId: string) => { + setRunningAgentId(agentId); + setAgentMode('runner'); + }; + + const handleCreateAgent = () => { + setEditingAgentId(null); + setAgentMode('studio'); + }; + + const handleBackToManager = () => { + setAgentMode('manager'); + setEditingAgentId(null); + setRunningAgentId(null); + }; + + if (agentMode === 'manager') { + return ( + + ); + } else if (agentMode === 'studio') { + return ( + + ); + } else if (agentMode === 'runner' && runningAgentId) { + return ( + + ); + } + } + + + + if (activePage === 'image-gen') { + return ; + } + + if (activePage === 'gallery') { + return ; + } + + if (activePage === 'n8n') { + return ; + } + + if (activePage === 'servers') { + return ; + } + + return ( +
+ + +
+ + +
+ {(() => { + switch (activePage) { + case 'tasks': + return ; + case 'community': + return ; + case 'settings': + return ; + case 'debug': + return ; + case 'help': + return ; + case 'notebooks': + return ; + case 'lumaui': + return ; + case 'lumaui-lite': + return ; + case 'dashboard': + default: + return ; + } + })()} +
+
+
+ ); + }; + + return ( + + +
+ {showOnboarding ? ( + + ) : ( + <> + {/* Smart rendering: Keep Clara mounted when processing, unmount when idle */} + {(activePage === 'clara' || isClaraProcessing) && ( +
+ +
+ )} + + {/* Render other content when not on Clara page */} + {activePage !== 'clara' && renderContent()} + + {/* Background processing indicator */} + {isClaraProcessing && activePage !== 'clara' && ( +
+
+ Clara is processing in background... +
+ )} + + )} +
+
+
+ ); +} + +export default App; + + diff --git a/src/assets/logo.png b/src/assets/logo.png new file mode 100644 index 00000000..19e2a328 Binary files /dev/null and b/src/assets/logo.png differ diff --git a/src/assets/mascot/Error_Clara.png b/src/assets/mascot/Error_Clara.png new file mode 100644 index 00000000..7fb1a2ec Binary files /dev/null and b/src/assets/mascot/Error_Clara.png differ diff --git a/docs/images/clara-mascot.png b/src/assets/mascot/Hi_Welcome_Clara.png similarity index 100% rename from docs/images/clara-mascot.png rename to src/assets/mascot/Hi_Welcome_Clara.png diff --git a/src/assets/mascot/Success_Clara.png b/src/assets/mascot/Success_Clara.png new file mode 100644 index 00000000..773c1f25 Binary files /dev/null and b/src/assets/mascot/Success_Clara.png differ diff --git a/src/assets/mascot/Tips_Clara.png b/src/assets/mascot/Tips_Clara.png new file mode 100644 index 00000000..d640e1e0 Binary files /dev/null and b/src/assets/mascot/Tips_Clara.png differ diff --git a/src/assets/mcp.svg b/src/assets/mcp.svg new file mode 100644 index 00000000..5cd83a8b --- /dev/null +++ b/src/assets/mcp.svg @@ -0,0 +1 @@ +ModelContextProtocol \ No newline at end of file diff --git a/src/assets/temo.png b/src/assets/temo.png new file mode 100644 index 00000000..2dd48596 Binary files /dev/null and b/src/assets/temo.png differ diff --git a/src/components/AgentBuilder/AgentBuilderToolbar.tsx b/src/components/AgentBuilder/AgentBuilderToolbar.tsx new file mode 100644 index 00000000..064f1a49 --- /dev/null +++ b/src/components/AgentBuilder/AgentBuilderToolbar.tsx @@ -0,0 +1,62 @@ +import React, { useState } from 'react'; +import { Button } from '@/components/ui/button'; +import { Plus, Download, Package, Upload } from 'lucide-react'; + +const AgentBuilderToolbar: React.FC = () => { + const [showWorkflowTemplates, setShowWorkflowTemplates] = useState(false); + const [currentFlow, setCurrentFlow] = useState(null); + const [showImportDialog, setShowImportDialog] = useState(false); + + const exportFlow = (format: string) => { + // Implementation of exportFlow function + }; + + return ( +
+ + + + + + + +
+ ); +}; + +export default AgentBuilderToolbar; \ No newline at end of file diff --git a/src/components/AgentBuilder/Canvas/Canvas.tsx b/src/components/AgentBuilder/Canvas/Canvas.tsx new file mode 100644 index 00000000..ee212f91 --- /dev/null +++ b/src/components/AgentBuilder/Canvas/Canvas.tsx @@ -0,0 +1,641 @@ +import React, { useCallback, useMemo, useEffect, useState } from 'react'; +import ReactFlow, { + Node, + Edge, + addEdge, + Connection, + useNodesState, + useEdgesState, + Background, + Controls, + MiniMap, + ReactFlowProvider, + OnConnect, + NodeTypes, + ReactFlowInstance, + OnNodesChange, + OnEdgesChange, + NodeChange, + EdgeChange, +} from 'reactflow'; +import 'reactflow/dist/style.css'; + +import { useAgentBuilder } from '../../../contexts/AgentBuilder/AgentBuilderContext'; +import CustomNode from '../Nodes/CustomNode'; + +// Import all built-in node components +import InputNode from '../Nodes/InputNode'; +import OutputNode from '../Nodes/OutputNode'; +import JsonParseNode from '../Nodes/JsonParseNode'; +import IfElseNode from '../Nodes/IfElseNode'; +import LLMNode from '../Nodes/LLMNode'; +import StructuredLLMNode from '../Nodes/StructuredLLMNode'; +import ImageInputNode from '../Nodes/ImageInputNode'; +import TextNode from '../Nodes/TextNode'; +import MathNode from '../Nodes/MathNode'; +import PDFInputNode from '../Nodes/PDFInputNode'; + +import FileUploadNode from '../Nodes/FileUploadNode'; +import WhisperTranscriptionNode from '../Nodes/WhisperTranscriptionNode'; +import CombineTextNode from '../Nodes/CombineTextNode'; +import JsonStringifyNode from '../Nodes/JsonStringifyNode'; +import APIRequestNode from '../Nodes/APIRequestNode'; +import TextToSpeechNode from '../Nodes/TextToSpeechNode'; +import SpeechToTextNode from '../Nodes/SpeechToTextNode'; +import StaticTextNode from '../Nodes/StaticTextNode'; +import AgentExecutorNode from '../Nodes/AgentExecutorNode'; +import { ComfyUIImageGenNode } from '../Nodes/ComfyUIImageGenNode'; +import NotebookWriterNode from '../Nodes/NotebookWriterNode'; +import NotebookChatNode from '../Nodes/NotebookChatNode'; + +// Debug: Log successful imports +console.log('Node imports loaded:', { + InputNode: !!InputNode, + OutputNode: !!OutputNode, + JsonParseNode: !!JsonParseNode, + IfElseNode: !!IfElseNode, + LLMNode: !!LLMNode, + StructuredLLMNode: !!StructuredLLMNode, + ImageInputNode: !!ImageInputNode, + TextNode: !!TextNode, + MathNode: !!MathNode, + PDFInputNode: !!PDFInputNode, + + FileUploadNode: !!FileUploadNode, + WhisperTranscriptionNode: !!WhisperTranscriptionNode, + CombineTextNode: !!CombineTextNode, + JsonStringifyNode: !!JsonStringifyNode, + APIRequestNode: !!APIRequestNode, + StaticTextNode: !!StaticTextNode, + AgentExecutorNode: !!AgentExecutorNode, + NotebookWriterNode: !!NotebookWriterNode, + NotebookChatNode: !!NotebookChatNode, + SpeechToTextNode: !!SpeechToTextNode, +}); + +// Define base node types with proper imports - moved outside component to ensure immediate availability +const baseNodeTypes: NodeTypes = { + 'input': InputNode, + 'output': OutputNode, + 'json-parse': JsonParseNode, + 'if-else': IfElseNode, + 'llm': LLMNode, + 'structured-llm': StructuredLLMNode, + 'image-input': ImageInputNode, + 'pdf-input': PDFInputNode, + + 'file-upload': FileUploadNode, + 'whisper-transcription': WhisperTranscriptionNode, + 'combine-text': CombineTextNode, + 'json-stringify': JsonStringifyNode, + 'api-request': APIRequestNode, + 'static-text': StaticTextNode, + 'text': TextNode, + 'math': MathNode, + 'agent-executor': AgentExecutorNode, + 'comfyui-image-gen': ComfyUIImageGenNode, + 'text-to-speech': TextToSpeechNode, + 'speech-to-text': SpeechToTextNode, + 'notebook-writer': NotebookWriterNode, + 'notebook-chat': NotebookChatNode, +}; + +// Debug: Log base node types immediately after definition +console.log('Base node types defined:', baseNodeTypes); + +// Wrapper component for custom nodes that provides the node definition +const CustomNodeWrapper: React.FC = React.memo((props) => { + const { customNodes } = useAgentBuilder(); + const nodeDefinition = customNodes.find(def => def.type === props.type); + + if (!nodeDefinition) { + console.error(`Custom node definition not found for type: ${props.type}`, { + availableTypes: customNodes.map(n => n.type), + requestedType: props.type + }); + return ( +
+
Custom node definition not found
+
Type: {props.type}
+
+ ); + } + + return ; +}); + +interface CanvasProps { + className?: string; +} + +const CanvasContent: React.FC = ({ className = '' }) => { + const { + nodes, + connections, + canvas, + customNodes, + executionResults, + addNode, + updateNode, + deleteNode, + addConnection, + deleteConnection, + updateCanvas, + selectNodes, + clearSelection, + addExecutionLog, + saveFlow, + } = useAgentBuilder(); + + // Clipboard state for copy-paste functionality + const [clipboard, setClipboard] = useState<{ + nodeData: any; + timestamp: number; + } | null>(null); + + // Create dynamic node types that include custom nodes - use useMemo for proper memoization + const nodeTypes = useMemo(() => { + const dynamicNodeTypes = { ...baseNodeTypes }; + + console.log('🔧 Creating nodeTypes - Base node types:', Object.keys(baseNodeTypes)); + console.log('🔧 Available custom nodes:', customNodes.map(n => ({ type: n.type, name: n.name }))); + + // Add all custom nodes to the nodeTypes + customNodes.forEach(customNodeDef => { + dynamicNodeTypes[customNodeDef.type] = CustomNodeWrapper; + console.log(`✅ Registered custom node type: ${customNodeDef.type} -> CustomNodeWrapper`); + }); + + const finalTypes = Object.keys(dynamicNodeTypes); + console.log(`🎯 Final nodeTypes (${finalTypes.length}):`, finalTypes); + console.log('🎯 Final nodeTypes object:', dynamicNodeTypes); + return dynamicNodeTypes; + }, [customNodes]); + + // Convert our internal format to ReactFlow format + const initialNodes: Node[] = useMemo(() => { + return nodes.map(node => { + // Get execution result for this node + const executionResult = executionResults[node.id]; + + // For output nodes, pass the execution result as inputValue + const nodeData: any = { + ...node.data, + label: node.name, + inputs: node.inputs, + outputs: node.outputs, + onUpdate: (updates: any) => updateNode(node.id, updates), + onDelete: () => deleteNode(node.id), + }; + + // Add execution results to node data based on node type + if (node.type === 'output' && executionResult !== undefined) { + nodeData.inputValue = executionResult; + } + + return { + id: node.id, + type: node.type, + position: node.position, + data: nodeData, + selected: canvas.selection.nodeIds.includes(node.id), + draggable: true, + }; + }); + }, [nodes, canvas.selection.nodeIds, executionResults, updateNode, deleteNode]); + + // Convert our internal connections to ReactFlow edges + const initialEdges: Edge[] = useMemo(() => { + return connections.map(connection => ({ + id: connection.id, + source: connection.sourceNodeId, + target: connection.targetNodeId, + sourceHandle: connection.sourcePortId, + targetHandle: connection.targetPortId, + type: 'default', + animated: false, + })); + }, []); + + // Use ReactFlow's internal state management + const [reactFlowNodes, setReactFlowNodes, onNodesChange] = useNodesState(initialNodes); + const [reactFlowEdges, setReactFlowEdges, onEdgesChange] = useEdgesState(initialEdges); + + // Sync our context state to ReactFlow state when it changes + useEffect(() => { + const newNodes: Node[] = nodes.map(node => { + // Get execution result for this node + const executionResult = executionResults[node.id]; + + const nodeData: any = { + ...node.data, + label: node.name, + inputs: node.inputs, + outputs: node.outputs, + onUpdate: (updates: any) => updateNode(node.id, updates), + onDelete: () => deleteNode(node.id), + }; + + // Add execution results to node data based on node type + if (node.type === 'output' && executionResult !== undefined) { + nodeData.inputValue = executionResult; + } + + return { + id: node.id, + type: node.type, + position: node.position, + data: nodeData, + selected: canvas.selection.nodeIds.includes(node.id), + draggable: true, + }; + }); + + // Only update if there are actual changes to prevent infinite loops + const hasChanges = + newNodes.length !== reactFlowNodes.length || + newNodes.some((node, index) => { + const existing = reactFlowNodes[index]; + return !existing || + node.id !== existing.id || + node.position.x !== existing.position.x || + node.position.y !== existing.position.y || + node.selected !== existing.selected || + JSON.stringify(node.data) !== JSON.stringify(existing.data); + }); + + if (hasChanges) { + setReactFlowNodes(newNodes); + } + }, [nodes, canvas.selection.nodeIds, executionResults, updateNode, deleteNode, reactFlowNodes, setReactFlowNodes]); + + // Sync connections to ReactFlow edges + useEffect(() => { + const newEdges: Edge[] = connections.map(connection => ({ + id: connection.id, + source: connection.sourceNodeId, + target: connection.targetNodeId, + sourceHandle: connection.sourcePortId, + targetHandle: connection.targetPortId, + type: 'default', + animated: false, + })); + + // Only update if there are actual changes + const hasChanges = + newEdges.length !== reactFlowEdges.length || + newEdges.some((edge, index) => { + const existing = reactFlowEdges[index]; + return !existing || edge.id !== existing.id; + }); + + if (hasChanges) { + setReactFlowEdges(newEdges); + } + }, [connections, reactFlowEdges, setReactFlowEdges]); + + // Handle ReactFlow node changes and sync back to context + const handleNodesChange: OnNodesChange = useCallback((changes: NodeChange[]) => { + onNodesChange(changes); + + changes.forEach((change) => { + if (change.type === 'position' && 'position' in change && change.position) { + // Update node position in our context + updateNode(change.id, { + position: change.position + }); + } else if (change.type === 'select' && 'selected' in change) { + // Handle selection changes + const allSelectedNodes = changes + .filter((c): c is typeof change => c.type === 'select' && 'selected' in c && c.selected) + .map(c => c.id); + + selectNodes(allSelectedNodes); + } else if (change.type === 'remove') { + // Handle node deletion (using close button on each node) + deleteNode(change.id); + } + }); + }, [onNodesChange, updateNode, selectNodes, deleteNode]); + + // Handle ReactFlow edge changes and sync back to context + const handleEdgesChange: OnEdgesChange = useCallback((changes: EdgeChange[]) => { + onEdgesChange(changes); + + changes.forEach((change) => { + if (change.type === 'remove') { + deleteConnection(change.id); + } + }); + }, [onEdgesChange, deleteConnection]); + + // Handle new connections + const onConnect: OnConnect = useCallback((connection: Connection) => { + if (connection.source && connection.target && connection.sourceHandle && connection.targetHandle) { + addConnection( + connection.source, + connection.sourceHandle, + connection.target, + connection.targetHandle + ); + } + }, [addConnection]); + + // Handle canvas click to clear selection + const onPaneClick = useCallback(() => { + clearSelection(); + }, [clearSelection]); + + // Handle viewport changes + const onMove = useCallback((_: any, viewport: { x: number; y: number; zoom: number }) => { + updateCanvas({ + viewport: { + x: viewport.x, + y: viewport.y, + zoom: viewport.zoom, + } + }); + }, [updateCanvas]); + + // Handle drag over for node dropping + const onDragOver = useCallback((event: React.DragEvent) => { + event.preventDefault(); + event.dataTransfer.dropEffect = 'move'; + }, []); + + // Handle node drop from palette + const onDrop = useCallback((event: React.DragEvent) => { + event.preventDefault(); + + const nodeType = event.dataTransfer.getData('application/reactflow'); + if (!nodeType) return; + + const reactFlowBounds = (event.target as Element).closest('.react-flow')?.getBoundingClientRect(); + if (!reactFlowBounds) return; + + const position = { + x: event.clientX - reactFlowBounds.left - canvas.viewport.x, + y: event.clientY - reactFlowBounds.top - canvas.viewport.y, + }; + + // Adjust for zoom + position.x = position.x / canvas.viewport.zoom; + position.y = position.y / canvas.viewport.zoom; + + addNode(nodeType, position); + }, [addNode, canvas.viewport]); + + // Debug: Log nodeTypes being passed to ReactFlow whenever nodeTypes changes + useEffect(() => { + const nodeTypeKeys = Object.keys(nodeTypes); + console.log(`🚀 ReactFlow nodeTypes being passed (${nodeTypeKeys.length}):`, nodeTypeKeys); + console.log('🚀 Custom nodes in nodeTypes:', nodeTypeKeys.filter(key => !Object.keys(baseNodeTypes).includes(key))); + }, [nodeTypes]); + + // Copy selected node to clipboard + const handleCopyNode = useCallback(() => { + if (canvas.selection.nodeIds.length !== 1) { + addExecutionLog({ + level: 'warning', + message: 'Please select exactly one node to copy' + }); + return; + } + + const selectedNodeId = canvas.selection.nodeIds[0]; + const nodeToCopy = nodes.find(node => node.id === selectedNodeId); + + if (!nodeToCopy) { + addExecutionLog({ + level: 'error', + message: 'Selected node not found' + }); + return; + } + + // Store node data in clipboard (excluding ID and position) + const { id, position, ...nodeDataToCopy } = nodeToCopy; + setClipboard({ + nodeData: { + ...nodeDataToCopy, + // Store original position as reference for relative pasting + originalPosition: position + }, + timestamp: Date.now() + }); + + addExecutionLog({ + level: 'success', + message: `Node "${nodeToCopy.name}" copied to clipboard`, + data: { nodeType: nodeToCopy.type } + }); + }, [canvas.selection.nodeIds, nodes, addExecutionLog]); + + // Paste node from clipboard + const handlePasteNode = useCallback(() => { + if (!clipboard || !clipboard.nodeData) return; + + // Check if clipboard data is not too old (5 minutes) + const clipboardAge = Date.now() - clipboard.timestamp; + if (clipboardAge > 5 * 60 * 1000) { + setClipboard(null); + addExecutionLog({ + level: 'warning', + message: 'Clipboard data expired' + }); + return; + } + + try { + const { nodeData } = clipboard; + const { originalPosition, ...restNodeData } = nodeData; + + // Calculate new position - offset from original position + const offsetX = 50; + const offsetY = 50; + const newPosition = { + x: originalPosition.x + offsetX, + y: originalPosition.y + offsetY + }; + + // Generate new unique ID + const newId = `${Date.now()}-${Math.random().toString(36).substr(2, 9)}`; + + // Create new node with copied data + const newNode = { + ...restNodeData, + id: newId, + name: `${restNodeData.name} (Copy)`, + position: newPosition, + // Ensure inputs and outputs have clean IDs (no references to original node) + inputs: restNodeData.inputs.map((input: any) => ({ ...input })), + outputs: restNodeData.outputs.map((output: any) => ({ ...output })) + }; + + // Add the new node using the context function (this will handle validation and state updates) + // We'll reconstruct it by using addNode with the type and then updating it + const addedNode = addNode(restNodeData.type, newPosition); + + // Update the added node with the copied data (excluding the auto-generated parts) + updateNode(addedNode.id, { + name: newNode.name, + data: restNodeData.data || {} + }); + + // Select the newly pasted node + selectNodes([addedNode.id]); + + addExecutionLog({ + level: 'success', + message: `Node "${newNode.name}" pasted successfully`, + data: { nodeType: restNodeData.type, newId: addedNode.id } + }); + + // Update clipboard position for next paste (cascade effect) + setClipboard(prev => prev ? { + ...prev, + nodeData: { + ...prev.nodeData, + originalPosition: newPosition + } + } : null); + + } catch (error) { + addExecutionLog({ + level: 'error', + message: `Failed to paste node: ${error instanceof Error ? error.message : 'Unknown error'}` + }); + } + }, [clipboard, addNode, updateNode, selectNodes, addExecutionLog]); + + // Handle save shortcut + const handleSave = useCallback(async () => { + try { + await saveFlow(); + addExecutionLog({ + level: 'success', + message: 'Workflow saved successfully' + }); + } catch (error) { + addExecutionLog({ + level: 'error', + message: `Failed to save workflow: ${error instanceof Error ? error.message : 'Unknown error'}` + }); + } + }, [saveFlow, addExecutionLog]); + + // Handle keyboard events for copy-paste and save (removed delete functionality) + useEffect(() => { + const handleKeyDown = (event: KeyboardEvent) => { + // Don't handle shortcuts when user is typing in input fields or text areas + if (event.target instanceof HTMLInputElement || + event.target instanceof HTMLTextAreaElement || + (event.target instanceof HTMLElement && event.target.isContentEditable)) { + return; + } + + // Handle save (Ctrl+S or Cmd+S) + if ((event.ctrlKey || event.metaKey) && event.key === 's') { + event.preventDefault(); + handleSave(); + return; + } + + // Handle copy (Ctrl+C or Cmd+C) + if ((event.ctrlKey || event.metaKey) && event.key === 'c' && canvas.selection.nodeIds.length === 1) { + event.preventDefault(); + handleCopyNode(); + } + + // Handle paste (Ctrl+V or Cmd+V) + else if ((event.ctrlKey || event.metaKey) && event.key === 'v' && clipboard) { + event.preventDefault(); + handlePasteNode(); + } + }; + + // Add event listener to document to catch keyboard events + document.addEventListener('keydown', handleKeyDown); + + // Cleanup + return () => { + document.removeEventListener('keydown', handleKeyDown); + }; + }, [canvas.selection.nodeIds, clipboard, handleCopyNode, handlePasteNode, handleSave]); + + // Category-based colors + const getCategoryColor = (node: Node): string => { + const nodeType = node.type; + switch (nodeType) { + case 'input': + case 'output': + return '#10b981'; + case 'json-parse': return '#3b82f6'; + case 'json-stringify': return '#3b82f6'; + case 'api-request': return '#10b981'; + case 'if-else': return '#84cc16'; + case 'llm': return '#ec4899'; + case 'structured-llm': return '#8b5cf6'; + case 'image-input': return '#f59e0b'; + case 'pdf-input': return '#3b82f6'; + case 'text': return '#84cc16'; + case 'math': return '#ec4899'; + default: return '#6b7280'; + } + }; + + return ( +
e.currentTarget.focus()}> + + + + + +
+ ); +}; + +const Canvas: React.FC = (props) => { + return ( + + + + ); +}; + +export default Canvas; \ No newline at end of file diff --git a/src/components/AgentBuilder/ExportModal.tsx b/src/components/AgentBuilder/ExportModal.tsx new file mode 100644 index 00000000..6f7128ad --- /dev/null +++ b/src/components/AgentBuilder/ExportModal.tsx @@ -0,0 +1,371 @@ +import React, { useState } from 'react'; +import { X, Download, Zap, FileText, Code, Sparkles, Info, ExternalLink, Copy, Check } from 'lucide-react'; + +interface ExportModalProps { + isOpen: boolean; + onClose: () => void; + onExport: (format: string) => Promise; + currentFlow: any; + hasCustomNodes: boolean; +} + +type ExportFormat = 'clara-native' | 'clara-sdk' | 'sdk-code'; + +interface ExportOption { + id: ExportFormat; + name: string; + description: string; + icon: React.ReactNode; + badge?: string; + features: string[]; + useCase: string; + fileExtension: string; + color: string; + gradient: string; + recommended?: boolean; +} + +const ExportModal: React.FC = ({ + isOpen, + onClose, + onExport, + currentFlow, + hasCustomNodes +}) => { + const [selectedFormat, setSelectedFormat] = useState('clara-sdk'); + const [isExporting, setIsExporting] = useState(false); + const [showPreview, setShowPreview] = useState(false); + const [exportSuccess, setExportSuccess] = useState(false); + + const exportOptions: ExportOption[] = [ + { + id: 'clara-native', + name: 'Standard JSON', + description: 'Classic Clara Agent Studio format for sharing and backup', + icon: , + features: [ + 'Compatible with all Clara versions', + 'Lightweight file size', + 'Easy to share and backup', + 'Standard JSON format' + ], + useCase: 'Best for sharing flows between Clara installations', + fileExtension: '.json', + color: 'blue', + gradient: 'from-blue-500 to-blue-600' + }, + { + id: 'clara-sdk', + name: 'SDK Enhanced', + description: 'JSON format with embedded custom nodes for SDK execution', + icon: , + badge: 'ENHANCED', + features: [ + 'Includes custom node definitions', + 'Self-contained execution', + 'SDK compatible', + 'Preserves all functionality' + ], + useCase: 'Perfect for running flows with the Clara SDK', + fileExtension: '.json', + color: 'green', + gradient: 'from-green-500 to-green-600', + recommended: hasCustomNodes + }, + { + id: 'sdk-code', + name: 'JavaScript Code', + description: 'Ready-to-use JavaScript module for direct integration', + icon: , + badge: 'NEW', + features: [ + 'Complete JavaScript class', + 'Zero configuration needed', + 'TypeScript friendly', + 'Production ready' + ], + useCase: 'Ideal for embedding in web applications and servers', + fileExtension: '.js', + color: 'purple', + gradient: 'from-purple-500 to-indigo-500' + } + ]; + + const selectedOption = exportOptions.find(opt => opt.id === selectedFormat)!; + + const handleExport = async () => { + if (!currentFlow) return; + + setIsExporting(true); + try { + await onExport(selectedFormat); + setExportSuccess(true); + setTimeout(() => { + setExportSuccess(false); + onClose(); + }, 2000); + } catch (error) { + console.error('Export failed:', error); + } finally { + setIsExporting(false); + } + }; + + const handleCopyInstallCommand = () => { + if (selectedFormat === 'sdk-code') { + navigator.clipboard.writeText('npm install clara-flow-sdk'); + } + }; + + const generatePreviewFilename = () => { + if (!currentFlow) return `flow${selectedOption.fileExtension}`; + const safeName = currentFlow.name.replace(/[^a-zA-Z0-9]/g, '_').toLowerCase(); + return `${safeName}${selectedOption.fileExtension}`; + }; + + if (!isOpen) return null; + + return ( +
+
+ {/* Header */} +
+
+
+ +
+
+

+ Export Flow +

+

+ Choose your export format and options +

+
+
+ +
+ +
+ {/* Export Options */} +
+

+ Select Export Format +

+ +
+ {exportOptions.map((option) => ( +
setSelectedFormat(option.id)} + > + {/* Recommended badge */} + {option.recommended && ( +
+ + RECOMMENDED +
+ )} + +
+
+ {option.icon} +
+
+
+

+ {option.name} +

+ {option.badge && ( + + {option.badge} + + )} +
+

+ {option.description} +

+
+ {option.features.map((feature, idx) => ( +
+
+ {feature} +
+ ))} +
+
+
+
+ ))} +
+ + {/* Custom Nodes Warning */} + {hasCustomNodes && selectedFormat === 'clara-native' && ( +
+
+ +
+

+ Custom Nodes Detected +

+

+ Standard JSON format won't include custom node execution code. + Consider using "SDK Enhanced" format instead. +

+
+
+
+ )} +
+ + {/* Preview & Details */} +
+

+ Export Details +

+ + {/* Selected Format Info */} +
+
+ {selectedOption.icon} +
+

{selectedOption.name}

+

{selectedOption.useCase}

+
+
+ +
+
Output File:
+
+ {generatePreviewFilename()} +
+
+
+ + {/* Flow Information */} + {currentFlow && ( +
+

+ Flow Information +

+
+
+ Name: + {currentFlow.name} +
+ {currentFlow.description && ( +
+ Description: + + {currentFlow.description} + +
+ )} +
+ Nodes: + + {/* We'd need to pass node count as prop */} + {currentFlow.nodeCount || 'Multiple'} + +
+
+ Custom Nodes: + + {hasCustomNodes ? 'Yes' : 'None'} + +
+
+
+ )} + + {/* Format-specific instructions */} + {selectedFormat === 'sdk-code' && ( +
+

+ + Setup Instructions +

+
+
+

+ 1. Install the Clara Flow SDK: +

+
+ npm install clara-flow-sdk + +
+
+
+

+ 2. Import and use the generated flow in your application +

+
+
+ +
+
+
+ )} + + {/* Export Actions */} +
+ + + +
+
+
+
+
+ ); +}; + +export default ExportModal; \ No newline at end of file diff --git a/src/components/AgentBuilder/NodeCreator/ComfyUIImageNodes.ts b/src/components/AgentBuilder/NodeCreator/ComfyUIImageNodes.ts new file mode 100644 index 00000000..e69de29b diff --git a/src/components/AgentBuilder/NodeCreator/CustomNodeLibrary.tsx b/src/components/AgentBuilder/NodeCreator/CustomNodeLibrary.tsx new file mode 100644 index 00000000..547f1fef --- /dev/null +++ b/src/components/AgentBuilder/NodeCreator/CustomNodeLibrary.tsx @@ -0,0 +1,437 @@ +import React, { useState } from 'react'; +import { Download, Upload, Share2, Star, Clock, User, Tag, Search, Filter, X } from 'lucide-react'; +import { CustomNodeDefinition } from '../../../types/agent/types'; +import { customNodeManager } from './CustomNodeManager'; + +interface CustomNodeLibraryProps { + isOpen: boolean; + onClose: () => void; + onNodeImported: () => void; +} + +const SAMPLE_COMMUNITY_NODES: CustomNodeDefinition[] = [ + { + id: 'community-text-formatter', + name: 'Text Formatter', + type: 'text-formatter', + category: 'text', + description: 'Format text with various transformations like uppercase, lowercase, title case, etc.', + icon: '📝', + version: '1.0.0', + author: 'Community', + inputs: [ + { id: 'text', name: 'Text', type: 'input', dataType: 'string', required: true, description: 'Text to format' } + ], + outputs: [ + { id: 'formatted', name: 'Formatted Text', type: 'output', dataType: 'string', description: 'Formatted text result' } + ], + properties: [ + { + id: 'format', + name: 'Format Type', + type: 'select', + required: true, + defaultValue: 'uppercase', + description: 'Type of formatting to apply', + options: [ + { label: 'Uppercase', value: 'uppercase' }, + { label: 'Lowercase', value: 'lowercase' }, + { label: 'Title Case', value: 'titlecase' }, + { label: 'Capitalize', value: 'capitalize' } + ] + } + ], + executionHandler: 'text-formatter-handler', + executionCode: `async function execute(inputs, properties, context) { + const text = inputs.text || ''; + const format = properties.format || 'uppercase'; + + let result; + switch (format) { + case 'uppercase': + result = text.toUpperCase(); + break; + case 'lowercase': + result = text.toLowerCase(); + break; + case 'titlecase': + result = text.split(' ').map(word => + word.charAt(0).toUpperCase() + word.slice(1).toLowerCase() + ).join(' '); + break; + case 'capitalize': + result = text.charAt(0).toUpperCase() + text.slice(1); + break; + default: + result = text; + } + + context.log('Formatted text:', result); + return { formatted: result }; +}`, + customMetadata: { + isUserCreated: true, + createdBy: 'Community User', + createdAt: '2024-01-15T10:00:00Z', + published: true, + downloadCount: 156, + rating: 4.5 + }, + metadata: { + tags: ['text', 'formatting', 'transform', 'utility'], + documentation: 'A utility node for formatting text in various ways.', + examples: ['Convert "hello world" to "HELLO WORLD"'] + } + }, + { + id: 'community-math-calculator', + name: 'Math Calculator', + type: 'math-calculator', + category: 'math', + description: 'Perform basic mathematical operations on two numbers', + icon: '🧮', + version: '1.2.0', + author: 'Community', + inputs: [ + { id: 'a', name: 'Number A', type: 'input', dataType: 'number', required: true, description: 'First number' }, + { id: 'b', name: 'Number B', type: 'input', dataType: 'number', required: true, description: 'Second number' } + ], + outputs: [ + { id: 'result', name: 'Result', type: 'output', dataType: 'number', description: 'Calculation result' } + ], + properties: [ + { + id: 'operation', + name: 'Operation', + type: 'select', + required: true, + defaultValue: 'add', + description: 'Mathematical operation to perform', + options: [ + { label: 'Add (+)', value: 'add' }, + { label: 'Subtract (-)', value: 'subtract' }, + { label: 'Multiply (×)', value: 'multiply' }, + { label: 'Divide (÷)', value: 'divide' }, + { label: 'Power (^)', value: 'power' } + ] + } + ], + executionHandler: 'math-calculator-handler', + executionCode: `async function execute(inputs, properties, context) { + const a = Number(inputs.a) || 0; + const b = Number(inputs.b) || 0; + const operation = properties.operation || 'add'; + + let result; + switch (operation) { + case 'add': + result = a + b; + break; + case 'subtract': + result = a - b; + break; + case 'multiply': + result = a * b; + break; + case 'divide': + if (b === 0) { + throw new Error('Division by zero is not allowed'); + } + result = a / b; + break; + case 'power': + result = Math.pow(a, b); + break; + default: + throw new Error('Unknown operation: ' + operation); + } + + context.log(\`\${a} \${operation} \${b} = \${result}\`); + return { result }; +}`, + customMetadata: { + isUserCreated: true, + createdBy: 'MathWiz', + createdAt: '2024-01-10T14:30:00Z', + published: true, + downloadCount: 89, + rating: 4.8 + }, + metadata: { + tags: ['math', 'calculator', 'arithmetic', 'numbers'], + documentation: 'Performs basic mathematical operations between two numbers.', + examples: ['Calculate 5 + 3 = 8', 'Calculate 10 / 2 = 5'] + } + } +]; + +const CustomNodeLibrary: React.FC = ({ isOpen, onClose, onNodeImported }) => { + const [searchTerm, setSearchTerm] = useState(''); + const [selectedCategory, setSelectedCategory] = useState('all'); + const [sortBy, setSortBy] = useState('downloads'); + const [importError, setImportError] = useState(null); + const [importSuccess, setImportSuccess] = useState(null); + + const categories = ['all', 'text', 'math', 'data', 'logic', 'ai', 'custom']; + + const filteredNodes = SAMPLE_COMMUNITY_NODES.filter(node => { + const matchesSearch = node.name.toLowerCase().includes(searchTerm.toLowerCase()) || + node.description.toLowerCase().includes(searchTerm.toLowerCase()) || + (node.metadata?.tags || []).some(tag => tag.toLowerCase().includes(searchTerm.toLowerCase())); + + const matchesCategory = selectedCategory === 'all' || node.category === selectedCategory; + + return matchesSearch && matchesCategory; + }).sort((a, b) => { + switch (sortBy) { + case 'downloads': + return (b.customMetadata.downloadCount || 0) - (a.customMetadata.downloadCount || 0); + case 'rating': + return (b.customMetadata.rating || 0) - (a.customMetadata.rating || 0); + case 'recent': + return new Date(b.customMetadata.createdAt).getTime() - new Date(a.customMetadata.createdAt).getTime(); + case 'name': + return a.name.localeCompare(b.name); + default: + return 0; + } + }); + + const handleImportNode = (node: CustomNodeDefinition) => { + try { + customNodeManager.registerCustomNode(node); + setImportSuccess(`Successfully imported "${node.name}"`); + setImportError(null); + onNodeImported(); + + // Clear success message after 3 seconds + setTimeout(() => setImportSuccess(null), 3000); + } catch (error) { + setImportError(error instanceof Error ? error.message : 'Failed to import node'); + setImportSuccess(null); + } + }; + + const handleImportFromFile = () => { + const input = document.createElement('input'); + input.type = 'file'; + input.accept = '.json'; + input.onchange = async (e) => { + const file = (e.target as HTMLInputElement).files?.[0]; + if (!file) return; + + try { + const text = await file.text(); + const result = customNodeManager.importCustomNodes(text); + + if (result.imported > 0) { + setImportSuccess(`Successfully imported ${result.imported} node(s)`); + setImportError(null); + onNodeImported(); + } + + if (result.errors.length > 0) { + setImportError(`Errors: ${result.errors.join(', ')}`); + } + } catch (error) { + setImportError(error instanceof Error ? error.message : 'Failed to import nodes'); + } + }; + input.click(); + }; + + const handleExportNodes = () => { + try { + const exportData = customNodeManager.exportCustomNodes(); + const blob = new Blob([exportData], { type: 'application/json' }); + const url = URL.createObjectURL(blob); + const a = document.createElement('a'); + a.href = url; + a.download = `custom-nodes-${new Date().toISOString().split('T')[0]}.json`; + document.body.appendChild(a); + a.click(); + document.body.removeChild(a); + URL.revokeObjectURL(url); + } catch (error) { + setImportError('Failed to export nodes'); + } + }; + + if (!isOpen) return null; + + return ( +
+
+ {/* Header */} +
+
+

+ Custom Node Library +

+

+ Discover and import custom nodes from the community +

+
+
+ + + +
+
+ + {/* Filters */} +
+
+
+
+ + setSearchTerm(e.target.value)} + className="w-full pl-10 pr-4 py-2 border border-gray-300 dark:border-gray-600 rounded-lg bg-white dark:bg-gray-700 text-gray-900 dark:text-gray-100 focus:outline-none focus:ring-2 focus:ring-purple-500" + /> +
+
+ +
+ + +
+ + +
+ + {/* Status Messages */} + {importError && ( +
+ {importError} +
+ )} + + {importSuccess && ( +
+ {importSuccess} +
+ )} +
+ + {/* Node Grid */} +
+
+ {filteredNodes.map((node) => ( +
+
+
{node.icon}
+
+

+ {node.name} +

+

+ by {node.customMetadata.createdBy} • v{node.version} +

+
+
+ + {node.customMetadata.downloadCount} +
+
+ + {node.customMetadata.rating} +
+
+
+
+ +

+ {node.description} +

+ +
+ {node.metadata?.tags?.slice(0, 3).map((tag, idx) => ( + + {tag} + + ))} +
+ +
+
+ {node.inputs.length} inputs • {node.outputs.length} outputs +
+ +
+
+ ))} +
+ + {filteredNodes.length === 0 && ( +
+
+ +
+

+ No nodes found +

+

+ Try adjusting your search or filter criteria +

+
+ )} +
+
+
+ ); +}; + +export default CustomNodeLibrary; \ No newline at end of file diff --git a/src/components/AgentBuilder/NodeCreator/CustomNodeManager.ts b/src/components/AgentBuilder/NodeCreator/CustomNodeManager.ts new file mode 100644 index 00000000..b7d1a528 --- /dev/null +++ b/src/components/AgentBuilder/NodeCreator/CustomNodeManager.ts @@ -0,0 +1,372 @@ +import { CustomNodeDefinition } from '../../../types/agent/types'; + +export interface CustomNodeExecutionContext { + log: (...args: any[]) => void; + fetch?: (url: string, options?: RequestInit) => Promise; + setTimeout?: (callback: () => void, delay: number) => number; + clearTimeout?: (id: number) => void; +} + +export class CustomNodeManager { + private customNodes: Map = new Map(); + private nodeComponents: Map = new Map(); + private executionResults: Map = new Map(); + + constructor() { + this.loadStoredNodes(); + } + + /** + * Register a custom node definition + */ + registerCustomNode(nodeDefinition: CustomNodeDefinition): void { + try { + console.log('CustomNodeManager: Registering node:', nodeDefinition); + this.validateNode(nodeDefinition); + this.customNodes.set(nodeDefinition.type, nodeDefinition); + + // Store in localStorage for persistence + this.saveToStorage(); + console.log('CustomNodeManager: Node registered and saved to storage'); + + console.log(`Custom node registered: ${nodeDefinition.name}`); + } catch (error) { + console.error('Failed to register custom node:', error); + throw error; + } + } + + /** + * Unregister a custom node + */ + unregisterCustomNode(nodeType: string): boolean { + const removed = this.customNodes.delete(nodeType); + if (removed) { + this.nodeComponents.delete(nodeType); + this.saveToStorage(); + } + return removed; + } + + /** + * Get all registered custom nodes + */ + getCustomNodes(): CustomNodeDefinition[] { + const nodes = Array.from(this.customNodes.values()); + console.log('CustomNodeManager: Getting custom nodes:', nodes); + return nodes; + } + + /** + * Get a specific custom node definition + */ + getCustomNode(nodeType: string): CustomNodeDefinition | undefined { + return this.customNodes.get(nodeType); + } + + /** + * Check if a node type is a custom node + */ + isCustomNode(nodeType: string): boolean { + return this.customNodes.has(nodeType); + } + + /** + * Execute a custom node + */ + async executeCustomNode( + nodeType: string, + inputs: Record, + properties: Record, + context?: Partial + ): Promise { + const nodeDefinition = this.customNodes.get(nodeType); + if (!nodeDefinition) { + throw new Error(`Custom node not found: ${nodeType}`); + } + + try { + // Create execution context + const executionContext: CustomNodeExecutionContext = { + log: context?.log || ((...args) => console.log(`[${nodeDefinition.name}]`, ...args)), + fetch: context?.fetch, + setTimeout: context?.setTimeout || setTimeout, + clearTimeout: context?.clearTimeout || clearTimeout + }; + + // Map input/output IDs to names for easier access + const mappedInputs = this.mapPortsToNames(inputs, nodeDefinition.inputs); + const mappedProperties = this.mapPropertiesToNames(properties, nodeDefinition.properties); + + // Execute the code in a sandboxed environment + const result = await this.sandboxedExecution( + nodeDefinition.executionCode, + mappedInputs, + mappedProperties, + executionContext + ); + + // Map result back to port IDs + const mappedResult = this.mapNamesToPortIds(result, nodeDefinition.outputs); + + return mappedResult; + } catch (error) { + console.error(`Custom node execution failed (${nodeType}):`, error); + throw error; + } + } + + /** + * Validate a custom node definition + */ + private validateNode(nodeDefinition: CustomNodeDefinition): void { + if (!nodeDefinition.name || !nodeDefinition.type) { + throw new Error('Node name and type are required'); + } + + if (!nodeDefinition.executionCode) { + throw new Error('Execution code is required'); + } + + // Basic security check - prevent dangerous patterns + const dangerousPatterns = [ + /require\s*\(/, + /import\s+/, + /eval\s*\(/, + /Function\s*\(/, + /process\./, + /global\./, + /window\./, + /document\./, + /__dirname/, + /__filename/, + /fs\./, + /path\./, + /child_process/, + /cluster/, + /os\./ + ]; + + for (const pattern of dangerousPatterns) { + if (pattern.test(nodeDefinition.executionCode)) { + throw new Error(`Potentially dangerous code pattern detected: ${pattern.source}`); + } + } + + // Validate that execute function exists + if (!nodeDefinition.executionCode.includes('async function execute') && + !nodeDefinition.executionCode.includes('function execute')) { + throw new Error('Execution code must contain an "execute" function'); + } + } + + /** + * Execute code in a sandboxed environment + */ + private async sandboxedExecution( + code: string, + inputs: Record, + properties: Record, + context: CustomNodeExecutionContext + ): Promise { + try { + // Create a restricted execution environment + const restrictedGlobals = { + // Allow basic JavaScript features + Object, Array, String, Number, Boolean, Date, Math, JSON, + Promise, setTimeout: context.setTimeout, clearTimeout: context.clearTimeout, + + // Provide controlled context + console: { + log: context.log, + warn: context.log, + error: context.log + }, + + // Controlled fetch if available + ...(context.fetch && { fetch: context.fetch }) + }; + + // Create the execution function with restricted scope + const executionFunction = new Function( + 'inputs', 'properties', 'context', + ...Object.keys(restrictedGlobals), + ` + "use strict"; + ${code} + return execute(inputs, properties, context); + ` + ); + + // Execute with timeout + const timeoutMs = 30000; // 30 second timeout + const timeoutPromise = new Promise((_, reject) => { + setTimeout(() => reject(new Error('Execution timeout')), timeoutMs); + }); + + const executionPromise = executionFunction( + inputs, + properties, + context, + ...Object.values(restrictedGlobals) + ); + + const result = await Promise.race([executionPromise, timeoutPromise]); + return result; + } catch (error) { + throw new Error(`Execution failed: ${error instanceof Error ? error.message : 'Unknown error'}`); + } + } + + /** + * Map port IDs to names for easier access in execution code + */ + private mapPortsToNames(data: Record, ports: any[]): Record { + const mapped: Record = {}; + for (const port of ports) { + if (data[port.id] !== undefined) { + mapped[port.name.toLowerCase().replace(/\s+/g, '_')] = data[port.id]; + } + } + return mapped; + } + + /** + * Map property IDs to names for easier access in execution code + */ + private mapPropertiesToNames(data: Record, properties: any[]): Record { + const mapped: Record = {}; + for (const property of properties) { + if (data[property.id] !== undefined) { + mapped[property.name.toLowerCase().replace(/\s+/g, '_')] = data[property.id]; + } else if (property.defaultValue !== undefined) { + mapped[property.name.toLowerCase().replace(/\s+/g, '_')] = property.defaultValue; + } + } + return mapped; + } + + /** + * Map result names back to port IDs + */ + private mapNamesToPortIds(result: Record, ports: any[]): Record { + const mapped: Record = {}; + for (const port of ports) { + const nameKey = port.name.toLowerCase().replace(/\s+/g, '_'); + if (result[nameKey] !== undefined) { + mapped[port.id] = result[nameKey]; + } + } + return mapped; + } + + /** + * Save custom nodes to localStorage + */ + private saveToStorage(): void { + try { + const nodes = Array.from(this.customNodes.values()); + console.log('CustomNodeManager: Saving nodes to localStorage:', nodes); + localStorage.setItem('custom_nodes', JSON.stringify(nodes)); + console.log('CustomNodeManager: Nodes saved to localStorage successfully'); + } catch (error) { + console.warn('Failed to save custom nodes to storage:', error); + } + } + + /** + * Load custom nodes from localStorage + */ + private loadStoredNodes(): void { + try { + const stored = localStorage.getItem('custom_nodes'); + if (stored) { + const nodes: CustomNodeDefinition[] = JSON.parse(stored); + for (const node of nodes) { + this.customNodes.set(node.type, node); + } + console.log(`Loaded ${nodes.length} custom nodes from storage`); + } + } catch (error) { + console.warn('Failed to load custom nodes from storage:', error); + } + } + + /** + * Export custom nodes for sharing + */ + exportCustomNodes(nodeTypes?: string[]): string { + const nodesToExport = nodeTypes + ? nodeTypes.map(type => this.customNodes.get(type)).filter(Boolean) + : Array.from(this.customNodes.values()); + + const exportData = { + version: '1.0.0', + exportedAt: new Date().toISOString(), + nodes: nodesToExport + }; + + return JSON.stringify(exportData, null, 2); + } + + /** + * Import custom nodes from export data + */ + importCustomNodes(exportData: string): { imported: number; errors: string[] } { + try { + const data = JSON.parse(exportData); + const errors: string[] = []; + let imported = 0; + + if (!data.nodes || !Array.isArray(data.nodes)) { + throw new Error('Invalid export format'); + } + + for (const node of data.nodes) { + try { + this.registerCustomNode(node); + imported++; + } catch (error) { + errors.push(`Failed to import ${node.name}: ${error instanceof Error ? error.message : 'Unknown error'}`); + } + } + + return { imported, errors }; + } catch (error) { + throw new Error(`Import failed: ${error instanceof Error ? error.message : 'Unknown error'}`); + } + } + + /** + * Get usage statistics for custom nodes + */ + getUsageStats(): Record { + // This would be enhanced with actual usage tracking + const stats: Record = {}; + for (const [type, node] of this.customNodes) { + stats[type] = node.customMetadata.downloadCount || 0; + } + return stats; + } + + /** + * Clear all custom nodes + */ + clearAllCustomNodes(): void { + this.customNodes.clear(); + this.nodeComponents.clear(); + localStorage.removeItem('custom_nodes'); + } + + /** + * Reload custom nodes from localStorage (public method) + */ + reloadFromStorage(): void { + console.log('CustomNodeManager: Reloading from storage...'); + this.loadStoredNodes(); + console.log('CustomNodeManager: Reload completed, current nodes:', Array.from(this.customNodes.values())); + } +} + +// Global instance +export const customNodeManager = new CustomNodeManager(); \ No newline at end of file diff --git a/src/components/AgentBuilder/NodeCreator/NodeCreator.tsx b/src/components/AgentBuilder/NodeCreator/NodeCreator.tsx new file mode 100644 index 00000000..afbe2c99 --- /dev/null +++ b/src/components/AgentBuilder/NodeCreator/NodeCreator.tsx @@ -0,0 +1,2062 @@ +import React, { useState, useRef, useEffect } from 'react'; +import { + X, Plus, Code, Eye, Save, Play, Settings, Type, Hash, + Braces, Upload, Download, Palette, Zap, Box, ArrowRight, + ArrowLeft, Check, AlertCircle, Info, Trash2, Copy, Move, + ChevronDown, ChevronRight, Monitor, Smartphone, Tablet, + Wand2, Brain, Sparkles, Loader2 +} from 'lucide-react'; +import { CustomNodeDefinition, NodePort, NodePropertyDefinition } from '../../../types/agent/types'; +import Monaco from '@monaco-editor/react'; +import { useProviders } from '../../../contexts/ProvidersContext'; +import { claraProviderService } from '../../../services/claraProviderService'; +import ResponsiveModal from '../ResponsiveModal'; + +interface NodeCreatorProps { + isOpen: boolean; + onClose: () => void; + onSave: (nodeDefinition: CustomNodeDefinition) => void; + editingNode?: CustomNodeDefinition | null; +} + +interface NodeCreatorState { + // Mode selection + creationMode: 'auto' | 'manual' | null; + + // Auto mode state + aiPrompt: string; + selectedProvider: string; + selectedModel: string; + isGenerating: boolean; + generationStatus: string; + generationError: string | null; + isGenerated: boolean; + + // Basic info + name: string; + description: string; + category: string; + icon: string; + tags: string[]; + + // Interface + inputs: NodePort[]; + outputs: NodePort[]; + properties: NodePropertyDefinition[]; + + // Code + executionCode: string; + + // Style + backgroundColor: string; + textColor: string; + borderColor: string; + + // Validation + errors: Record; +} + +const CATEGORIES = [ + 'basic', 'input', 'output', 'data', 'logic', 'ai', 'media', 'text', 'math', 'custom' +]; + +const DATA_TYPES = [ + 'string', 'number', 'boolean', 'object', 'array', 'any', 'file', 'image', 'json' +]; + +const PROPERTY_TYPES = [ + 'string', 'number', 'boolean', 'select', 'multiselect', 'json', 'code', 'color', 'textarea' +]; + +const COMMON_ICONS = [ + '🔧', '⚡', '🎯', '📊', '🧠', '🔄', '📝', '🔍', '🎨', '📱', + '💾', '🌐', '🔐', '📈', '🎵', '📷', '🗂️', '⚙️', '🚀', '💡' +]; + +// Structured output schema for node generation +const NODE_GENERATION_SCHEMA = { + type: "object", + properties: { + name: { + type: "string", + description: "Node name (e.g., 'Text Processor', 'Math Calculator')" + }, + description: { + type: "string", + description: "Brief description of what the node does" + }, + category: { + type: "string", + enum: CATEGORIES, + description: "Node category" + }, + icon: { + type: "string", + description: "Single emoji icon for the node" + }, + inputs: { + type: "array", + items: { + type: "object", + properties: { + name: { type: "string", description: "Input port name" }, + dataType: { type: "string", enum: DATA_TYPES, description: "Data type" }, + required: { type: "boolean", description: "Whether input is required" }, + description: { type: "string", description: "Input description" } + }, + required: ["name", "dataType", "required"] + } + }, + outputs: { + type: "array", + items: { + type: "object", + properties: { + name: { type: "string", description: "Output port name" }, + dataType: { type: "string", enum: DATA_TYPES, description: "Data type" }, + description: { type: "string", description: "Output description" } + }, + required: ["name", "dataType"] + } + }, + properties: { + type: "array", + items: { + type: "object", + properties: { + name: { type: "string", description: "Property name" }, + type: { type: "string", enum: PROPERTY_TYPES, description: "Property type" }, + required: { type: "boolean", description: "Whether property is required" }, + defaultValue: { description: "Default value for the property" }, + description: { type: "string", description: "Property description" }, + options: { + type: "array", + items: { type: "string" }, + description: "Options for select/multiselect types" + } + }, + required: ["name", "type", "required"] + } + }, + executionCode: { + type: "string", + description: "JavaScript execution function code" + }, + tags: { + type: "array", + items: { type: "string" }, + description: "Tags for categorization" + } + }, + required: ["name", "description", "category", "icon", "inputs", "outputs", "executionCode"] +}; + +type StepType = 'mode' | 'auto' | 'basic' | 'interface' | 'code' | 'style' | 'preview'; + +const NodeCreator: React.FC = ({ + isOpen, + onClose, + onSave, + editingNode = null +}) => { + const [currentStep, setCurrentStep] = useState('mode'); + const { providers } = useProviders(); + const [availableModels, setAvailableModels] = useState([]); + + // Generate a unique ID for nodes + const generateId = () => `node-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`; + + const [state, setState] = useState({ + creationMode: 'auto', + aiPrompt: '', + selectedProvider: '', + selectedModel: '', + isGenerating: false, + generationStatus: '', + generationError: null, + isGenerated: false, + name: '', + description: '', + category: 'custom', + icon: '🔧', + tags: [], + inputs: [], + outputs: [], + properties: [], + executionCode: `// Custom Node Execution Function +async function execute(inputs, properties, context) { + // Access inputs: inputs.inputName + // Access properties: properties.propertyName + // Use context.log() for logging + + try { + // Your custom logic here + const result = inputs.input || 'Hello from custom node!'; + + context.log('Processing input:', result); + + // Return outputs object + return { + output: result + }; + } catch (error) { + context.log('Error:', error.message); + throw error; + } +}`, + backgroundColor: '#ffffff', + borderColor: '#6b7280', + textColor: '#374151', + errors: {} + }); + + const monacoRef = useRef(null); + + // Load providers and models on mount + useEffect(() => { + const loadProviders = async () => { + try { + const enabledProviders = providers.filter(p => p.isEnabled); + if (enabledProviders.length > 0) { + const primaryProvider = enabledProviders.find(p => p.isPrimary) || enabledProviders[0]; + updateState({ selectedProvider: primaryProvider.id }); + + // Load models for the primary provider + const models = await claraProviderService.getModels(primaryProvider.id); + setAvailableModels(models); + + // Select a default text model + const textModel = models.find((m: any) => m.type === 'text' || m.type === 'multimodal'); + if (textModel) { + updateState({ selectedModel: textModel.id }); + } + } + } catch (error) { + console.error('Failed to load providers and models:', error); + } + }; + + if (isOpen && !editingNode) { + loadProviders(); + } + }, [isOpen, providers, editingNode]); + + // Handle provider change + const handleProviderChange = async (providerId: string) => { + updateState({ selectedProvider: providerId, selectedModel: '' }); + try { + const models = await claraProviderService.getModels(providerId); + setAvailableModels(models); + + // Select first available text model + const textModel = models.find((m: any) => m.type === 'text' || m.type === 'multimodal'); + if (textModel) { + updateState({ selectedModel: textModel.id }); + } + } catch (error) { + console.error('Failed to load models for provider:', error); + } + }; + + // Generate node using AI with structured output + const generateNode = async () => { + if (!state.aiPrompt.trim() || !state.selectedProvider || !state.selectedModel) { + updateState({ generationError: 'Please provide a prompt, select a provider, and choose a model.' }); + return; + } + + updateState({ isGenerating: true, generationError: null }); + + try { + // Create a comprehensive JSON schema for the node definition + const nodeSchema = { + name: "CustomNodeDefinition", + strict: true, + schema: { + type: "object", + properties: { + name: { + type: "string", + description: "Clear, descriptive name for the node (e.g., 'JSON Parser', 'Email Sender')" + }, + description: { + type: "string", + description: "Detailed explanation of what the node does and its purpose" + }, + category: { + type: "string", + enum: ["basic", "input", "output", "data", "logic", "ai", "media", "text", "math", "custom"], + description: "Category that best fits this node's functionality" + }, + icon: { + type: "string", + description: "Single emoji that represents the node's function" + }, + inputs: { + type: "array", + items: { + type: "object", + properties: { + name: { + type: "string", + description: "Input port name (camelCase, no spaces)" + }, + label: { + type: "string", + description: "Human-readable label for the input" + }, + type: { + type: "string", + enum: ["string", "number", "boolean", "object", "array"], + description: "Data type this input accepts" + }, + description: { + type: "string", + description: "What this input is used for" + }, + required: { + type: "boolean", + description: "Whether this input is mandatory" + } + }, + required: ["name", "label", "type", "description", "required"], + additionalProperties: false + }, + description: "Array of input ports for the node" + }, + outputs: { + type: "array", + items: { + type: "object", + properties: { + name: { + type: "string", + description: "Output port name (camelCase, no spaces)" + }, + label: { + type: "string", + description: "Human-readable label for the output" + }, + type: { + type: "string", + enum: ["string", "number", "boolean", "object", "array"], + description: "Data type this output provides" + }, + description: { + type: "string", + description: "What this output contains" + } + }, + required: ["name", "label", "type", "description"], + additionalProperties: false + }, + description: "Array of output ports for the node" + }, + properties: { + type: "array", + items: { + type: "object", + properties: { + name: { + type: "string", + description: "Property name (camelCase, no spaces)" + }, + label: { + type: "string", + description: "Human-readable label for the property" + }, + type: { + type: "string", + enum: ["string", "number", "boolean", "text", "select"], + description: "Type of input control for this property" + }, + description: { + type: "string", + description: "What this property configures" + }, + defaultValue: { + type: "string", + description: "Default value for this property" + } + }, + required: ["name", "label", "type", "description", "defaultValue"], + additionalProperties: false + }, + description: "Array of configurable properties for the node" + }, + executionCode: { + type: "string", + description: "Complete JavaScript function body that implements the node logic. Use 'inputs.inputName' to access inputs, 'properties.propertyName' for properties, and return an object with output names as keys." + }, + tags: { + type: "array", + items: { + type: "string" + }, + description: "Relevant tags for searching and organizing this node" + } + }, + required: ["name", "description", "category", "icon", "inputs", "outputs", "properties", "executionCode", "tags"], + additionalProperties: false + } + }; + + // Create the structured request messages with better prompt engineering + const messages = [ + { + role: "system", + content: `You are an expert at creating custom nodes for a visual programming interface. + +IMPORTANT: The execution code MUST follow this exact template: + +\`\`\`javascript +async function execute(inputs, properties, context) { + // Your code here - access inputs and properties by name + // Example: const input_value = inputs.input_name; + // Example: const config = properties.property_name; + + try { + // Main logic here + const result = {}; + + // Return outputs by name + // Example: return { output_name: result_value }; + return result; + } catch (error) { + context.log('Error:', error.message); + throw error; + } +} +\`\`\` + +Rules: +1. MUST have an "execute" function that takes (inputs, properties, context) +2. Access inputs by name: inputs.input_name (lowercase, underscores for spaces) +3. Access properties by name: properties.property_name +4. Return an object with outputs by name: { output_name: value } +5. Use context.log() for logging, not console.log +6. Handle errors with try/catch +7. Make the function async if you need to use await +8. Only use safe JavaScript - no require(), import, eval(), etc. + +Generate ONLY valid JSON that matches the schema. No explanations or markdown.` + }, + { + role: "user", + content: `Create a custom node for: ${state.aiPrompt} + +Requirements: +- Name should be clear and descriptive +- Description should explain what the node does +- Choose appropriate category from: basic, input, output, data, logic, ai, media, text, math, custom +- Select a suitable emoji icon +- Define logical inputs and outputs with correct data types +- Add useful configuration properties +- Include complete working JavaScript execution code following the template +- Add relevant tags` + } + ]; + + // Use the providers from the component's hook call + const selectedProviderData = providers.find((p: any) => p.id === state.selectedProvider); + + if (!selectedProviderData || !selectedProviderData.isEnabled) { + throw new Error(`Provider ${state.selectedProvider} not found or not enabled`); + } + + // Use claraProviderService to make the structured output request + // First, ensure the provider is set correctly + const currentProvider = claraProviderService.getCurrentProvider(); + if (!currentProvider || currentProvider.id !== state.selectedProvider) { + // Update the provider if needed + claraProviderService.updateProvider(selectedProviderData); + } + + // Create the request body for structured output + // Extract actual model name from provider:model format if present + let actualModelId = state.selectedModel; + if (actualModelId.includes(':')) { + const parts = actualModelId.split(':'); + actualModelId = parts.slice(1).join(':'); // Everything after the first colon + console.log(`Extracted model ID: "${state.selectedModel}" -> "${actualModelId}"`); + } + + const requestBody = { + model: actualModelId, + messages: messages, + temperature: 0.3, + max_tokens: 10000, + response_format: { + type: "json_schema", + json_schema: nodeSchema + } + }; + + // Make the request using claraProviderService's current client + const client = claraProviderService.getCurrentClient(); + if (!client) { + throw new Error('No API client available'); + } + + // Call the private request method through reflection + const response = await (client as any).request('/chat/completions', 'POST', requestBody); + + if (!response?.choices?.[0]?.message?.content) { + throw new Error('No response content received from API'); + } + + // Parse the structured response + const generatedNodeData = JSON.parse(response.choices[0].message.content); + + // Transform the generated data to match our internal format + const transformedInputs = generatedNodeData.inputs.map((input: any) => ({ + id: `input_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`, + name: input.name, + type: 'input' as const, + dataType: input.type, + required: input.required, + description: input.description + })); + + const transformedOutputs = generatedNodeData.outputs.map((output: any) => ({ + id: `output_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`, + name: output.name, + type: 'output' as const, + dataType: output.type, + description: output.description + })); + + const transformedProperties = generatedNodeData.properties.map((property: any) => ({ + id: `prop_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`, + name: property.name, + type: property.type, + required: false, + defaultValue: property.defaultValue, + description: property.description, + options: property.options + })); + + // Apply the generated data to state + updateState({ + name: generatedNodeData.name, + description: generatedNodeData.description, + category: generatedNodeData.category, + icon: generatedNodeData.icon, + tags: generatedNodeData.tags || [], + inputs: transformedInputs, + outputs: transformedOutputs, + properties: transformedProperties, + executionCode: generatedNodeData.executionCode || '', + isGenerated: true, + generationStatus: 'Generated successfully! You can review and modify the node before saving.', + generationError: null + }); + + // Validate the generated execution code + const codeValidation = validateExecutionCode(generatedNodeData.executionCode || ''); + if (!codeValidation.isValid) { + updateState({ + generationStatus: `⚠️ Generated successfully, but execution code needs adjustment: ${codeValidation.error}`, + errors: { + code: codeValidation.error || 'Generated code needs to be fixed' + } + }); + } + + console.log('Node generated successfully:', generatedNodeData); + + // Move to basic step to review generated content + setCurrentStep('basic'); + + } catch (error) { + console.error('Node generation failed:', error); + updateState({ + generationError: error instanceof Error ? error.message : 'Failed to generate node', + generationStatus: '❌ Generation failed' + }); + } finally { + updateState({ isGenerating: false }); + } + }; + + // Validate execution code format + const validateExecutionCode = (code: string): { isValid: boolean; error?: string } => { + if (!code.trim()) { + return { isValid: false, error: 'Execution code is required' }; + } + + // Check for execute function + const hasExecuteFunction = /(?:async\s+)?function\s+execute\s*\(/.test(code) || + /execute\s*[:=]\s*(?:async\s+)?(?:function\s*)?\(/.test(code); + + if (!hasExecuteFunction) { + return { + isValid: false, + error: 'Code must contain an "execute" function. Use the template: async function execute(inputs, properties, context) { ... }' + }; + } + + // Check for dangerous patterns + const dangerousPatterns = [ + { pattern: /require\s*\(/, message: 'require() is not allowed for security reasons' }, + { pattern: /import\s+/, message: 'import statements are not allowed' }, + { pattern: /eval\s*\(/, message: 'eval() is not allowed for security reasons' }, + { pattern: /Function\s*\(/, message: 'Function constructor is not allowed' }, + { pattern: /process\./, message: 'process object is not allowed' }, + { pattern: /global\./, message: 'global object is not allowed' }, + { pattern: /window\./, message: 'window object is not allowed' }, + { pattern: /document\./, message: 'document object is not allowed' } + ]; + + for (const { pattern, message } of dangerousPatterns) { + if (pattern.test(code)) { + return { isValid: false, error: message }; + } + } + + return { isValid: true }; + }; + + // Generate dynamic execution code with the correct template + const generateDynamicExecutionCode = (): string => { + const inputVariables = state.inputs.map(input => { + const varName = input.name.toLowerCase().replace(/\s+/g, '_'); + return ` const ${varName} = inputs.${varName};`; + }).join('\n'); + + const propertyVariables = state.properties.map(prop => { + const varName = prop.name.toLowerCase().replace(/\s+/g, '_'); + return ` const ${varName} = properties.${varName};`; + }).join('\n'); + + const outputVariables = state.outputs.map(output => { + const varName = output.name.toLowerCase().replace(/\s+/g, '_'); + return ` ${varName}: null, // Set your result here`; + }).join('\n'); + + const inputComments = state.inputs.length > 0 ? + ` // Available inputs:\n${state.inputs.map(input => ` // - ${input.name} (${input.dataType}): ${input.description || 'No description'}`).join('\n')}\n` : ''; + + const propertyComments = state.properties.length > 0 ? + ` // Available properties:\n${state.properties.map(prop => ` // - ${prop.name} (${prop.type}): ${prop.description || 'No description'}`).join('\n')}\n` : ''; + + const outputComments = state.outputs.length > 0 ? + ` // Expected outputs:\n${state.outputs.map(output => ` // - ${output.name} (${output.dataType}): ${output.description || 'No description'}`).join('\n')}\n` : ''; + + return `async function execute(inputs, properties, context) { +${inputComments}${propertyComments}${outputComments} + try { +${inputVariables} +${propertyVariables} + + // Your main logic here + context.log('Processing node...'); + + // Example processing: + // const result = processData(input_data, some_property); + + // Return outputs by name + return { +${outputVariables} + }; + } catch (error) { + context.log('Error:', error.message); + throw error; + } +}`; + }; + + // Update execution code when interface changes + useEffect(() => { + if (!editingNode && (state.inputs.length > 0 || state.outputs.length > 0 || state.properties.length > 0)) { + // Only update if we're not editing an existing node and have some interface defined + const newCode = generateDynamicExecutionCode(); + if (state.executionCode === '' || state.executionCode.includes('// Custom Node Execution Function')) { + updateState({ executionCode: newCode }); + } + } + }, [state.inputs, state.outputs, state.properties, editingNode]); + + // Load editing node data + useEffect(() => { + if (editingNode) { + setState({ + creationMode: 'manual', // Always use manual mode for editing + aiPrompt: '', + selectedProvider: '', + selectedModel: '', + isGenerating: false, + generationStatus: '', + generationError: null, + isGenerated: false, + name: editingNode.name, + description: editingNode.description, + category: editingNode.category, + icon: editingNode.icon, + tags: editingNode.metadata?.tags || [], + inputs: editingNode.inputs, + outputs: editingNode.outputs, + properties: editingNode.properties, + executionCode: editingNode.executionCode || '', + backgroundColor: editingNode.uiConfig?.backgroundColor || '#ffffff', + borderColor: '#6b7280', + textColor: '#374151', + errors: {} + }); + // Skip mode selection for editing + setCurrentStep('basic'); + } else { + // Reset for new node creation + setCurrentStep('mode'); + } + }, [editingNode]); + + const updateState = (updates: Partial) => { + setState(prev => ({ ...prev, ...updates })); + }; + + const validateCurrentStep = (): boolean => { + const errors: Record = {}; + + switch (currentStep) { + case 'mode': + // No validation needed for mode selection + break; + case 'auto': + if (state.creationMode === 'auto') { + if (!state.aiPrompt.trim()) errors.aiPrompt = 'Please describe the node you want to create'; + if (!state.selectedProvider) errors.selectedProvider = 'Please select an AI provider'; + if (!state.selectedModel) errors.selectedModel = 'Please select a model'; + } + break; + case 'basic': + if (!state.name.trim()) errors.name = 'Name is required'; + if (!state.description.trim()) errors.description = 'Description is required'; + break; + case 'interface': + if (state.inputs.length === 0 && state.outputs.length === 0) { + errors.interface = 'At least one input or output is required'; + } + break; + case 'code': + if (!state.executionCode.trim()) { + errors.code = 'Execution code is required'; + } else { + // Validate execution code format + const codeValidation = validateExecutionCode(state.executionCode); + if (!codeValidation.isValid) { + errors.code = codeValidation.error || 'Invalid execution code format'; + } + } + break; + } + + updateState({ errors }); + return Object.keys(errors).length === 0; + }; + + const nextStep = () => { + if (!validateCurrentStep()) return; + + const steps = ['mode', 'auto', 'basic', 'interface', 'code', 'style', 'preview'] as const; + const currentIndex = steps.indexOf(currentStep); + + // Skip auto step if manual mode is selected + if (currentStep === 'mode' && state.creationMode === 'manual') { + setCurrentStep('basic'); + return; + } + + // Skip to basic step if coming from auto mode with generated data + if (currentStep === 'auto' && state.name) { + setCurrentStep('basic'); + return; + } + + // Go to auto step from mode if auto mode is selected + if (currentStep === 'mode' && state.creationMode === 'auto') { + setCurrentStep('auto'); + return; + } + + if (currentIndex < steps.length - 1) { + setCurrentStep(steps[currentIndex + 1]); + } + }; + + const prevStep = () => { + const steps = ['mode', 'auto', 'basic', 'interface', 'code', 'style', 'preview'] as const; + const currentIndex = steps.indexOf(currentStep); + + // Handle back navigation properly + if (currentStep === 'basic' && state.creationMode === 'manual') { + setCurrentStep('mode'); + return; + } + + if (currentStep === 'basic' && state.creationMode === 'auto') { + setCurrentStep('auto'); + return; + } + + if (currentStep === 'auto') { + setCurrentStep('mode'); + return; + } + + if (currentIndex > 0) { + setCurrentStep(steps[currentIndex - 1]); + } + }; + + const addInput = () => { + const newInput: NodePort = { + id: `input_${Date.now()}`, + name: `Input ${state.inputs.length + 1}`, + type: 'input', + dataType: 'string', + required: false, + description: '' + }; + updateState({ inputs: [...state.inputs, newInput] }); + }; + + const addOutput = () => { + const newOutput: NodePort = { + id: `output_${Date.now()}`, + name: `Output ${state.outputs.length + 1}`, + type: 'output', + dataType: 'string', + description: '' + }; + updateState({ outputs: [...state.outputs, newOutput] }); + }; + + const addProperty = () => { + const newProperty: NodePropertyDefinition = { + id: `prop_${Date.now()}`, + name: `Property ${state.properties.length + 1}`, + type: 'string', + required: false, + defaultValue: '', + description: '' + }; + updateState({ properties: [...state.properties, newProperty] }); + }; + + const updateInput = (index: number, updates: Partial) => { + const newInputs = [...state.inputs]; + newInputs[index] = { ...newInputs[index], ...updates }; + updateState({ inputs: newInputs }); + }; + + const updateOutput = (index: number, updates: Partial) => { + const newOutputs = [...state.outputs]; + newOutputs[index] = { ...newOutputs[index], ...updates }; + updateState({ outputs: newOutputs }); + }; + + const updateProperty = (index: number, updates: Partial) => { + const newProperties = [...state.properties]; + newProperties[index] = { ...newProperties[index], ...updates }; + updateState({ properties: newProperties }); + }; + + const removeInput = (index: number) => { + updateState({ inputs: state.inputs.filter((_, i) => i !== index) }); + }; + + const removeOutput = (index: number) => { + updateState({ outputs: state.outputs.filter((_, i) => i !== index) }); + }; + + const removeProperty = (index: number) => { + updateState({ properties: state.properties.filter((_, i) => i !== index) }); + }; + + const handleSave = () => { + if (!validateCurrentStep()) return; + + // Validate execution code format before saving + const codeValidation = validateExecutionCode(state.executionCode); + if (!codeValidation.isValid) { + updateState({ + errors: { + ...state.errors, + code: codeValidation.error || 'Invalid execution code format' + } + }); + setCurrentStep('code'); // Go to code step to show error + return; + } + + const nodeDefinition: CustomNodeDefinition = { + id: editingNode?.id || generateId(), + name: state.name, + type: state.name.toLowerCase().replace(/\s+/g, '-'), + category: state.category, + description: state.description, + icon: state.icon, + version: '1.0.0', + author: 'User', + inputs: state.inputs, + outputs: state.outputs, + properties: state.properties, + executionHandler: 'custom-node-handler', + executionCode: state.executionCode, + uiConfig: { + backgroundColor: state.backgroundColor, + iconUrl: undefined, + customStyling: '' + }, + customMetadata: { + isUserCreated: true, + createdBy: 'current-user', + createdAt: editingNode?.customMetadata?.createdAt || new Date().toISOString(), + sharedWith: editingNode?.customMetadata?.sharedWith || [], + published: editingNode?.customMetadata?.published || false, + downloadCount: editingNode?.customMetadata?.downloadCount || 0, + rating: editingNode?.customMetadata?.rating || 0 + }, + metadata: { + tags: state.tags, + documentation: '', + examples: [] + } + }; + + onSave(nodeDefinition); + onClose(); + }; + + // Test the execution code + const testCode = async () => { + try { + // Create mock inputs based on node definition + const mockInputs: Record = {}; + state.inputs.forEach(input => { + switch (input.dataType) { + case 'string': + mockInputs[input.name] = 'test string'; + break; + case 'number': + mockInputs[input.name] = 42; + break; + case 'boolean': + mockInputs[input.name] = true; + break; + case 'object': + mockInputs[input.name] = { test: 'value' }; + break; + case 'array': + mockInputs[input.name] = ['item1', 'item2']; + break; + default: + mockInputs[input.name] = 'test'; + } + }); + + // Create mock properties + const mockProperties: Record = {}; + state.properties.forEach(prop => { + mockProperties[prop.name] = prop.defaultValue; + }); + + // Execute the code + const AsyncFunction = Object.getPrototypeOf(async function(){}).constructor; + const executor = new AsyncFunction('inputs', 'properties', 'fetch', state.executionCode); + const result = await executor(mockInputs, mockProperties, fetch); + + updateState({ + errors: { ...state.errors, code: '' } + }); + + console.log('Test execution result:', result); + return result; + } catch (error) { + updateState({ + errors: { ...state.errors, code: `Test failed: ${error instanceof Error ? error.message : 'Unknown error'}` } + }); + console.error('Code test failed:', error); + throw error; + } + }; + + if (!isOpen) return null; + + const renderStepIndicator = () => { + // Filter steps based on creation mode + const allSteps = [ + { key: 'mode', label: 'Mode', icon: Wand2 }, + { key: 'auto', label: 'Auto', icon: Brain }, + { key: 'basic', label: 'Basic', icon: Info }, + { key: 'interface', label: 'Interface', icon: Box }, + { key: 'code', label: 'Code', icon: Code }, + { key: 'style', label: 'Style', icon: Palette }, + { key: 'preview', label: 'Preview', icon: Eye } + ]; + + // Filter steps based on creation mode + const steps = state.creationMode === 'manual' + ? allSteps.filter(step => step.key !== 'auto') + : allSteps; + + return ( +
+ {steps.map((step, index) => { + const Icon = step.icon; + const isActive = step.key === currentStep; + const stepIndex = allSteps.findIndex(s => s.key === step.key); + const currentStepIndex = allSteps.findIndex(s => s.key === currentStep); + const isCompleted = currentStepIndex > stepIndex; + + return ( + +
+
+ {isCompleted ? : } +
+ + {step.label} + +
+ {index < steps.length - 1 && ( + + )} +
+ ); + })} +
+ ); + }; + + const renderBasicStep = () => ( +
+
+ + updateState({ name: e.target.value })} + placeholder="My Custom Node" + className="w-full px-3 py-2 border border-gray-300 dark:border-gray-600 rounded-lg bg-white dark:bg-gray-700 text-gray-900 dark:text-gray-100 focus:outline-none focus:ring-2 focus:ring-sakura-500" + /> + {state.errors.name && ( +

{state.errors.name}

+ )} +
+ +
+ +