Some checks failed
CI / Check / macos-latest (push) Has been cancelled
CI / Check / ubuntu-latest (push) Has been cancelled
CI / Check / windows-latest (push) Has been cancelled
CI / Test / macos-latest (push) Has been cancelled
CI / Test / ubuntu-latest (push) Has been cancelled
CI / Test / windows-latest (push) Has been cancelled
CI / Clippy (push) Has been cancelled
CI / Format (push) Has been cancelled
CI / Security Audit (push) Has been cancelled
CI / Secrets Scan (push) Has been cancelled
CI / Install Script Smoke Test (push) Has been cancelled
545 lines
20 KiB
JavaScript
545 lines
20 KiB
JavaScript
// OpenFang Setup Wizard — First-run guided setup (Provider + Agent + Channel)
|
|
'use strict';
|
|
|
|
function wizardPage() {
|
|
return {
|
|
step: 1,
|
|
totalSteps: 6,
|
|
loading: false,
|
|
error: '',
|
|
|
|
// Step 2: Provider setup
|
|
providers: [],
|
|
selectedProvider: '',
|
|
apiKeyInput: '',
|
|
testingProvider: false,
|
|
testResult: null,
|
|
savingKey: false,
|
|
keySaved: false,
|
|
|
|
// Step 3: Agent creation
|
|
templates: [
|
|
{
|
|
id: 'assistant',
|
|
name: 'General Assistant',
|
|
description: 'A versatile helper for everyday tasks, answering questions, and providing recommendations.',
|
|
icon: 'GA',
|
|
category: 'General',
|
|
provider: 'deepseek',
|
|
model: 'deepseek-chat',
|
|
profile: 'balanced',
|
|
system_prompt: 'You are a helpful, friendly assistant. Provide clear, accurate, and concise responses. Ask clarifying questions when needed.'
|
|
},
|
|
{
|
|
id: 'coder',
|
|
name: 'Code Helper',
|
|
description: 'A programming-focused agent that writes, reviews, and debugs code across multiple languages.',
|
|
icon: 'CH',
|
|
category: 'Development',
|
|
provider: 'deepseek',
|
|
model: 'deepseek-chat',
|
|
profile: 'precise',
|
|
system_prompt: 'You are an expert programmer. Help users write clean, efficient code. Explain your reasoning. Follow best practices and conventions for the language being used.'
|
|
},
|
|
{
|
|
id: 'researcher',
|
|
name: 'Researcher',
|
|
description: 'An analytical agent that breaks down complex topics, synthesizes information, and provides cited summaries.',
|
|
icon: 'RS',
|
|
category: 'Research',
|
|
provider: 'gemini',
|
|
model: 'gemini-2.5-flash',
|
|
profile: 'balanced',
|
|
system_prompt: 'You are a research analyst. Break down complex topics into clear explanations. Provide structured analysis with key findings. Cite sources when available.'
|
|
},
|
|
{
|
|
id: 'writer',
|
|
name: 'Writer',
|
|
description: 'A creative writing agent that helps with drafting, editing, and improving written content of all kinds.',
|
|
icon: 'WR',
|
|
category: 'Writing',
|
|
provider: 'deepseek',
|
|
model: 'deepseek-chat',
|
|
profile: 'creative',
|
|
system_prompt: 'You are a skilled writer and editor. Help users create polished content. Adapt your tone and style to match the intended audience. Offer constructive suggestions for improvement.'
|
|
},
|
|
{
|
|
id: 'data-analyst',
|
|
name: 'Data Analyst',
|
|
description: 'A data-focused agent that helps analyze datasets, create queries, and interpret statistical results.',
|
|
icon: 'DA',
|
|
category: 'Development',
|
|
provider: 'gemini',
|
|
model: 'gemini-2.5-flash',
|
|
profile: 'precise',
|
|
system_prompt: 'You are a data analysis expert. Help users understand their data, write SQL/Python queries, and interpret results. Present findings clearly with actionable insights.'
|
|
},
|
|
{
|
|
id: 'devops',
|
|
name: 'DevOps Engineer',
|
|
description: 'A systems-focused agent for CI/CD, infrastructure, Docker, and deployment troubleshooting.',
|
|
icon: 'DO',
|
|
category: 'Development',
|
|
provider: 'deepseek',
|
|
model: 'deepseek-chat',
|
|
profile: 'precise',
|
|
system_prompt: 'You are a DevOps engineer. Help with CI/CD pipelines, Docker, Kubernetes, infrastructure as code, and deployment. Prioritize reliability and security.'
|
|
},
|
|
{
|
|
id: 'support',
|
|
name: 'Customer Support',
|
|
description: 'A professional, empathetic agent for handling customer inquiries and resolving issues.',
|
|
icon: 'CS',
|
|
category: 'Business',
|
|
provider: 'groq',
|
|
model: 'llama-3.3-70b-versatile',
|
|
profile: 'balanced',
|
|
system_prompt: 'You are a professional customer support representative. Be empathetic, patient, and solution-oriented. Acknowledge concerns before offering solutions. Escalate complex issues appropriately.'
|
|
},
|
|
{
|
|
id: 'tutor',
|
|
name: 'Tutor',
|
|
description: 'A patient educational agent that explains concepts step-by-step and adapts to the learner\'s level.',
|
|
icon: 'TU',
|
|
category: 'General',
|
|
provider: 'groq',
|
|
model: 'llama-3.3-70b-versatile',
|
|
profile: 'balanced',
|
|
system_prompt: 'You are a patient and encouraging tutor. Explain concepts step by step, starting from fundamentals. Use analogies and examples. Check understanding before moving on. Adapt to the learner\'s pace.'
|
|
},
|
|
{
|
|
id: 'api-designer',
|
|
name: 'API Designer',
|
|
description: 'An agent specialized in RESTful API design, OpenAPI specs, and integration architecture.',
|
|
icon: 'AD',
|
|
category: 'Development',
|
|
provider: 'deepseek',
|
|
model: 'deepseek-chat',
|
|
profile: 'precise',
|
|
system_prompt: 'You are an API design expert. Help users design clean, consistent RESTful APIs following best practices. Cover endpoint naming, request/response schemas, error handling, and versioning.'
|
|
},
|
|
{
|
|
id: 'meeting-notes',
|
|
name: 'Meeting Notes',
|
|
description: 'Summarizes meeting transcripts into structured notes with action items and key decisions.',
|
|
icon: 'MN',
|
|
category: 'Business',
|
|
provider: 'groq',
|
|
model: 'llama-3.3-70b-versatile',
|
|
profile: 'precise',
|
|
system_prompt: 'You are a meeting summarizer. When given a meeting transcript or notes, produce a structured summary with: key decisions, action items (with owners), discussion highlights, and follow-up questions.'
|
|
}
|
|
],
|
|
selectedTemplate: 0,
|
|
agentName: 'my-assistant',
|
|
creatingAgent: false,
|
|
createdAgent: null,
|
|
|
|
// Step 3: Category filtering
|
|
templateCategory: 'All',
|
|
get templateCategories() {
|
|
var cats = { 'All': true };
|
|
this.templates.forEach(function(t) { if (t.category) cats[t.category] = true; });
|
|
return Object.keys(cats);
|
|
},
|
|
get filteredTemplates() {
|
|
var cat = this.templateCategory;
|
|
if (cat === 'All') return this.templates;
|
|
return this.templates.filter(function(t) { return t.category === cat; });
|
|
},
|
|
|
|
// Step 3: Profile/tool descriptions
|
|
profileDescriptions: {
|
|
minimal: { label: 'Minimal', desc: 'Read-only file access' },
|
|
coding: { label: 'Coding', desc: 'Files + shell + web fetch' },
|
|
research: { label: 'Research', desc: 'Web search + file read/write' },
|
|
balanced: { label: 'Balanced', desc: 'General-purpose tool set' },
|
|
precise: { label: 'Precise', desc: 'Focused tool set for accuracy' },
|
|
creative: { label: 'Creative', desc: 'Full tools with creative emphasis' },
|
|
full: { label: 'Full', desc: 'All 35+ tools' }
|
|
},
|
|
profileInfo: function(name) { return this.profileDescriptions[name] || { label: name, desc: '' }; },
|
|
|
|
// Step 4: Try It chat
|
|
tryItMessages: [],
|
|
tryItInput: '',
|
|
tryItSending: false,
|
|
suggestedMessages: {
|
|
'General': ['What can you help me with?', 'Tell me a fun fact', 'Summarize the latest AI news'],
|
|
'Development': ['Write a Python hello world', 'Explain async/await', 'Review this code snippet'],
|
|
'Research': ['Explain quantum computing simply', 'Compare React vs Vue', 'What are the latest trends in AI?'],
|
|
'Writing': ['Help me write a professional email', 'Improve this paragraph', 'Write a blog intro about AI'],
|
|
'Business': ['Draft a meeting agenda', 'How do I handle a complaint?', 'Create a project status update']
|
|
},
|
|
get currentSuggestions() {
|
|
var tpl = this.templates[this.selectedTemplate];
|
|
var cat = tpl ? tpl.category : 'General';
|
|
return this.suggestedMessages[cat] || this.suggestedMessages['General'];
|
|
},
|
|
async sendTryItMessage(text) {
|
|
if (!text || !text.trim() || !this.createdAgent || this.tryItSending) return;
|
|
text = text.trim();
|
|
this.tryItInput = '';
|
|
this.tryItMessages.push({ role: 'user', text: text });
|
|
this.tryItSending = true;
|
|
try {
|
|
var res = await OpenFangAPI.post('/api/agents/' + this.createdAgent.id + '/message', { message: text });
|
|
this.tryItMessages.push({ role: 'agent', text: res.response || '(no response)' });
|
|
localStorage.setItem('of-first-msg', 'true');
|
|
} catch(e) {
|
|
this.tryItMessages.push({ role: 'agent', text: 'Error: ' + (e.message || 'Could not reach agent') });
|
|
}
|
|
this.tryItSending = false;
|
|
},
|
|
|
|
// Step 5: Channel setup (optional)
|
|
channelType: '',
|
|
channelOptions: [
|
|
{
|
|
name: 'telegram',
|
|
display_name: 'Telegram',
|
|
icon: 'TG',
|
|
description: 'Connect your agent to a Telegram bot for messaging.',
|
|
token_label: 'Bot Token',
|
|
token_placeholder: '123456:ABC-DEF1234ghIkl-zyx57W2v1u123ew11',
|
|
token_env: 'TELEGRAM_BOT_TOKEN',
|
|
help: 'Create a bot via @BotFather on Telegram to get your token.'
|
|
},
|
|
{
|
|
name: 'discord',
|
|
display_name: 'Discord',
|
|
icon: 'DC',
|
|
description: 'Connect your agent to a Discord server via bot token.',
|
|
token_label: 'Bot Token',
|
|
token_placeholder: 'MTIz...abc',
|
|
token_env: 'DISCORD_BOT_TOKEN',
|
|
help: 'Create a Discord application at discord.com/developers and add a bot.'
|
|
},
|
|
{
|
|
name: 'slack',
|
|
display_name: 'Slack',
|
|
icon: 'SL',
|
|
description: 'Connect your agent to a Slack workspace.',
|
|
token_label: 'Bot Token',
|
|
token_placeholder: 'xoxb-...',
|
|
token_env: 'SLACK_BOT_TOKEN',
|
|
help: 'Create a Slack app at api.slack.com/apps and install it to your workspace.'
|
|
}
|
|
],
|
|
channelToken: '',
|
|
configuringChannel: false,
|
|
channelConfigured: false,
|
|
|
|
// Step 5: Summary
|
|
setupSummary: {
|
|
provider: '',
|
|
agent: '',
|
|
channel: ''
|
|
},
|
|
|
|
// ── Lifecycle ──
|
|
|
|
async loadData() {
|
|
this.loading = true;
|
|
this.error = '';
|
|
try {
|
|
await this.loadProviders();
|
|
} catch(e) {
|
|
this.error = e.message || 'Could not load setup data.';
|
|
}
|
|
this.loading = false;
|
|
},
|
|
|
|
// ── Navigation ──
|
|
|
|
nextStep() {
|
|
if (this.step === 3 && !this.createdAgent) {
|
|
// Skip "Try It" if no agent was created
|
|
this.step = 5;
|
|
} else if (this.step < this.totalSteps) {
|
|
this.step++;
|
|
}
|
|
},
|
|
|
|
prevStep() {
|
|
if (this.step === 5 && !this.createdAgent) {
|
|
// Skip back past "Try It" if no agent was created
|
|
this.step = 3;
|
|
} else if (this.step > 1) {
|
|
this.step--;
|
|
}
|
|
},
|
|
|
|
goToStep(n) {
|
|
if (n >= 1 && n <= this.totalSteps) {
|
|
if (n === 4 && !this.createdAgent) return; // Can't go to Try It without agent
|
|
this.step = n;
|
|
}
|
|
},
|
|
|
|
stepLabel(n) {
|
|
var labels = ['Welcome', 'Provider', 'Agent', 'Try It', 'Channel', 'Done'];
|
|
return labels[n - 1] || '';
|
|
},
|
|
|
|
get canGoNext() {
|
|
if (this.step === 2) return this.keySaved || this.hasConfiguredProvider;
|
|
if (this.step === 3) return this.agentName.trim().length > 0;
|
|
return true;
|
|
},
|
|
|
|
get hasConfiguredProvider() {
|
|
var self = this;
|
|
return this.providers.some(function(p) {
|
|
return p.auth_status === 'configured';
|
|
});
|
|
},
|
|
|
|
// ── Step 2: Providers ──
|
|
|
|
async loadProviders() {
|
|
try {
|
|
var data = await OpenFangAPI.get('/api/providers');
|
|
this.providers = data.providers || [];
|
|
// Pre-select first unconfigured provider, or first one
|
|
var unconfigured = this.providers.filter(function(p) {
|
|
return p.auth_status !== 'configured' && p.api_key_env;
|
|
});
|
|
if (unconfigured.length > 0) {
|
|
this.selectedProvider = unconfigured[0].id;
|
|
} else if (this.providers.length > 0) {
|
|
this.selectedProvider = this.providers[0].id;
|
|
}
|
|
} catch(e) { this.providers = []; }
|
|
},
|
|
|
|
get selectedProviderObj() {
|
|
var self = this;
|
|
var match = this.providers.filter(function(p) { return p.id === self.selectedProvider; });
|
|
return match.length > 0 ? match[0] : null;
|
|
},
|
|
|
|
get popularProviders() {
|
|
var popular = ['anthropic', 'openai', 'gemini', 'groq', 'deepseek', 'openrouter'];
|
|
return this.providers.filter(function(p) {
|
|
return popular.indexOf(p.id) >= 0;
|
|
}).sort(function(a, b) {
|
|
return popular.indexOf(a.id) - popular.indexOf(b.id);
|
|
});
|
|
},
|
|
|
|
get otherProviders() {
|
|
var popular = ['anthropic', 'openai', 'gemini', 'groq', 'deepseek', 'openrouter'];
|
|
return this.providers.filter(function(p) {
|
|
return popular.indexOf(p.id) < 0;
|
|
});
|
|
},
|
|
|
|
selectProvider(id) {
|
|
this.selectedProvider = id;
|
|
this.apiKeyInput = '';
|
|
this.testResult = null;
|
|
this.keySaved = false;
|
|
},
|
|
|
|
providerHelp: function(id) {
|
|
var help = {
|
|
anthropic: { url: 'https://console.anthropic.com/settings/keys', text: 'Get your key from the Anthropic Console' },
|
|
openai: { url: 'https://platform.openai.com/api-keys', text: 'Get your key from the OpenAI Platform' },
|
|
gemini: { url: 'https://aistudio.google.com/apikey', text: 'Get your key from Google AI Studio' },
|
|
groq: { url: 'https://console.groq.com/keys', text: 'Get your key from the Groq Console (free tier available)' },
|
|
deepseek: { url: 'https://platform.deepseek.com/api_keys', text: 'Get your key from the DeepSeek Platform (very affordable)' },
|
|
openrouter: { url: 'https://openrouter.ai/keys', text: 'Get your key from OpenRouter (access 100+ models with one key)' },
|
|
mistral: { url: 'https://console.mistral.ai/api-keys', text: 'Get your key from the Mistral Console' },
|
|
together: { url: 'https://api.together.xyz/settings/api-keys', text: 'Get your key from Together AI' },
|
|
fireworks: { url: 'https://fireworks.ai/account/api-keys', text: 'Get your key from Fireworks AI' },
|
|
perplexity: { url: 'https://www.perplexity.ai/settings/api', text: 'Get your key from Perplexity Settings' },
|
|
cohere: { url: 'https://dashboard.cohere.com/api-keys', text: 'Get your key from the Cohere Dashboard' },
|
|
xai: { url: 'https://console.x.ai/', text: 'Get your key from the xAI Console' }
|
|
};
|
|
return help[id] || null;
|
|
},
|
|
|
|
providerIsConfigured(p) {
|
|
return p && p.auth_status === 'configured';
|
|
},
|
|
|
|
async saveKey() {
|
|
var provider = this.selectedProviderObj;
|
|
if (!provider) return;
|
|
var key = this.apiKeyInput.trim();
|
|
if (!key) {
|
|
OpenFangToast.error('Please enter an API key');
|
|
return;
|
|
}
|
|
this.savingKey = true;
|
|
try {
|
|
await OpenFangAPI.post('/api/providers/' + encodeURIComponent(provider.id) + '/key', { key: key });
|
|
this.apiKeyInput = '';
|
|
this.keySaved = true;
|
|
this.setupSummary.provider = provider.display_name;
|
|
OpenFangToast.success('API key saved for ' + provider.display_name);
|
|
await this.loadProviders();
|
|
// Auto-test after saving
|
|
await this.testKey();
|
|
} catch(e) {
|
|
OpenFangToast.error('Failed to save key: ' + e.message);
|
|
}
|
|
this.savingKey = false;
|
|
},
|
|
|
|
async testKey() {
|
|
var provider = this.selectedProviderObj;
|
|
if (!provider) return;
|
|
this.testingProvider = true;
|
|
this.testResult = null;
|
|
try {
|
|
var result = await OpenFangAPI.post('/api/providers/' + encodeURIComponent(provider.id) + '/test', {});
|
|
this.testResult = result;
|
|
if (result.status === 'ok') {
|
|
OpenFangToast.success(provider.display_name + ' connected (' + (result.latency_ms || '?') + 'ms)');
|
|
} else {
|
|
OpenFangToast.error(provider.display_name + ': ' + (result.error || 'Connection failed'));
|
|
}
|
|
} catch(e) {
|
|
this.testResult = { status: 'error', error: e.message };
|
|
OpenFangToast.error('Test failed: ' + e.message);
|
|
}
|
|
this.testingProvider = false;
|
|
},
|
|
|
|
// ── Step 3: Agent creation ──
|
|
|
|
selectTemplate(index) {
|
|
this.selectedTemplate = index;
|
|
var tpl = this.templates[index];
|
|
if (tpl) {
|
|
this.agentName = tpl.name.toLowerCase().replace(/\s+/g, '-');
|
|
}
|
|
},
|
|
|
|
async createAgent() {
|
|
var tpl = this.templates[this.selectedTemplate];
|
|
if (!tpl) return;
|
|
var name = this.agentName.trim();
|
|
if (!name) {
|
|
OpenFangToast.error('Please enter a name for your agent');
|
|
return;
|
|
}
|
|
|
|
// Use the provider the user just configured, or the template default
|
|
var provider = tpl.provider;
|
|
var model = tpl.model;
|
|
if (this.selectedProviderObj && this.providerIsConfigured(this.selectedProviderObj)) {
|
|
provider = this.selectedProviderObj.id;
|
|
// Use a sensible default model for the provider
|
|
model = this.defaultModelForProvider(provider) || tpl.model;
|
|
}
|
|
|
|
var toml = '[agent]\n';
|
|
toml += 'name = "' + name.replace(/"/g, '\\"') + '"\n';
|
|
toml += 'description = "' + tpl.description.replace(/"/g, '\\"') + '"\n';
|
|
toml += 'profile = "' + tpl.profile + '"\n\n';
|
|
toml += '[model]\nprovider = "' + provider + '"\n';
|
|
toml += 'name = "' + model + '"\n\n';
|
|
toml += '[prompt]\nsystem = """\n' + tpl.system_prompt + '\n"""\n';
|
|
|
|
this.creatingAgent = true;
|
|
try {
|
|
var res = await OpenFangAPI.post('/api/agents', { manifest_toml: toml });
|
|
if (res.agent_id) {
|
|
this.createdAgent = { id: res.agent_id, name: res.name || name };
|
|
this.setupSummary.agent = res.name || name;
|
|
OpenFangToast.success('Agent "' + (res.name || name) + '" created');
|
|
await Alpine.store('app').refreshAgents();
|
|
} else {
|
|
OpenFangToast.error('Failed: ' + (res.error || 'Unknown error'));
|
|
}
|
|
} catch(e) {
|
|
OpenFangToast.error('Failed to create agent: ' + e.message);
|
|
}
|
|
this.creatingAgent = false;
|
|
},
|
|
|
|
defaultModelForProvider(providerId) {
|
|
var defaults = {
|
|
anthropic: 'claude-sonnet-4-20250514',
|
|
openai: 'gpt-4o',
|
|
gemini: 'gemini-2.5-flash',
|
|
groq: 'llama-3.3-70b-versatile',
|
|
deepseek: 'deepseek-chat',
|
|
openrouter: 'openrouter/auto',
|
|
mistral: 'mistral-large-latest',
|
|
together: 'meta-llama/Llama-3-70b-chat-hf',
|
|
fireworks: 'accounts/fireworks/models/llama-v3p1-70b-instruct',
|
|
perplexity: 'llama-3.1-sonar-large-128k-online',
|
|
cohere: 'command-r-plus',
|
|
xai: 'grok-2'
|
|
};
|
|
return defaults[providerId] || '';
|
|
},
|
|
|
|
// ── Step 5: Channel setup ──
|
|
|
|
selectChannel(name) {
|
|
if (this.channelType === name) {
|
|
this.channelType = '';
|
|
this.channelToken = '';
|
|
} else {
|
|
this.channelType = name;
|
|
this.channelToken = '';
|
|
}
|
|
},
|
|
|
|
get selectedChannelObj() {
|
|
var self = this;
|
|
var match = this.channelOptions.filter(function(ch) { return ch.name === self.channelType; });
|
|
return match.length > 0 ? match[0] : null;
|
|
},
|
|
|
|
async configureChannel() {
|
|
var ch = this.selectedChannelObj;
|
|
if (!ch) return;
|
|
var token = this.channelToken.trim();
|
|
if (!token) {
|
|
OpenFangToast.error('Please enter the ' + ch.token_label);
|
|
return;
|
|
}
|
|
this.configuringChannel = true;
|
|
try {
|
|
var fields = {};
|
|
fields[ch.token_env.toLowerCase()] = token;
|
|
fields.token = token;
|
|
await OpenFangAPI.post('/api/channels/' + ch.name + '/configure', { fields: fields });
|
|
this.channelConfigured = true;
|
|
this.setupSummary.channel = ch.display_name;
|
|
OpenFangToast.success(ch.display_name + ' configured and activated.');
|
|
} catch(e) {
|
|
OpenFangToast.error('Failed: ' + (e.message || 'Unknown error'));
|
|
}
|
|
this.configuringChannel = false;
|
|
},
|
|
|
|
// ── Step 6: Finish ──
|
|
|
|
finish() {
|
|
localStorage.setItem('openfang-onboarded', 'true');
|
|
Alpine.store('app').showOnboarding = false;
|
|
// Navigate to agents with chat if an agent was created, otherwise overview
|
|
if (this.createdAgent) {
|
|
var agent = this.createdAgent;
|
|
Alpine.store('app').pendingAgent = { id: agent.id, name: agent.name, model_provider: '?', model_name: '?' };
|
|
window.location.hash = 'agents';
|
|
} else {
|
|
window.location.hash = 'overview';
|
|
}
|
|
},
|
|
|
|
finishAndDismiss() {
|
|
localStorage.setItem('openfang-onboarded', 'true');
|
|
Alpine.store('app').showOnboarding = false;
|
|
window.location.hash = 'overview';
|
|
}
|
|
};
|
|
}
|