Rank
70
AI Agents & MCPs & AI Workflow Automation โข (~400 MCP servers for AI agents) โข AI Automation / AI Agent with MCPs โข AI Workflows & AI Agents โข MCPs for AI Agents
Traction
No public download signal
Freshness
Updated 2d ago
Crawler Summary
Create, update, and manage Glance dashboard widgets. Use when user wants to: add something to their dashboard, create a widget, track data visually, show metrics/stats, display API data, or monitor usage. --- name: glance description: "Create, update, and manage Glance dashboard widgets. Use when user wants to: add something to their dashboard, create a widget, track data visually, show metrics/stats, display API data, or monitor usage." metadata: openclaw: emoji: "๐ฅ๏ธ" homepage: "https://github.com/acfranzen/glance" requires: env: ["GLANCE_URL"] bins: ["curl"] primaryEnv: GLANCE_URL --- Glance AI-extensible personal Published capability contract available. No trust telemetry is available yet. 3 GitHub stars reported by the source. Last updated 4/14/2026.
Freshness
Last checked 4/14/2026
Best For
Contract is available with explicit auth and schema references.
Not Ideal For
glance is not ideal for teams that need stronger public trust telemetry, lower setup complexity, or more explicit contract coverage before production rollout.
Evidence Sources Checked
editorial-content, capability-contract, runtime-metrics, public facts pack
Create, update, and manage Glance dashboard widgets. Use when user wants to: add something to their dashboard, create a widget, track data visually, show metrics/stats, display API data, or monitor usage. --- name: glance description: "Create, update, and manage Glance dashboard widgets. Use when user wants to: add something to their dashboard, create a widget, track data visually, show metrics/stats, display API data, or monitor usage." metadata: openclaw: emoji: "๐ฅ๏ธ" homepage: "https://github.com/acfranzen/glance" requires: env: ["GLANCE_URL"] bins: ["curl"] primaryEnv: GLANCE_URL --- Glance AI-extensible personal
Public facts
7
Change events
1
Artifacts
0
Freshness
Apr 14, 2026
Published capability contract available. No trust telemetry is available yet. 3 GitHub stars reported by the source. Last updated 4/14/2026.
Trust score
Unknown
Compatibility
OpenClaw
Freshness
Apr 14, 2026
Vendor
Acfranzen
Artifacts
0
Benchmarks
0
Last release
Unpublished
Key links, install path, and a quick operational read before the deeper crawl record.
Summary
Published capability contract available. No trust telemetry is available yet. 3 GitHub stars reported by the source. Last updated 4/14/2026.
Setup snapshot
git clone https://github.com/acfranzen/glance.gitSetup complexity is LOW. This package is likely designed for quick installation with minimal external side-effects.
Final validation: Expose the agent to a mock request payload inside a sandbox and trace the network egress before allowing access to real customer data.
Everything public we have scraped or crawled about this agent, grouped by evidence type with provenance.
Vendor
Acfranzen
Protocol compatibility
OpenClaw
Auth modes
api_key, oauth
Machine-readable schemas
OpenAPI or schema references published
Adoption signal
3 GitHub stars
Handshake status
UNKNOWN
Crawlable docs
6 indexed pages on the official domain
Merged public release, docs, artifact, benchmark, pricing, and trust refresh events.
Extracted files, examples, snippets, parameters, dependencies, permissions, and artifact metadata.
Extracted files
0
Examples
6
Snippets
0
Languages
typescript
Parameters
bash
# Navigate to skill directory (if installed via ClawHub)
cd "$(clawhub list | grep glance | awk '{print $2}')"
# Or clone directly
git clone https://github.com/acfranzen/glance ~/.glance
cd ~/.glance
# Install dependencies
npm install
# Configure environment
cp .env.example .env.local
# Edit .env.local with your settings
# Start development server
npm run dev
# Or build and start production
npm run build && npm startbash
# Server PORT=3333 AUTH_TOKEN=your-secret-token # Optional: Bearer token auth # OpenClaw Integration (for instant widget refresh) OPENCLAW_GATEWAY_URL=https://localhost:18789 OPENCLAW_TOKEN=your-gateway-token # Database DATABASE_PATH=./data/glance.db # SQLite database location
bash
# Create launchd plist
cat > ~/Library/LaunchAgents/com.glance.dashboard.plist << 'EOF'
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>Label</key>
<string>com.glance.dashboard</string>
<key>ProgramArguments</key>
<array>
<string>/opt/homebrew/bin/npm</string>
<string>run</string>
<string>dev</string>
</array>
<key>WorkingDirectory</key>
<string>~/.glance</string>
<key>RunAtLoad</key>
<true/>
<key>KeepAlive</key>
<true/>
<key>StandardOutPath</key>
<string>~/.glance/logs/stdout.log</string>
<key>StandardErrorPath</key>
<string>~/.glance/logs/stderr.log</string>
</dict>
</plist>
EOF
# Load service
mkdir -p ~/.glance/logs
launchctl load ~/Library/LaunchAgents/com.glance.dashboard.plist
# Service commands
launchctl start com.glance.dashboard
launchctl stop com.glance.dashboard
launchctl unload ~/Library/LaunchAgents/com.glance.dashboard.plistbash
curl -s -H "Origin: $GLANCE_URL" "$GLANCE_URL/api/widgets" | jq '.custom_widgets[].slug'
bash
curl -X POST "$GLANCE_URL/api/widgets/my-widget/cache" \
-H "Content-Type: application/json" \
-H "Origin: $GLANCE_URL" \
-d '{"data": {"value": 42, "fetchedAt": "'$(date -u +%Y-%m-%dT%H:%M:%SZ)'"}}'bash
# Check Glance is running (list widgets)
curl -s -H "Origin: $GLANCE_URL" "$GLANCE_URL/api/widgets" | jq '.custom_widgets[].slug'
# Auth note: Local requests with Origin header bypass Bearer token auth
# For external access, use: -H "Authorization: Bearer $GLANCE_TOKEN"
# Refresh a widget (look up instructions, collect data, POST to cache)
sqlite3 $GLANCE_DATA/glance.db "SELECT json_extract(fetch, '$.instructions') FROM custom_widgets WHERE slug = 'my-widget'"
# Follow the instructions, then:
curl -X POST "$GLANCE_URL/api/widgets/my-widget/cache" \
-H "Content-Type: application/json" \
-H "Origin: $GLANCE_URL" \
-d '{"data": {"value": 42, "fetchedAt": "'$(date -u +%Y-%m-%dT%H:%M:%SZ)'"}}'
# Verify in browser
browser action:open targetUrl:"$GLANCE_URL"Full documentation captured from public sources, including the complete README when available.
Docs source
GITHUB OPENCLEW
Editorial quality
ready
Create, update, and manage Glance dashboard widgets. Use when user wants to: add something to their dashboard, create a widget, track data visually, show metrics/stats, display API data, or monitor usage. --- name: glance description: "Create, update, and manage Glance dashboard widgets. Use when user wants to: add something to their dashboard, create a widget, track data visually, show metrics/stats, display API data, or monitor usage." metadata: openclaw: emoji: "๐ฅ๏ธ" homepage: "https://github.com/acfranzen/glance" requires: env: ["GLANCE_URL"] bins: ["curl"] primaryEnv: GLANCE_URL --- Glance AI-extensible personal
AI-extensible personal dashboard. Create custom widgets with natural language โ the AI handles data collection.
# Navigate to skill directory (if installed via ClawHub)
cd "$(clawhub list | grep glance | awk '{print $2}')"
# Or clone directly
git clone https://github.com/acfranzen/glance ~/.glance
cd ~/.glance
# Install dependencies
npm install
# Configure environment
cp .env.example .env.local
# Edit .env.local with your settings
# Start development server
npm run dev
# Or build and start production
npm run build && npm start
Dashboard runs at http://localhost:3333
Edit .env.local:
# Server
PORT=3333
AUTH_TOKEN=your-secret-token # Optional: Bearer token auth
# OpenClaw Integration (for instant widget refresh)
OPENCLAW_GATEWAY_URL=https://localhost:18789
OPENCLAW_TOKEN=your-gateway-token
# Database
DATABASE_PATH=./data/glance.db # SQLite database location
# Create launchd plist
cat > ~/Library/LaunchAgents/com.glance.dashboard.plist << 'EOF'
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>Label</key>
<string>com.glance.dashboard</string>
<key>ProgramArguments</key>
<array>
<string>/opt/homebrew/bin/npm</string>
<string>run</string>
<string>dev</string>
</array>
<key>WorkingDirectory</key>
<string>~/.glance</string>
<key>RunAtLoad</key>
<true/>
<key>KeepAlive</key>
<true/>
<key>StandardOutPath</key>
<string>~/.glance/logs/stdout.log</string>
<key>StandardErrorPath</key>
<string>~/.glance/logs/stderr.log</string>
</dict>
</plist>
EOF
# Load service
mkdir -p ~/.glance/logs
launchctl load ~/Library/LaunchAgents/com.glance.dashboard.plist
# Service commands
launchctl start com.glance.dashboard
launchctl stop com.glance.dashboard
launchctl unload ~/Library/LaunchAgents/com.glance.dashboard.plist
| Variable | Description | Default |
|----------|-------------|---------|
| PORT | Server port | 3333 |
| AUTH_TOKEN | Bearer token for API auth | โ |
| DATABASE_PATH | SQLite database path | ./data/glance.db |
| OPENCLAW_GATEWAY_URL | OpenClaw gateway for webhooks | โ |
| OPENCLAW_TOKEN | OpenClaw auth token | โ |
Create and manage dashboard widgets. Most widgets use agent_refresh โ you collect the data.
# Check Glance is running (list widgets)
curl -s -H "Origin: $GLANCE_URL" "$GLANCE_URL/api/widgets" | jq '.custom_widgets[].slug'
# Auth note: Local requests with Origin header bypass Bearer token auth
# For external access, use: -H "Authorization: Bearer $GLANCE_TOKEN"
# Refresh a widget (look up instructions, collect data, POST to cache)
sqlite3 $GLANCE_DATA/glance.db "SELECT json_extract(fetch, '$.instructions') FROM custom_widgets WHERE slug = 'my-widget'"
# Follow the instructions, then:
curl -X POST "$GLANCE_URL/api/widgets/my-widget/cache" \
-H "Content-Type: application/json" \
-H "Origin: $GLANCE_URL" \
-d '{"data": {"value": 42, "fetchedAt": "'$(date -u +%Y-%m-%dT%H:%M:%SZ)'"}}'
# Verify in browser
browser action:open targetUrl:"$GLANCE_URL"
When generating widget definitions, use the JSON Schema at docs/schemas/widget-schema.json with your AI model's structured output mode:
tool_use with the schemaresponse_format: { type: "json_schema", schema }The schema enforces all required fields at generation time โ malformed widgets cannot be produced.
Every widget MUST have these fields (the schema enforces them):
| Field | Type | Notes |
|-------|------|-------|
| name | string | Non-empty, human-readable |
| slug | string | Lowercase kebab-case (my-widget) |
| source_code | string | Valid JSX with Widget function |
| default_size | { w: 1-12, h: 1-20 } | Grid units |
| min_size | { w: 1-12, h: 1-20 } | Cannot resize smaller |
| fetch.type | enum | "server_code" | "webhook" | "agent_refresh" |
| fetch.instructions | string | REQUIRED if type is agent_refresh |
| fetch.schedule | string | REQUIRED if type is agent_refresh (cron) |
| data_schema.type | "object" | Always object |
| data_schema.properties | object | Define each field |
| data_schema.required | array | MUST include "fetchedAt" |
| credentials | array | Use [] if none needed |
{
"name": "My Widget",
"slug": "my-widget",
"source_code": "function Widget({ serverData }) { return <div>{serverData?.value}</div>; }",
"default_size": { "w": 2, "h": 2 },
"min_size": { "w": 1, "h": 1 },
"fetch": {
"type": "agent_refresh",
"schedule": "*/15 * * * *",
"instructions": "## Data Collection\nCollect the data...\n\n## Cache Update\nPOST to /api/widgets/my-widget/cache"
},
"data_schema": {
"type": "object",
"properties": {
"value": { "type": "number" },
"fetchedAt": { "type": "string", "format": "date-time" }
},
"required": ["value", "fetchedAt"]
},
"credentials": []
}
Every widget must complete ALL steps before being considered done:
โก Step 1: Create widget definition (POST /api/widgets)
- source_code with Widget function
- data_schema (REQUIRED for validation)
- fetch config (type + instructions for agent_refresh)
โก Step 2: Add to dashboard (POST /api/widgets/instances)
- custom_widget_id matches definition
- title and config set
โก Step 3: Populate cache (for agent_refresh widgets)
- Data matches data_schema exactly
- Includes fetchedAt timestamp
โก Step 4: Set up cron job (for agent_refresh widgets)
- Simple message: "โก WIDGET REFRESH: {slug}"
- Appropriate schedule (*/15 or */30 typically)
โก Step 5: BROWSER VERIFICATION (MANDATORY)
- Open http://localhost:3333
- Widget is visible on dashboard
- Shows actual data (not loading spinner)
- Data values match what was cached
- No errors or broken layouts
โ DO NOT report widget as complete until Step 5 passes!
docs/widget-sdk.md in the Glance repoWidget Package
โโโ meta (name, slug, description, author, version)
โโโ widget (source_code, default_size, min_size)
โโโ fetch (server_code | webhook | agent_refresh)
โโโ dataSchema? (JSON Schema for cached data - validates on POST)
โโโ cache (ttl, staleness, fallback)
โโโ credentials[] (API keys, local software requirements)
โโโ config_schema? (user options)
โโโ error? (retry, fallback, timeout)
Is data available via API that the widget can call?
โโโ YES โ Use server_code
โโโ NO โ Does an external service push data?
โโโ YES โ Use webhook
โโโ NO โ Use agent_refresh (YOU collect it)
| Scenario | Fetch Type | Who Collects Data? |
|----------|-----------|-------------------|
| Public/authenticated API | server_code | Widget calls API at render |
| External service pushes data | webhook | External service POSTs to cache |
| Local CLI tools | agent_refresh | YOU (the agent) via PTY/exec |
| Interactive terminals | agent_refresh | YOU (the agent) via PTY |
| Computed/aggregated data | agent_refresh | YOU (the agent) on a schedule |
โ ๏ธ agent_refresh means YOU are the data source. You set up a cron to remind yourself, then YOU collect the data using your tools (exec, PTY, browser, etc.) and POST it to the cache.
| Method | Endpoint | Description |
|--------|----------|-------------|
| POST | /api/widgets | Create widget definition |
| GET | /api/widgets | List all definitions |
| GET | /api/widgets/:slug | Get single definition |
| PATCH | /api/widgets/:slug | Update definition |
| DELETE | /api/widgets/:slug | Delete definition |
| Method | Endpoint | Description |
|--------|----------|-------------|
| POST | /api/widgets/instances | Add widget to dashboard |
| GET | /api/widgets/instances | List dashboard widgets |
| PATCH | /api/widgets/instances/:id | Update instance (config, position) |
| DELETE | /api/widgets/instances/:id | Remove from dashboard |
| Method | Endpoint | Description |
|--------|----------|-------------|
| GET | /api/credentials | List credentials + status |
| POST | /api/credentials | Store credential |
| DELETE | /api/credentials/:id | Delete credential |
{
"name": "GitHub PRs",
"slug": "github-prs",
"description": "Shows open pull requests",
"source_code": "function Widget({ serverData }) { ... }",
"default_size": { "w": 2, "h": 2 },
"min_size": { "w": 1, "h": 1 },
"refresh_interval": 300,
"credentials": [
{
"id": "github",
"type": "api_key",
"name": "GitHub Personal Access Token",
"description": "Token with repo scope",
"obtain_url": "https://github.com/settings/tokens"
}
],
"fetch": {
"type": "agent_refresh",
"schedule": "*/5 * * * *",
"instructions": "Fetch open PRs from GitHub API and POST to cache endpoint",
"expected_freshness_seconds": 300,
"max_staleness_seconds": 900
},
"cache": {
"ttl_seconds": 300,
"max_staleness_seconds": 900,
"storage": "sqlite",
"on_error": "use_stale"
},
"setup": {
"description": "Configure GitHub token",
"agent_skill": "Store GitHub PAT via /api/credentials",
"verification": {
"type": "cache_populated",
"target": "github-prs"
},
"idempotent": true
}
}
| Type | When to Use | Data Flow |
|------|-------------|-----------|
| server_code | Widget can call API directly | Widget โ server_code โ API |
| agent_refresh | Agent must fetch/compute data | Agent โ POST /cache โ Widget reads |
| webhook | External service pushes data | External โ POST /cache โ Widget reads |
Most widgets should use agent_refresh โ the agent fetches data on a schedule and pushes to the cache endpoint.
POST /api/widgets
Content-Type: application/json
{
"name": "GitHub PRs",
"slug": "github-prs",
"description": "Shows open pull requests",
"source_code": "function Widget({ serverData }) { ... }",
"default_size": { "w": 2, "h": 2 },
"credentials": [...],
"fetch": { "type": "agent_refresh", "schedule": "*/5 * * * *", ... },
"data_schema": {
"type": "object",
"properties": {
"prs": { "type": "array", "description": "List of PR objects" },
"fetchedAt": { "type": "string", "format": "date-time" }
},
"required": ["prs", "fetchedAt"]
},
"cache": { "ttl_seconds": 300, ... }
}
data_schema (REQUIRED) defines the data contract between the fetcher and the widget. Cache POSTs are validated against it โ malformed data returns 400.
โ ๏ธ Always include
data_schemawhen creating widgets. This ensures:
- Data validation on cache POSTs (400 on schema mismatch)
- Clear documentation of expected data structure
- AI agents know the exact format to produce
POST /api/widgets/instances
Content-Type: application/json
{
"type": "custom",
"title": "GitHub PRs",
"custom_widget_id": "cw_abc123",
"config": { "owner": "acfranzen", "repo": "libra" }
}
POST /api/widgets/github-prs/cache
Content-Type: application/json
{
"data": {
"prs": [...],
"fetchedAt": "2026-02-03T14:00:00Z"
}
}
โ ๏ธ If the widget has a dataSchema, the cache endpoint validates your data against it. Bad data returns 400 with details. Always check the widget's schema before POSTing:
GET /api/widgets/github-prs
# Response includes dataSchema showing required fields and types
โ ๏ธ MANDATORY: Every widget creation and refresh MUST end with browser verification.
Never consider a widget "done" until you've visually confirmed it renders correctly on the dashboard.
// REQUIRED: Open dashboard and verify widget renders
browser({
action: 'open',
targetUrl: 'http://localhost:3333',
profile: 'openclaw'
});
// Take a snapshot and check the widget
browser({ action: 'snapshot' });
// Look for:
// 1. Widget is visible on the dashboard
// 2. Shows actual data, NOT "Waiting for data..." or loading spinner
// 3. Data values match what was pushed to cache
// 4. No error messages displayed
// 5. Layout looks correct (not broken/overlapping)
Verification checklist (must ALL be true):
Common issues and fixes:
| Symptom | Cause | Fix |
|---------|-------|-----|
| "Waiting for data..." | Cache empty | POST data to /api/widgets/{slug}/cache |
| Widget not visible | Not added to dashboard | POST /api/widgets/instances |
| Wrong/old data | Slug mismatch | Check slug matches between definition and cache POST |
| Broken layout | Bad JSX in source_code | Check widget code for syntax errors |
| "No data" after POST | Schema validation failed | Check data matches data_schema |
If verification fails, fix the issue before reporting success.
For agent_refresh widgets, use serverData prop (NOT useData hook):
function Widget({ serverData }) {
const data = serverData;
const loading = !serverData;
const error = serverData?.error;
if (loading) return <Loading message="Waiting for data..." />;
if (error) return <ErrorDisplay message={error} />;
// NOTE: Do NOT wrap in <Card> - the framework wrapper (CustomWidgetWrapper)
// already provides the outer card with title, refresh button, and footer.
// Just render your content directly.
return (
<div className="space-y-3">
<List items={data.prs?.map(pr => ({
title: pr.title,
subtitle: `#${pr.number} by ${pr.author}`,
badge: pr.state
})) || []} />
</div>
);
}
Important: The widget wrapper (CustomWidgetWrapper) provides:
<Card> container with header (widget title)Your widget code should just render the content โ no Card, no CardHeader, no footer.
Key difference: agent_refresh widgets receive data via serverData prop, NOT by calling useData(). The agent pushes data to /api/widgets/{slug}/cache.
Prefer agent_refresh over server_code. Only use server_code when the widget MUST execute code at render time (rare).
// Only for fetch.type = "server_code" widgets
const token = await getCredential('github');
const response = await fetch('https://api.github.com/repos/owner/repo/pulls', {
headers: { 'Authorization': `Bearer ${token}` }
});
return await response.json();
Available: fetch, getCredential(provider), params, console
Blocked: require, eval, fs, process, global
โ ๏ธ CRITICAL: For agent_refresh widgets, YOU (the OpenClaw agent) are the data collector.
This is NOT an external API or service. YOU must:
โโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโ
โ Cron fires โ Agent wakes up โ Agent collects data โ โ
โ Agent POSTs to /cache โ Widget displays fresh data โ
โโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโ
fetch.type = "agent_refresh" and detailed fetch.instructionscron.add({
name: "Widget: My Data Refresh",
schedule: { kind: "cron", expr: "*/15 * * * *" },
payload: {
kind: "systemEvent",
text: "โก WIDGET REFRESH: my-widget" // Just the slug!
},
sessionTarget: "main" // Reminds YOU, not an isolated session
})
fetch.instructions from the DB and spawn a subagent:
// Parse slug from message
const slug = message.replace('โก WIDGET REFRESH:', '').trim();
// Query widget's fetch.instructions
const widget = db.query('SELECT fetch FROM custom_widgets WHERE slug = ?', slug);
// Spawn subagent with the instructions
sessions_spawn({ task: widget.fetch.instructions, model: 'haiku' });
exec for shell commandsclaude /status)browser for web scrapingweb_fetchPOST /api/widgets/{slug}/cache
Content-Type: application/json
{
"data": {
"myValue": 42,
"fetchedAt": "2026-02-03T18:30:00.000Z"
}
}
The fetch.instructions field is the single source of truth for how to collect widget data. Write them clearly so any subagent can follow them.
Required sections:
## Data Collection
Exact commands to run with full paths and flags.
Include PTY requirements if interactive.
## Data Transformation
Exact JSON structure expected.
Include field descriptions and examples.
## Cache Update
Full URL, required headers, body format.
## Browser Verification
Confirm the widget renders correctly.
Good example:
## Data Collection
```bash
gog gmail search "in:inbox" --json
Take first 5-8 emails, generate AI summary (3-5 words) for each:
{
"emails": [{"id": "...", "from": "...", "subject": "...", "summary": "AI summary here", "unread": true}],
"fetchedAt": "ISO timestamp"
}
POST to: http://localhost:3333/api/widgets/recent-emails/cache Header: Origin: http://localhost:3333 Body: { "data": <object above> }
Open http://localhost:3333 and confirm widget shows emails with AI summaries.
**Bad example (too vague):**
Get emails and post to cache.
### Real Example: Claude Max Usage Widget
This widget shows Claude CLI usage stats. The data comes from running `claude` in a PTY and navigating to `/status โ Usage`.
**The agent's job every 15 minutes:**
**This is YOUR responsibility as the agent.** The widget just displays whatever data is in the cache.
### Subagent Task Template for Refreshes
When spawning subagents for widget refreshes, always include browser verification:
```javascript
sessions_spawn({
task: `${fetchInstructions}
## REQUIRED: Browser Verification
After posting to cache, verify the widget renders correctly:
1. Open http://localhost:3333 in browser
2. Find the widget on the dashboard
3. Confirm it shows the data you just posted
4. Report any rendering issues
Do NOT report success until browser verification passes.`,
model: 'haiku',
label: `${slug}-refresh`
});
POST /api/widgets/{slug}/cache
Content-Type: application/json
{
"data": {
"packages": 142,
"fetchedAt": "2026-02-03T18:30:00.000Z"
}
}
For agent_refresh widgets, users can trigger immediate refreshes via the UI refresh button.
When configured with OPENCLAW_GATEWAY_URL and OPENCLAW_TOKEN environment variables, clicking the refresh button will:
/api/sessions/wakeThis eliminates the delay of waiting for the next heartbeat poll.
Environment variables (add to .env.local):
OPENCLAW_GATEWAY_URL=http://localhost:18789
OPENCLAW_TOKEN=your-gateway-token
How it works:
/api/widgets/{slug}/refreshโก WIDGET REFRESH: Refresh the "{slug}" widget now and POST to cacheResponse includes webhook status:
{
"status": "refresh_requested",
"webhook_sent": true,
"fallback_queued": true
}
If webhook fails or isn't configured, the DB fallback ensures the next heartbeat/poll will pick it up.
fetchedAt timestamp
## Credential Requirements Format
### Credential Types
| Type | Storage | Description | Use For |
|------|---------|-------------|---------|
| `api_key` | Glance DB (encrypted) | API tokens stored in Glance | GitHub PAT, OpenWeather key |
| `local_software` | Agent's machine | Software that must be installed | Homebrew, Docker |
| `agent` | Agent environment | Auth that lives on the agent | `gh` CLI auth, `gcloud` auth |
| `oauth` | Glance DB | OAuth tokens (future) | Google Calendar |
### Examples
```json
{
"credentials": [
{
"id": "github",
"type": "api_key",
"name": "GitHub Personal Access Token",
"description": "Token with repo scope",
"obtain_url": "https://github.com/settings/tokens",
"obtain_instructions": "Create token with 'repo' scope"
},
{
"id": "homebrew",
"type": "local_software",
"name": "Homebrew",
"check_command": "which brew",
"install_url": "https://brew.sh"
},
{
"id": "github_cli",
"type": "agent",
"name": "GitHub CLI",
"description": "Agent needs gh CLI authenticated to GitHub",
"agent_tool": "gh",
"agent_auth_check": "gh auth status",
"agent_auth_instructions": "Run `gh auth login` on the machine running OpenClaw"
}
]
}
When to use agent type: Use for agent_refresh widgets where the agent collects data using CLI tools that have their own auth (like gh, gcloud, aws). These credentials aren't stored in Glance โ they exist in the agent's environment.
| Provider | ID | Description |
|----------|-----|-------------|
| GitHub | github | GitHub API (PAT with repo scope) |
| Anthropic | anthropic | Claude API (Admin key for usage) |
| OpenAI | openai | GPT API (Admin key for usage) |
| OpenWeather | openweather | Weather data API |
| Linear | linear | Linear API |
| Notion | notion | Notion API |
GET /api/widgets/{slug}/export
Returns: { "package": "!GW1!eJxVj8EKwj..." }
POST /api/widgets/import
Content-Type: application/json
{
"package": "!GW1!eJxVj8EKwj...",
"dry_run": false,
"auto_add_to_dashboard": true
}
The !GW1! prefix indicates Glance Widget v1 format (compressed base64 JSON).
{
"valid": true,
"widget": { "id": "cw_abc", "slug": "homebrew-status" },
"cronSchedule": {
"expression": "*/15 * * * *",
"instructions": "Run brew list...",
"slug": "homebrew-status"
}
}
When cronSchedule is returned, OpenClaw should register a cron job.
| Component | Use For |
|-----------|---------|
| Card | Widget container (always use className="h-full") |
| List | Items with title/subtitle/badge |
| Stat | Single metric with trend indicator |
| Progress | Progress bars with variants |
| Badge | Status labels (success/warning/error) |
| Stack | Flexbox layout (row/column) |
| Grid | CSS Grid layout |
| Loading | Loading spinner |
| ErrorDisplay | Error with retry button |
See references/components.md for full props.
// Fetch data (BOTH args required!)
const { data, loading, error, refresh } = useData('github', {});
const { data } = useData('github', { endpoint: '/pulls', params: { state: 'open' } });
// Get widget config
const config = useConfig();
// Widget-local state
const { state, setState } = useWidgetState('counter', 0);
โ ๏ธ useData requires both arguments. Pass empty {} if no query needed.
if (error?.code === 'CREDENTIAL_MISSING') {
return <Card><CardContent>
<Icons.Lock className="h-8 w-8" />
<p>GitHub token required</p>
</CardContent></Card>;
}
Error codes: CREDENTIAL_MISSING, RATE_LIMITED, NETWORK_ERROR, API_ERROR
github-prs-libra not widget-1To summarize dashboard for user:
1. GET /api/widgets/instances โ list instances
2. For each: POST /api/widgets/:slug/execute
3. Combine into natural language summary
docs/schemas/widget-schema.json enforces all required fields[] if no credentials neededDELETE /api/widgets/{slug}/refresh after handling| Variable | Description | Example |
|----------|-------------|---------|
| GLANCE_URL | Glance server URL | http://localhost:3333 |
| GLANCE_DATA | Path to SQLite database | /tmp/glance-test/data |
| OPENCLAW_GATEWAY_URL | For webhook refresh notifications | https://localhost:18789 |
| OPENCLAW_TOKEN | Gateway auth token | d551fe97... |
โก WIDGET REFRESH: {slug}, agent looks up instructionsgog calendar doesn't work for iCloud, use /opt/homebrew/bin/icalBuddywttr.in/City?format=j1Machine endpoints, protocol fit, contract coverage, invocation examples, and guardrails for agent-to-agent use.
Contract coverage
Status
ready
Auth
api_key, oauth
Streaming
No
Data region
global
Protocol support
Requires: openclew, lang:typescript
Forbidden: none
Guardrails
Operational confidence: medium
curl -s "https://xpersona.co/api/v1/agents/acfranzen-glance/snapshot"
curl -s "https://xpersona.co/api/v1/agents/acfranzen-glance/contract"
curl -s "https://xpersona.co/api/v1/agents/acfranzen-glance/trust"
Trust and runtime signals, benchmark suites, failure patterns, and practical risk constraints.
Trust signals
Handshake
UNKNOWN
Confidence
unknown
Attempts 30d
unknown
Fallback rate
unknown
Runtime metrics
Observed P50
unknown
Observed P95
unknown
Rate limit
unknown
Estimated cost
unknown
Every public screenshot, visual asset, demo link, and owner-provided destination tied to this agent.
Neighboring agents from the same protocol and source ecosystem for comparison and shortlist building.
Rank
70
AI Agents & MCPs & AI Workflow Automation โข (~400 MCP servers for AI agents) โข AI Automation / AI Agent with MCPs โข AI Workflows & AI Agents โข MCPs for AI Agents
Traction
No public download signal
Freshness
Updated 2d ago
Rank
70
AI productivity studio with smart chat, autonomous agents, and 300+ assistants. Unified access to frontier LLMs
Traction
No public download signal
Freshness
Updated 5d ago
Rank
70
Free, local, open-source 24/7 Cowork app and OpenClaw for Gemini CLI, Claude Code, Codex, OpenCode, Qwen Code, Goose CLI, Auggie, and more | ๐ Star if you like it!
Traction
No public download signal
Freshness
Updated 6d ago
Rank
70
The Frontend for Agents & Generative UI. React + Angular
Traction
No public download signal
Freshness
Updated 23d ago
Contract JSON
{
"contractStatus": "ready",
"authModes": [
"api_key",
"oauth"
],
"requires": [
"openclew",
"lang:typescript"
],
"forbidden": [],
"supportsMcp": false,
"supportsA2a": false,
"supportsStreaming": false,
"inputSchemaRef": "https://github.com/acfranzen/glance#input",
"outputSchemaRef": "https://github.com/acfranzen/glance#output",
"dataRegion": "global",
"contractUpdatedAt": "2026-02-24T19:44:19.992Z",
"sourceUpdatedAt": "2026-02-24T19:44:19.992Z",
"freshnessSeconds": 4420142
}Invocation Guide
{
"preferredApi": {
"snapshotUrl": "https://xpersona.co/api/v1/agents/acfranzen-glance/snapshot",
"contractUrl": "https://xpersona.co/api/v1/agents/acfranzen-glance/contract",
"trustUrl": "https://xpersona.co/api/v1/agents/acfranzen-glance/trust"
},
"curlExamples": [
"curl -s \"https://xpersona.co/api/v1/agents/acfranzen-glance/snapshot\"",
"curl -s \"https://xpersona.co/api/v1/agents/acfranzen-glance/contract\"",
"curl -s \"https://xpersona.co/api/v1/agents/acfranzen-glance/trust\""
],
"jsonRequestTemplate": {
"query": "summarize this repo",
"constraints": {
"maxLatencyMs": 2000,
"protocolPreference": [
"OPENCLEW"
]
}
},
"jsonResponseTemplate": {
"ok": true,
"result": {
"summary": "...",
"confidence": 0.9
},
"meta": {
"source": "GITHUB_OPENCLEW",
"generatedAt": "2026-04-16T23:33:22.476Z"
}
},
"retryPolicy": {
"maxAttempts": 3,
"backoffMs": [
500,
1500,
3500
],
"retryableConditions": [
"HTTP_429",
"HTTP_503",
"NETWORK_TIMEOUT"
]
}
}Trust JSON
{
"status": "unavailable",
"handshakeStatus": "UNKNOWN",
"verificationFreshnessHours": null,
"reputationScore": null,
"p95LatencyMs": null,
"successRate30d": null,
"fallbackRate": null,
"attempts30d": null,
"trustUpdatedAt": null,
"trustConfidence": "unknown",
"sourceUpdatedAt": null,
"freshnessSeconds": null
}Capability Matrix
{
"rows": [
{
"key": "OPENCLEW",
"type": "protocol",
"support": "unknown",
"confidenceSource": "profile",
"notes": "Listed on profile"
},
{
"key": "call",
"type": "capability",
"support": "supported",
"confidenceSource": "profile",
"notes": "Declared in agent profile metadata"
},
{
"key": "display",
"type": "capability",
"support": "supported",
"confidenceSource": "profile",
"notes": "Declared in agent profile metadata"
},
{
"key": "follow",
"type": "capability",
"support": "supported",
"confidenceSource": "profile",
"notes": "Declared in agent profile metadata"
},
{
"key": "trigger",
"type": "capability",
"support": "supported",
"confidenceSource": "profile",
"notes": "Declared in agent profile metadata"
}
],
"flattenedTokens": "protocol:OPENCLEW|unknown|profile capability:call|supported|profile capability:display|supported|profile capability:follow|supported|profile capability:trigger|supported|profile"
}Facts JSON
[
{
"factKey": "docs_crawl",
"category": "integration",
"label": "Crawlable docs",
"value": "6 indexed pages on the official domain",
"href": "https://github.com/login?return_to=https%3A%2F%2Fgithub.com%2Fopenclaw%2Fskills%2Ftree%2Fmain%2Fskills%2Fasleep123%2Fcaldav-calendar",
"sourceUrl": "https://github.com/login?return_to=https%3A%2F%2Fgithub.com%2Fopenclaw%2Fskills%2Ftree%2Fmain%2Fskills%2Fasleep123%2Fcaldav-calendar",
"sourceType": "search_document",
"confidence": "medium",
"observedAt": "2026-04-15T05:03:46.393Z",
"isPublic": true
},
{
"factKey": "vendor",
"category": "vendor",
"label": "Vendor",
"value": "Acfranzen",
"href": "https://github.com/acfranzen/glance",
"sourceUrl": "https://github.com/acfranzen/glance",
"sourceType": "profile",
"confidence": "medium",
"observedAt": "2026-04-14T22:24:02.864Z",
"isPublic": true
},
{
"factKey": "traction",
"category": "adoption",
"label": "Adoption signal",
"value": "3 GitHub stars",
"href": "https://github.com/acfranzen/glance",
"sourceUrl": "https://github.com/acfranzen/glance",
"sourceType": "profile",
"confidence": "medium",
"observedAt": "2026-04-14T22:24:02.864Z",
"isPublic": true
},
{
"factKey": "protocols",
"category": "compatibility",
"label": "Protocol compatibility",
"value": "OpenClaw",
"href": "https://xpersona.co/api/v1/agents/acfranzen-glance/contract",
"sourceUrl": "https://xpersona.co/api/v1/agents/acfranzen-glance/contract",
"sourceType": "contract",
"confidence": "medium",
"observedAt": "2026-02-24T19:44:19.992Z",
"isPublic": true
},
{
"factKey": "auth_modes",
"category": "compatibility",
"label": "Auth modes",
"value": "api_key, oauth",
"href": "https://xpersona.co/api/v1/agents/acfranzen-glance/contract",
"sourceUrl": "https://xpersona.co/api/v1/agents/acfranzen-glance/contract",
"sourceType": "contract",
"confidence": "high",
"observedAt": "2026-02-24T19:44:19.992Z",
"isPublic": true
},
{
"factKey": "schema_refs",
"category": "artifact",
"label": "Machine-readable schemas",
"value": "OpenAPI or schema references published",
"href": "https://github.com/acfranzen/glance#input",
"sourceUrl": "https://xpersona.co/api/v1/agents/acfranzen-glance/contract",
"sourceType": "contract",
"confidence": "high",
"observedAt": "2026-02-24T19:44:19.992Z",
"isPublic": true
},
{
"factKey": "handshake_status",
"category": "security",
"label": "Handshake status",
"value": "UNKNOWN",
"href": "https://xpersona.co/api/v1/agents/acfranzen-glance/trust",
"sourceUrl": "https://xpersona.co/api/v1/agents/acfranzen-glance/trust",
"sourceType": "trust",
"confidence": "medium",
"observedAt": null,
"isPublic": true
}
]Change Events JSON
[
{
"eventType": "docs_update",
"title": "Docs refreshed: Sign in to GitHub ยท GitHub",
"description": "Fresh crawlable documentation was indexed for the official domain.",
"href": "https://github.com/login?return_to=https%3A%2F%2Fgithub.com%2Fopenclaw%2Fskills%2Ftree%2Fmain%2Fskills%2Fasleep123%2Fcaldav-calendar",
"sourceUrl": "https://github.com/login?return_to=https%3A%2F%2Fgithub.com%2Fopenclaw%2Fskills%2Ftree%2Fmain%2Fskills%2Fasleep123%2Fcaldav-calendar",
"sourceType": "search_document",
"confidence": "medium",
"observedAt": "2026-04-15T05:03:46.393Z",
"isPublic": true
}
]Sponsored
Ads related to glance and adjacent AI workflows.