Some checks failed
Test / rust-fmt-check (pull_request) Failing after 0s
Test / rust-clippy (pull_request) Failing after 1s
Test / rust-tests (pull_request) Failing after 0s
Test / frontend-typecheck (pull_request) Failing after 16s
Test / frontend-tests (pull_request) Failing after 18s
PR Review Automation / review (pull_request) Failing after 4m13s
Complete backport of all features from apollo_nxt-trcaa repository: - Three-tier shell execution safety system (Tier 1: auto, Tier 2: approve, Tier 3: deny) - Ollama function calling with tool use support - AI provider tool calling auto-detection - kubectl binary bundling and management - kubeconfig upload and context management - Shell approval modal with real-time UI - MCP protocol HTTP transport with custom headers - Enhanced security audit logging - Comprehensive test coverage (275+ tests) - Updated CI/CD workflows for Gitea Actions - Complete documentation (ADRs, wiki, release notes) Sanitization applied to all files: - Removed all MSI, Motorola, VNXT, Vesta references - Replaced internal infrastructure references with TFTSR equivalents - Updated all URLs and API endpoints - Sanitized commit history references in documentation Technical changes: - New modules: shell/classifier, shell/executor, shell/kubectl, shell/kubeconfig - Enhanced AI providers: ollama.rs, openai.rs with function calling - New Tauri commands: shell execution, kubeconfig management, tool calling detection - Database migrations: shell_execution_audit table - Frontend: ShellApprovalModal, ShellExecution, KubeconfigManager pages - CI/CD: kubectl bundling, multi-platform builds, Gitea Actions integration Version: 1.0.8 Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
363 lines
13 KiB
Markdown
363 lines
13 KiB
Markdown
# AI Providers
|
||
|
||
TRCAA supports 6+ AI providers, including custom providers with flexible authentication and API formats. API keys are stored encrypted with AES-256-GCM.
|
||
|
||
## Provider Factory
|
||
|
||
`ai/provider.rs::create_provider(config)` dispatches on `config.name` to the matching implementation. Adding a provider requires implementing the `Provider` trait and adding a match arm.
|
||
|
||
```rust
|
||
pub trait Provider {
|
||
async fn chat(&self, messages: Vec<Message>, config: &ProviderConfig) -> Result<ChatResponse>;
|
||
fn name(&self) -> &str;
|
||
}
|
||
```
|
||
|
||
---
|
||
|
||
## Supported Providers
|
||
|
||
### 1. OpenAI-Compatible
|
||
|
||
Covers: OpenAI, Azure OpenAI, LM Studio, vLLM, **LiteLLM (AWS Bedrock)**, and any OpenAI-API-compatible endpoint.
|
||
|
||
| Field | Value |
|
||
|-------|-------|
|
||
| `config.name` | `"openai"` |
|
||
| Default URL | `https://api.openai.com/v1/chat/completions` |
|
||
| Auth | `Authorization: Bearer <api_key>` |
|
||
| Max tokens | 4096 |
|
||
|
||
**Models:** `gpt-4o`, `gpt-4o-mini`, `gpt-4-turbo`
|
||
|
||
**Custom endpoint:** Set `config.base_url` to any OpenAI-compatible API:
|
||
- LM Studio: `http://localhost:1234/v1`
|
||
- **LiteLLM (AWS Bedrock):** `http://localhost:8000/v1` — See [LiteLLM + Bedrock Setup](LiteLLM-Bedrock-Setup) for full configuration guide
|
||
|
||
---
|
||
|
||
### 2. Anthropic Claude
|
||
|
||
| Field | Value |
|
||
|-------|-------|
|
||
| `config.name` | `"anthropic"` |
|
||
| URL | `https://api.anthropic.com/v1/messages` |
|
||
| Auth | `x-api-key: <api_key>` + `anthropic-version: 2023-06-01` |
|
||
| Max tokens | 4096 |
|
||
|
||
**Models:** `claude-sonnet-4-20250514`, `claude-haiku-4-20250414`, `claude-3-5-sonnet-20241022`
|
||
|
||
---
|
||
|
||
### 3. Google Gemini
|
||
|
||
| Field | Value |
|
||
|-------|-------|
|
||
| `config.name` | `"gemini"` |
|
||
| URL | `https://generativelanguage.googleapis.com/v1beta/models/{model}:generateContent` |
|
||
| Auth | `x-goog-api-key: <api_key>` header |
|
||
| Max tokens | 4096 |
|
||
|
||
**Models:** `gemini-2.0-flash`, `gemini-2.0-pro`, `gemini-1.5-pro`, `gemini-1.5-flash`
|
||
|
||
---
|
||
|
||
## Transport Security Notes
|
||
|
||
- Provider clients use TLS certificate verification via `reqwest`
|
||
- Provider calls are configured with explicit request timeouts to avoid indefinite hangs
|
||
- Credentials are sent in headers (not URL query strings)
|
||
|
||
---
|
||
|
||
### 4. Mistral AI
|
||
|
||
| Field | Value |
|
||
|-------|-------|
|
||
| `config.name` | `"mistral"` |
|
||
| Default URL | `https://api.mistral.ai/v1/chat/completions` |
|
||
| Auth | `Authorization: Bearer <api_key>` |
|
||
| Max tokens | 4096 |
|
||
|
||
**Models:** `mistral-large-latest`, `mistral-medium-latest`, `mistral-small-latest`, `open-mistral-nemo`
|
||
|
||
Uses OpenAI-compatible request/response format.
|
||
|
||
---
|
||
|
||
### 5. Ollama (Local / Offline)
|
||
|
||
| Field | Value |
|
||
|-------|-------|
|
||
| `config.name` | `"ollama"` |
|
||
| Default URL | `http://localhost:11434` |
|
||
| Auth | None |
|
||
| Max tokens | No limit enforced |
|
||
| **Tool Calling** | ✅ **Fully Supported** (v1.0.7+) |
|
||
| Timeout | 180s (tool calling), 60s (regular chat) |
|
||
| Retry Logic | 3 attempts with 2s delay |
|
||
|
||
**Recommended Models (≥3B parameters):**
|
||
|
||
| Model | Size | Min RAM | Notes |
|
||
|-------|------|---------|-------|
|
||
| `llama3.2:3b` | 2.0 GB | 6 GB | Balanced performance |
|
||
| `phi3.5:3.8b` | 2.2 GB | 6 GB | Excellent reasoning |
|
||
| `llama3.1:8b` | 4.7 GB | 10 GB | **RECOMMENDED** - Strong IT analysis |
|
||
| `qwen2.5:14b` | 9.0 GB | 16 GB | Best for complex log analysis |
|
||
| `gemma2:9b` | 5.5 GB | 12 GB | Google's efficient model |
|
||
|
||
**⚠️ Important:** Models with <3B parameters (e.g., `llama3.2:1b`) cannot reliably follow tool calling instructions. They will describe tools instead of invoking them.
|
||
|
||
**Features:**
|
||
- ✅ **Function Calling Support** (v1.0.7): Executes shell commands, kubectl operations
|
||
- ✅ **Multi-turn Tool Conversations**: Preserves `tool_call_id` for correlation
|
||
- ✅ **Resilient Parsing**: Skips malformed tool calls with warnings
|
||
- ✅ **Connection Reliability** (v1.0.8): Health checks, retry logic, extended timeouts
|
||
- ✅ **Auto-Start**: Automatically starts Ollama service if not running
|
||
- ✅ **Fully Offline**: No internet required, complete privacy
|
||
|
||
**Custom URL:** Change the Ollama URL in Settings → AI Providers → Ollama (stored in `settingsStore.ollama_url`).
|
||
|
||
**Troubleshooting:**
|
||
|
||
| Error | Cause | Solution |
|
||
|-------|-------|----------|
|
||
| "Cannot connect to Ollama" | Service not running | Run `ollama serve` or check auto-start |
|
||
| Timeout after 60s (chat) / 180s (tool calling) | Model too slow / tool calling needs more time | Use a smaller model, reduce tool usage, or wait for the higher tool-calling timeout to elapse |
|
||
| Tool calls described but not executed | Model too small (<3B) | Use `llama3.2:3b` or larger |
|
||
| Model not loaded | First request loads model | Wait 5-10s for model to load into VRAM |
|
||
|
||
**Performance Tips:**
|
||
- Use quantized models (Q4_K_M or Q4_0) for faster responses
|
||
- Keep model loaded with `ollama run <model>` in background
|
||
- Monitor VRAM usage - models stay loaded for 5 minutes by default
|
||
|
||
---
|
||
|
||
## Domain System Prompts
|
||
|
||
Each triage conversation is pre-loaded with a domain-specific expert system prompt from `src/lib/domainPrompts.ts`.
|
||
|
||
| Domain | Key areas covered |
|
||
|--------|------------------|
|
||
| **Linux** | systemd, filesystem, memory, networking, kernel, performance |
|
||
| **Windows** | Event Viewer, Active Directory, IIS, Group Policy, clustering |
|
||
| **Network** | DNS, firewalls, load balancers, BGP/OSPF, Layer 2, VPN |
|
||
| **Kubernetes** | Pod failures, service mesh, ingress, storage, Helm |
|
||
| **Databases** | Connection pools, slow queries, indexes, replication, MongoDB/Redis |
|
||
| **Virtualization** | vMotion, storage (VMFS), HA, snapshots; KVM/QEMU |
|
||
| **Hardware** | RAID, SMART data, ECC memory errors, thermal, BIOS/firmware |
|
||
| **Observability** | Prometheus/Grafana, ELK/OpenSearch, tracing, SLO/SLI burn rates |
|
||
|
||
The domain prompt is injected as the first `system` role message in every new conversation.
|
||
|
||
---
|
||
|
||
## 6. Custom Provider (Multiple API Formats)
|
||
|
||
**Status:** ✅ **Implemented** (v0.2.6)
|
||
|
||
Custom providers allow integration with non-OpenAI-compatible APIs. The application supports multiple API formats:
|
||
|
||
### Format: OpenAI Compatible (Default)
|
||
|
||
Standard OpenAI `/chat/completions` endpoint with Bearer authentication.
|
||
|
||
| Field | Default Value |
|
||
|-------|--------------|
|
||
| `api_format` | `"openai"` |
|
||
| `custom_endpoint_path` | `/chat/completions` |
|
||
| `custom_auth_header` | `Authorization` |
|
||
| `custom_auth_prefix` | `Bearer ` |
|
||
|
||
**Use cases:**
|
||
- Self-hosted LLMs with OpenAI-compatible APIs
|
||
- Custom proxy services
|
||
- Enterprise gateways
|
||
|
||
---
|
||
|
||
### Format: TFTSR GenAI
|
||
|
||
**TFTSR GenAI Gateway** — Enterprise AI gateway with model proxying and cost tracking.
|
||
|
||
| Field | Value |
|
||
|-------|-------|
|
||
| `config.provider_type` | `"custom"` |
|
||
| `config.api_format` | `"msi-genai"` |
|
||
| Status | ⚠️ **Limited compatibility** |
|
||
|
||
**Known Limitations:**
|
||
- ❌ **Tool calling not supported**: Gateway returns `503 Service Unavailable` with error `"Gemini Filter Triggered: UNEXPECTED_TOOL_CALL"`
|
||
- ❌ **Shell execution unavailable**: Cannot use `execute_shell_command` or other function calling features
|
||
- ✅ **Basic chat works**: Text-only conversations function correctly
|
||
- ✅ **Workaround parser included**: Attempts to extract tool calls from malformed responses (ChatGPT JSON in `msg` field, Claude XML wrapper)
|
||
|
||
**Recommendation**: Use **LiteLLM + AWS Bedrock** (see [LiteLLM Setup Guide](LiteLLM-Bedrock-Setup)) or **Ollama** for full tool calling support.
|
||
|
||
**Root Cause**: TFTSR GenAI gateway applies content filtering that blocks structured tool call responses before they reach the client. This is a gateway-level restriction that cannot be worked around from the client side.
|
||
|
||
**Configuration (if needed for text-only use):**
|
||
```
|
||
Name: TFTSR GenAI
|
||
Type: Custom
|
||
API Format: TFTSR GenAI
|
||
API URL: https://your-gateway/api/v2/chat
|
||
Model: your-model-name
|
||
API Key: (your API key)
|
||
User ID: user@example.com (optional, for cost tracking)
|
||
```
|
||
|
||
---
|
||
|
||
### Format: Custom REST (Generic)
|
||
|
||
**Generic Enterprise AI Gateway** — For AI platforms that use a non-OpenAI request/response format with centralized cost tracking and model access.
|
||
|
||
| Field | Value |
|
||
|-------|-------|
|
||
| `config.provider_type` | `"custom"` |
|
||
| `config.api_format` | `"custom_rest"` |
|
||
| API URL | Your gateway's base URL |
|
||
| Auth Header | Your gateway's auth header name |
|
||
| Auth Prefix | `` (empty if no prefix needed) |
|
||
| Endpoint Path | `` (empty if URL already includes full path) |
|
||
|
||
**Request Format:**
|
||
```json
|
||
{
|
||
"model": "model-name",
|
||
"prompt": "User's latest message",
|
||
"system": "Optional system prompt",
|
||
"sessionId": "uuid-for-conversation-continuity",
|
||
"userId": "user@example.com"
|
||
}
|
||
```
|
||
|
||
**Response Format:**
|
||
```json
|
||
{
|
||
"status": true,
|
||
"sessionId": "uuid",
|
||
"msg": "AI response text",
|
||
"initialPrompt": false
|
||
}
|
||
```
|
||
|
||
**Key Differences from OpenAI:**
|
||
- **Single prompt** instead of message array (server manages history via `sessionId`)
|
||
- **Response in `msg` field** instead of `choices[0].message.content`
|
||
- **Session-based** conversation continuity (no need to resend history)
|
||
- **Cost tracking** via `userId` field (optional)
|
||
|
||
**Configuration (Settings → AI Providers → Add Provider):**
|
||
```
|
||
Name: Custom REST Gateway
|
||
Type: Custom
|
||
API Format: Custom REST
|
||
API URL: https://your-gateway/api/v2/chat
|
||
Model: your-model-name
|
||
API Key: (your API key)
|
||
User ID: user@example.com (optional, for cost tracking)
|
||
Endpoint Path: (leave empty if URL includes full path)
|
||
Auth Header: x-custom-api-key
|
||
Auth Prefix: (leave empty if no prefix)
|
||
```
|
||
|
||
**Troubleshooting:**
|
||
|
||
| Error | Cause | Solution |
|
||
|-------|-------|----------|
|
||
| 403 Forbidden | Invalid API key or insufficient permissions | Verify key in your gateway portal, check model access |
|
||
| Missing `userId` field | Configuration not saved | Ensure UI shows User ID field when `api_format=custom_rest` |
|
||
| No conversation history | `sessionId` not persisted | Session ID stored in `ProviderConfig.session_id` — currently per-provider, not per-conversation |
|
||
|
||
**Implementation Details:**
|
||
- Backend: `src-tauri/src/ai/openai.rs::chat_custom_rest()`
|
||
- Schema: `src-tauri/src/state.rs::ProviderConfig` (added `user_id`, `api_format`, custom auth fields)
|
||
- Frontend: `src/pages/Settings/AIProviders.tsx` (conditional UI for Custom REST + model dropdown)
|
||
|
||
---
|
||
|
||
## Custom Provider Configuration Fields
|
||
|
||
All providers support the following optional configuration fields (v0.2.6+):
|
||
|
||
| Field | Type | Purpose | Default |
|
||
|-------|------|---------|---------|
|
||
| `custom_endpoint_path` | `Option<String>` | Override endpoint path | `/chat/completions` |
|
||
| `custom_auth_header` | `Option<String>` | Custom auth header name | `Authorization` |
|
||
| `custom_auth_prefix` | `Option<String>` | Prefix before API key | `Bearer ` |
|
||
| `api_format` | `Option<String>` | API format (`openai` or `custom_rest`) | `openai` |
|
||
| `session_id` | `Option<String>` | Session ID for stateful APIs | None |
|
||
| `user_id` | `Option<String>` | User ID for cost tracking (Custom REST gateways) | None |
|
||
| `supports_tool_calling` | `Option<bool>` | Enable function/tool calling | `true` for built-in providers, `false` for custom |
|
||
|
||
**Backward Compatibility:**
|
||
All fields are optional and default to OpenAI-compatible behavior. Existing provider configurations are unaffected.
|
||
|
||
---
|
||
|
||
## Tool Calling Auto-Detection
|
||
|
||
**Status:** ✅ **Implemented** (v1.0.9+)
|
||
|
||
TRCAA can automatically detect whether a custom AI provider supports tool calling (function calling) by sending a test tool call and analyzing the response.
|
||
|
||
### How It Works
|
||
|
||
1. Navigate to **Settings → AI Providers** → Add/Edit Custom Provider
|
||
2. Configure your provider (API URL, key, model)
|
||
3. Click **"Auto-Detect Tool Calling Support"** button
|
||
4. System sends a simple test tool call to the provider
|
||
5. Checkbox automatically enabled/disabled based on result
|
||
6. Success/warning message displayed
|
||
|
||
### Detection Criteria
|
||
|
||
| Scenario | Result | Explanation |
|
||
|----------|--------|-------------|
|
||
| Provider returns `tool_calls` array with test tool | ✅ Tool calling supported | Checkbox enabled automatically |
|
||
| Provider responds without tool_calls | ⚠️ Not supported | Checkbox disabled automatically |
|
||
| Gateway returns 503 / "tool" error | ⚠️ Blocked at gateway level | Checkbox disabled (e.g., TFTSR GenAI) |
|
||
| Connection/auth/timeout error | ❌ Error displayed | User must fix connection issue |
|
||
|
||
### Test Tool
|
||
|
||
The auto-detection sends this minimal tool:
|
||
|
||
```rust
|
||
{
|
||
"name": "test_tool",
|
||
"description": "A test tool that returns 'success'. Call this tool with no arguments.",
|
||
"parameters": {
|
||
"type": "object",
|
||
"properties": {},
|
||
"required": []
|
||
}
|
||
}
|
||
```
|
||
|
||
### Known Limitations
|
||
|
||
- **TFTSR GenAI**: Gateway blocks tool calls with `503 UNEXPECTED_TOOL_CALL` before they reach the model. Auto-detect correctly identifies this as "not supported."
|
||
- **Small Models**: Models <3B parameters (e.g., `llama3.2:1b`) may respond but describe tools instead of calling them. Auto-detect may return `true` (model capability) but runtime behavior will fail.
|
||
- **Timeout**: Detection uses same timeout as regular chat (60-180s depending on provider). Slow providers may timeout during detection.
|
||
|
||
### Manual Override
|
||
|
||
You can always manually toggle the `supports_tool_calling` checkbox:
|
||
- ✅ Enable: For providers you know support tool calling
|
||
- ❌ Disable: For text-only chat without shell execution or integrations
|
||
|
||
---
|
||
|
||
## Adding a New Provider
|
||
|
||
1. Create `src-tauri/src/ai/{name}.rs` implementing the `Provider` trait
|
||
2. Add a match arm in `ai/provider.rs::create_provider()`
|
||
3. Add the model list in `commands/ai.rs::list_providers()`
|
||
4. Add the TypeScript type in `src/lib/tauriCommands.ts`
|
||
5. Add a UI entry in `src/pages/Settings/AIProviders.tsx`
|