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
300 lines
11 KiB
JavaScript
300 lines
11 KiB
JavaScript
// OpenFang Skills Page — OpenClaw/ClawHub ecosystem + local skills + MCP servers
|
|
'use strict';
|
|
|
|
function skillsPage() {
|
|
return {
|
|
tab: 'installed',
|
|
skills: [],
|
|
loading: true,
|
|
loadError: '',
|
|
|
|
// ClawHub state
|
|
clawhubSearch: '',
|
|
clawhubResults: [],
|
|
clawhubBrowseResults: [],
|
|
clawhubLoading: false,
|
|
clawhubError: '',
|
|
clawhubSort: 'trending',
|
|
clawhubNextCursor: null,
|
|
installingSlug: null,
|
|
installResult: null,
|
|
_searchTimer: null,
|
|
|
|
// Skill detail modal
|
|
skillDetail: null,
|
|
detailLoading: false,
|
|
|
|
// MCP servers
|
|
mcpServers: [],
|
|
mcpLoading: false,
|
|
|
|
// Category definitions from the OpenClaw ecosystem
|
|
categories: [
|
|
{ id: 'coding', name: 'Coding & IDEs' },
|
|
{ id: 'git', name: 'Git & GitHub' },
|
|
{ id: 'web', name: 'Web & Frontend' },
|
|
{ id: 'devops', name: 'DevOps & Cloud' },
|
|
{ id: 'browser', name: 'Browser & Automation' },
|
|
{ id: 'search', name: 'Search & Research' },
|
|
{ id: 'ai', name: 'AI & LLMs' },
|
|
{ id: 'data', name: 'Data & Analytics' },
|
|
{ id: 'productivity', name: 'Productivity' },
|
|
{ id: 'communication', name: 'Communication' },
|
|
{ id: 'media', name: 'Media & Streaming' },
|
|
{ id: 'notes', name: 'Notes & PKM' },
|
|
{ id: 'security', name: 'Security' },
|
|
{ id: 'cli', name: 'CLI Utilities' },
|
|
{ id: 'marketing', name: 'Marketing & Sales' },
|
|
{ id: 'finance', name: 'Finance' },
|
|
{ id: 'smart-home', name: 'Smart Home & IoT' },
|
|
{ id: 'docs', name: 'PDF & Documents' },
|
|
],
|
|
|
|
runtimeBadge: function(rt) {
|
|
var r = (rt || '').toLowerCase();
|
|
if (r === 'python' || r === 'py') return { text: 'PY', cls: 'runtime-badge-py' };
|
|
if (r === 'node' || r === 'nodejs' || r === 'js' || r === 'javascript') return { text: 'JS', cls: 'runtime-badge-js' };
|
|
if (r === 'wasm' || r === 'webassembly') return { text: 'WASM', cls: 'runtime-badge-wasm' };
|
|
if (r === 'prompt_only' || r === 'prompt' || r === 'promptonly') return { text: 'PROMPT', cls: 'runtime-badge-prompt' };
|
|
return { text: r.toUpperCase().substring(0, 4), cls: 'runtime-badge-prompt' };
|
|
},
|
|
|
|
sourceBadge: function(source) {
|
|
if (!source) return { text: 'Local', cls: 'badge-dim' };
|
|
switch (source.type) {
|
|
case 'clawhub': return { text: 'ClawHub', cls: 'badge-info' };
|
|
case 'openclaw': return { text: 'OpenClaw', cls: 'badge-info' };
|
|
case 'bundled': return { text: 'Built-in', cls: 'badge-success' };
|
|
default: return { text: 'Local', cls: 'badge-dim' };
|
|
}
|
|
},
|
|
|
|
formatDownloads: function(n) {
|
|
if (!n) return '0';
|
|
if (n >= 1000000) return (n / 1000000).toFixed(1) + 'M';
|
|
if (n >= 1000) return (n / 1000).toFixed(1) + 'K';
|
|
return n.toString();
|
|
},
|
|
|
|
async loadSkills() {
|
|
this.loading = true;
|
|
this.loadError = '';
|
|
try {
|
|
var data = await OpenFangAPI.get('/api/skills');
|
|
this.skills = (data.skills || []).map(function(s) {
|
|
return {
|
|
name: s.name,
|
|
description: s.description || '',
|
|
version: s.version || '',
|
|
author: s.author || '',
|
|
runtime: s.runtime || 'unknown',
|
|
tools_count: s.tools_count || 0,
|
|
tags: s.tags || [],
|
|
enabled: s.enabled !== false,
|
|
source: s.source || { type: 'local' },
|
|
has_prompt_context: !!s.has_prompt_context
|
|
};
|
|
});
|
|
} catch(e) {
|
|
this.skills = [];
|
|
this.loadError = e.message || 'Could not load skills.';
|
|
}
|
|
this.loading = false;
|
|
},
|
|
|
|
async loadData() {
|
|
await this.loadSkills();
|
|
},
|
|
|
|
// Debounced search — fires 350ms after user stops typing
|
|
onSearchInput() {
|
|
if (this._searchTimer) clearTimeout(this._searchTimer);
|
|
var q = this.clawhubSearch.trim();
|
|
if (!q) {
|
|
this.clawhubResults = [];
|
|
this.clawhubError = '';
|
|
return;
|
|
}
|
|
var self = this;
|
|
this._searchTimer = setTimeout(function() { self.searchClawHub(); }, 350);
|
|
},
|
|
|
|
// ClawHub search
|
|
async searchClawHub() {
|
|
if (!this.clawhubSearch.trim()) {
|
|
this.clawhubResults = [];
|
|
return;
|
|
}
|
|
this.clawhubLoading = true;
|
|
this.clawhubError = '';
|
|
try {
|
|
var data = await OpenFangAPI.get('/api/clawhub/search?q=' + encodeURIComponent(this.clawhubSearch.trim()) + '&limit=20');
|
|
this.clawhubResults = data.items || [];
|
|
if (data.error) this.clawhubError = data.error;
|
|
} catch(e) {
|
|
this.clawhubResults = [];
|
|
this.clawhubError = e.message || 'Search failed';
|
|
}
|
|
this.clawhubLoading = false;
|
|
},
|
|
|
|
// Clear search and go back to browse
|
|
clearSearch() {
|
|
this.clawhubSearch = '';
|
|
this.clawhubResults = [];
|
|
this.clawhubError = '';
|
|
if (this._searchTimer) clearTimeout(this._searchTimer);
|
|
},
|
|
|
|
// ClawHub browse by sort
|
|
async browseClawHub(sort) {
|
|
this.clawhubSort = sort || 'trending';
|
|
this.clawhubLoading = true;
|
|
this.clawhubError = '';
|
|
this.clawhubNextCursor = null;
|
|
try {
|
|
var data = await OpenFangAPI.get('/api/clawhub/browse?sort=' + this.clawhubSort + '&limit=20');
|
|
this.clawhubBrowseResults = data.items || [];
|
|
this.clawhubNextCursor = data.next_cursor || null;
|
|
if (data.error) this.clawhubError = data.error;
|
|
} catch(e) {
|
|
this.clawhubBrowseResults = [];
|
|
this.clawhubError = e.message || 'Browse failed';
|
|
}
|
|
this.clawhubLoading = false;
|
|
},
|
|
|
|
// ClawHub load more results
|
|
async loadMoreClawHub() {
|
|
if (!this.clawhubNextCursor || this.clawhubLoading) return;
|
|
this.clawhubLoading = true;
|
|
try {
|
|
var data = await OpenFangAPI.get('/api/clawhub/browse?sort=' + this.clawhubSort + '&limit=20&cursor=' + encodeURIComponent(this.clawhubNextCursor));
|
|
this.clawhubBrowseResults = this.clawhubBrowseResults.concat(data.items || []);
|
|
this.clawhubNextCursor = data.next_cursor || null;
|
|
} catch(e) {
|
|
// silently fail on load more
|
|
}
|
|
this.clawhubLoading = false;
|
|
},
|
|
|
|
// Show skill detail
|
|
async showSkillDetail(slug) {
|
|
this.detailLoading = true;
|
|
this.skillDetail = null;
|
|
this.installResult = null;
|
|
try {
|
|
var data = await OpenFangAPI.get('/api/clawhub/skill/' + encodeURIComponent(slug));
|
|
this.skillDetail = data;
|
|
} catch(e) {
|
|
OpenFangToast.error('Failed to load skill details');
|
|
}
|
|
this.detailLoading = false;
|
|
},
|
|
|
|
closeDetail() {
|
|
this.skillDetail = null;
|
|
this.installResult = null;
|
|
},
|
|
|
|
// Install from ClawHub
|
|
async installFromClawHub(slug) {
|
|
this.installingSlug = slug;
|
|
this.installResult = null;
|
|
try {
|
|
var data = await OpenFangAPI.post('/api/clawhub/install', { slug: slug });
|
|
this.installResult = data;
|
|
if (data.warnings && data.warnings.length > 0) {
|
|
OpenFangToast.success('Skill "' + data.name + '" installed with ' + data.warnings.length + ' warning(s)');
|
|
} else {
|
|
OpenFangToast.success('Skill "' + data.name + '" installed successfully');
|
|
}
|
|
// Update installed state in detail modal if open
|
|
if (this.skillDetail && this.skillDetail.slug === slug) {
|
|
this.skillDetail.installed = true;
|
|
}
|
|
await this.loadSkills();
|
|
} catch(e) {
|
|
var msg = e.message || 'Install failed';
|
|
if (msg.includes('already_installed')) {
|
|
OpenFangToast.error('Skill is already installed');
|
|
} else if (msg.includes('SecurityBlocked')) {
|
|
OpenFangToast.error('Skill blocked by security scan');
|
|
} else {
|
|
OpenFangToast.error('Install failed: ' + msg);
|
|
}
|
|
}
|
|
this.installingSlug = null;
|
|
},
|
|
|
|
// Uninstall
|
|
uninstallSkill: function(name) {
|
|
var self = this;
|
|
OpenFangToast.confirm('Uninstall Skill', 'Uninstall skill "' + name + '"? This cannot be undone.', async function() {
|
|
try {
|
|
await OpenFangAPI.post('/api/skills/uninstall', { name: name });
|
|
OpenFangToast.success('Skill "' + name + '" uninstalled');
|
|
await self.loadSkills();
|
|
} catch(e) {
|
|
OpenFangToast.error('Failed to uninstall skill: ' + e.message);
|
|
}
|
|
});
|
|
},
|
|
|
|
// Create prompt-only skill
|
|
async createDemoSkill(skill) {
|
|
try {
|
|
await OpenFangAPI.post('/api/skills/create', {
|
|
name: skill.name,
|
|
description: skill.description,
|
|
runtime: 'prompt_only',
|
|
prompt_context: skill.prompt_context || skill.description
|
|
});
|
|
OpenFangToast.success('Skill "' + skill.name + '" created');
|
|
this.tab = 'installed';
|
|
await this.loadSkills();
|
|
} catch(e) {
|
|
OpenFangToast.error('Failed to create skill: ' + e.message);
|
|
}
|
|
},
|
|
|
|
// Load MCP servers
|
|
async loadMcpServers() {
|
|
this.mcpLoading = true;
|
|
try {
|
|
var data = await OpenFangAPI.get('/api/mcp/servers');
|
|
this.mcpServers = data;
|
|
} catch(e) {
|
|
this.mcpServers = { configured: [], connected: [], total_configured: 0, total_connected: 0 };
|
|
}
|
|
this.mcpLoading = false;
|
|
},
|
|
|
|
// Category search on ClawHub
|
|
searchCategory: function(cat) {
|
|
this.clawhubSearch = cat.name;
|
|
this.searchClawHub();
|
|
},
|
|
|
|
// Quick start skills (prompt-only, zero deps)
|
|
quickStartSkills: [
|
|
{ name: 'code-review-guide', description: 'Adds code review best practices and checklist to agent context.', prompt_context: 'You are an expert code reviewer. When reviewing code:\n1. Check for bugs and logic errors\n2. Evaluate code style and readability\n3. Look for security vulnerabilities\n4. Suggest performance improvements\n5. Verify error handling\n6. Check test coverage' },
|
|
{ name: 'writing-style', description: 'Configurable writing style guide for content generation.', prompt_context: 'Follow these writing guidelines:\n- Use clear, concise language\n- Prefer active voice over passive voice\n- Keep paragraphs short (3-4 sentences)\n- Use bullet points for lists\n- Maintain consistent tone throughout' },
|
|
{ name: 'api-design', description: 'REST API design patterns and conventions.', prompt_context: 'When designing REST APIs:\n- Use nouns for resources, not verbs\n- Use HTTP methods correctly (GET, POST, PUT, DELETE)\n- Return appropriate status codes\n- Use pagination for list endpoints\n- Version your API\n- Document all endpoints' },
|
|
{ name: 'security-checklist', description: 'OWASP-aligned security review checklist.', prompt_context: 'Security review checklist (OWASP aligned):\n- Input validation on all user inputs\n- Output encoding to prevent XSS\n- Parameterized queries to prevent SQL injection\n- Authentication and session management\n- Access control checks\n- CSRF protection\n- Security headers\n- Error handling without information leakage' },
|
|
],
|
|
|
|
// Check if skill is installed by slug
|
|
isSkillInstalled: function(slug) {
|
|
return this.skills.some(function(s) {
|
|
return s.source && s.source.type === 'clawhub' && s.source.slug === slug;
|
|
});
|
|
},
|
|
|
|
// Check if skill is installed by name
|
|
isSkillInstalledByName: function(name) {
|
|
return this.skills.some(function(s) { return s.name === name; });
|
|
},
|
|
};
|
|
}
|