* System plugin tier with mark/unmark fanout (schema v39)
Adds a mandatory plugin tier so admins can pin a small set of curated
plugins into every user's stack from day one. Marking a plugin via the
new toggle on /admin/marketplaces materializes resource_grants for every
group and user_plugin_optouts subscriptions for every user, so the
existing resolver pulls the plugin into every served set without a new
filter layer. Hooks on user-create (Google OAuth, magic-link, admin
POST, scheduler) and group-create propagate the same materialization to
new principals. UI locks: /admin/access disables the checkbox with a
SYSTEM pill; /marketplace cards swap the "In stack" green pill for an
amber "Required" badge with shield icon; the plugin detail install
button reads "Required by your org"; /my-ai-stack toggle is disabled.
Bypass paths return 409 (DELETE /api/admin/grants for system grants,
PUT /api/my-stack/curated/.../{enabled:false}, DELETE
/api/marketplace/curated/.../install). Unmark only flips the flag —
materialized rows persist so admins curate cleanup at their leisure
through the now-unlocked /admin/access checkboxes.
* Marketplace UX polish + drop legacy /store and /my-ai-stack pages
Two-part cleanup post-v39:
(1) Page deletion. /store and /my-ai-stack were already replaced by
/marketplace?tab=flea and /marketplace?tab=my respectively, but the
standalone routes lingered. Hard delete in dev mode — no redirects,
stale bookmarks 404. The /store/new upload wizard, the flea
detail/edit pages, the admin queue, and all /api/store/* +
/api/my-stack endpoints (CLI consumers) stay. Internal hardcoded
hrefs in the upload wizard's Cancel button and the advanced-setup
page repointed to the marketplace tabs.
(2) Detail-page install button rework. The single button that morphed
between "+ Add to my stack" and "✓ In your stack" did not
communicate uninstall affordance. The installed state now renders an
inline white status label *before* a separate red-bordered
"✕ Remove from stack" button on the same row, both at identical
height to avoid layout shift. System plugins keep their locked amber
"✓ Required by your org" pill (no Remove button — API refuses 409).
The post-action hint panel now fires on remove too with the title
flipped to "✓ Removed from your stack" — Claude Code needs the same
/update-agnes-plugins refresh either way.
Also: /admin/marketplaces Details modal "Mark as system" toggle
redesigned. The button was near-invisible (matched neutral row
metadata). It's now a balanced amber-toned chip with shield icon
and a structured confirm modal replacing the native confirm() dialog
that summarizes fanout consequences before commit.
* Move stack-hint inside hero with glass-on-gradient styling
The post-action hint card ("✓ Added to your stack" with the
/update-agnes-plugins recipe) used to live below the hero in
panel-what (gray card on white page body). Clicking add/remove
inserted/removed it between the hero and content, shifting the
panels below — a noticeable scroll jump.
The hint is now anchored inside the hero's top-right corner alongside
the install/remove buttons, both as flex children of an absolutely
positioned .actions container. The card uses a translucent
white-on-glass treatment that adopts the hero's kind color (blue for
plugin, green for skill, purple for agent) without per-kind branching.
Hero is always tall enough (160px photo) to contain the action+hint
stack without overflow, so toggling the hint visibility doesn't grow
the hero or shift body content.
The hero-head grid reserves a third 300px column for the absolute
actions overlay so meta gets the proper 1fr free space instead of
being squeezed by a padding-right hack. Responsive breakpoint at
1100px reflows the actions stack below hero-head when the viewport
isn't wide enough to keep meta + actions side-by-side comfortably.
* Add optional -DataPath bind mount to run-local-dev.ps1
When the operator wants to inspect DuckDB files (system.duckdb, extracts,
marketplaces, store/, …) directly from Windows Explorer, the named volume
inside the Docker Desktop WSL VM isn't reachable. The new -DataPath param
generates a transient compose override that rebinds /data on app, scheduler,
extract (and Caddy's /srv:ro mirror) to a Windows host folder.
Fully additive — when -DataPath is omitted everything behaves exactly as
before: no override file is generated, $composeFiles array is unchanged,
finally cleanup is a no-op. Existing positional invocations
(.\run-local-dev.ps1 up | down | logs) keep binding to $Action because
$DataPath is a named-only parameter with no Position attribute.
The override is written via [System.IO.File]::WriteAllText so the YAML is
BOM-less across PS 5.1 / 7+ — Compose rejects BOM-prefixed YAML on Windows.
The override file is unique per PID and removed in the script's finally
block so concurrent invocations and crashes don't leak files.
* factor mark_system fanout into UserCuratedSubscriptionsRepository
The endpoint imported UserCuratedSubscriptionsRepository, ignored it
(noqa: F841), then duplicated the user-side fanout SQL inline. Adds
fanout_system_for_plugin() symmetric to the existing
fanout_system_for_user() and routes mark_plugin_system through it —
removes the dead import + 14 lines of inline SQL, returns the same
`affected_users` delta count, no behavior change.
* drop customer-specific path from .ps1 example
Per CLAUDE.md vendor-agnostic OSS rule: replaced
C:\\Business\\Groupon\\Agnes\\agnes-data with the generic
C:\\Users\\<you>\\agnes-data placeholder so the docstring
example reads cleanly on any reviewer's box.
* release: 0.48.0 + parallelize Release-workflow pytest
Cuts the release shipped via #228 #230 #231 #232 #233 #234 #236 #237 #238
#239 #240 plus this PR (#241). Major changes:
- System plugin tier (schema v39) — admins mark a plugin mandatory; fans
out RBAC grants + subscriptions to every existing user/group plus
hooks for new principals
- BREAKING: removed standalone /store + /my-ai-stack page routes
(replaced by /marketplace?tab=flea + /marketplace?tab=my)
- Setup-prompt + bootstrap recovery fixes (#240)
- DuckDB CHECKPOINT-on-shutdown + 60s compose grace (#235)
- Marketplace + flea-market UX polish, agnes-metadata.json enrichment
Bonus: switch release.yml test step to `-n auto` (matches ci.yml).
Single-threaded was 15-20 min and frequently the bottleneck on PR
mergeability — now ~6 min.
---------
Co-authored-by: Minas Arustamyan <arustamyan.minas@gmail.com>
Co-authored-by: ZdenekSrotyr <zdenek.srotyr@keboola.com>
879 lines
41 KiB
HTML
879 lines
41 KiB
HTML
{% extends "base.html" %}
|
|
{% block title %}Curated Marketplaces — {{ config.INSTANCE_NAME }}{% endblock %}
|
|
|
|
{% block content %}
|
|
<style>
|
|
/* Override base.html's 800px .container cap for this wide table. */
|
|
.container:has(.marketplaces-page) { max-width: none; padding: 24px 16px; }
|
|
.marketplaces-page { max-width: 1400px; margin: 0 auto; padding: 0; }
|
|
.marketplaces-toolbar {
|
|
display: flex; justify-content: space-between; align-items: center;
|
|
gap: 16px; margin-bottom: 20px; flex-wrap: wrap;
|
|
}
|
|
.marketplaces-title { margin: 0; font-size: 22px; font-weight: 600; }
|
|
.marketplaces-search {
|
|
flex: 1; max-width: 360px;
|
|
padding: 8px 12px 8px 36px;
|
|
border: 1px solid var(--border, #e5e7eb);
|
|
border-radius: 8px;
|
|
font-size: 13px;
|
|
background: var(--surface, #fff) url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' width='14' height='14' viewBox='0 0 24 24' fill='none' stroke='%236b7280' stroke-width='2'><circle cx='11' cy='11' r='8'/><path d='m21 21-4.35-4.35'/></svg>") no-repeat 12px center;
|
|
}
|
|
.marketplaces-search:focus { outline: 2px solid var(--primary, #6366f1); outline-offset: -1px; }
|
|
|
|
.marketplaces-table-wrap {
|
|
background: var(--surface, #fff);
|
|
border: 1px solid var(--border, #e5e7eb);
|
|
border-radius: 12px;
|
|
overflow-x: auto;
|
|
}
|
|
.marketplaces-table { width: 100%; border-collapse: collapse; font-size: 13px; }
|
|
.marketplaces-table thead th {
|
|
text-align: left; padding: 12px 16px;
|
|
background: var(--border-light, #f9fafb);
|
|
border-bottom: 1px solid var(--border, #e5e7eb);
|
|
font-weight: 600; color: var(--text-secondary, #6b7280);
|
|
font-size: 11px; text-transform: uppercase; letter-spacing: 0.4px;
|
|
}
|
|
.marketplaces-table tbody td {
|
|
padding: 12px 16px;
|
|
border-bottom: 1px solid var(--border-light, #f3f4f6);
|
|
vertical-align: middle;
|
|
}
|
|
.marketplaces-table tbody tr:last-child td { border-bottom: none; }
|
|
.marketplaces-table tbody tr:hover { background: var(--border-light, #fafafa); }
|
|
|
|
.mp-cell { display: flex; align-items: center; gap: 10px; }
|
|
.mp-avatar {
|
|
width: 32px; height: 32px; border-radius: 8px;
|
|
display: flex; align-items: center; justify-content: center;
|
|
background: var(--primary-light, #eef2ff); color: var(--primary, #6366f1);
|
|
flex-shrink: 0;
|
|
}
|
|
.mp-meta { display: flex; flex-direction: column; gap: 2px; min-width: 0; }
|
|
.mp-meta .slug { font-weight: 500; color: var(--text-primary, #111827); font-family: var(--font-mono, ui-monospace, monospace); font-size: 12px; }
|
|
.mp-meta .name { font-size: 11px; color: var(--text-secondary, #6b7280); }
|
|
|
|
/* Curator cell — full name on top, email muted below. Mirrors the
|
|
Marketplace cell two-line treatment so visually paired. */
|
|
.mp-curator-cell { display: flex; flex-direction: column; gap: 2px; min-width: 0; max-width: 220px; }
|
|
.mp-curator-cell .name { font-size: 12px; color: var(--text-primary, #111827); white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
|
|
.mp-curator-cell .email { font-size: 11px; color: var(--text-secondary, #6b7280); white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
|
|
|
|
.mp-url {
|
|
font-family: var(--font-mono, ui-monospace, monospace);
|
|
font-size: 12px; color: var(--text-secondary, #6b7280);
|
|
max-width: 220px; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; display: inline-block;
|
|
vertical-align: middle;
|
|
}
|
|
.mp-branch-pill {
|
|
display: inline-block; padding: 2px 8px; border-radius: 999px;
|
|
font-size: 11px; font-weight: 500;
|
|
background: #e0e7ff; color: #3730a3;
|
|
font-family: var(--font-mono, ui-monospace, monospace);
|
|
}
|
|
.mp-sha { font-family: var(--font-mono, ui-monospace, monospace); font-size: 12px; color: var(--text-primary, #111827); }
|
|
.mp-muted { color: var(--text-secondary, #9ca3af); }
|
|
.mp-err-badge {
|
|
display: inline-block; padding: 2px 8px; border-radius: 999px;
|
|
background: #fee2e2; color: #b91c1c;
|
|
font-size: 11px; font-weight: 500;
|
|
cursor: help;
|
|
}
|
|
.mp-ok-dot { color: #047857; font-weight: 600; }
|
|
.mp-token-dot {
|
|
display: inline-block; width: 10px; height: 10px; border-radius: 50%;
|
|
background: #cbd5e1;
|
|
}
|
|
.mp-token-dot.has-token { background: #10b981; }
|
|
|
|
.date-cell { color: var(--text-secondary, #6b7280); font-size: 12px; white-space: nowrap; }
|
|
|
|
.row-actions { display: flex; gap: 6px; justify-content: flex-end; flex-wrap: nowrap; white-space: nowrap; }
|
|
.icon-btn {
|
|
background: transparent; border: 1px solid var(--border, #e5e7eb); border-radius: 6px;
|
|
padding: 5px 10px; font-size: 12px; cursor: pointer;
|
|
color: var(--text-secondary, #6b7280); transition: all 0.15s;
|
|
text-decoration: none; line-height: 1.4;
|
|
white-space: nowrap;
|
|
}
|
|
.icon-btn:hover { color: var(--text-primary, #111827); border-color: #cbd5e1; background: #f9fafb; }
|
|
.icon-btn.primary { color: var(--primary, #6366f1); border-color: #c7d2fe; }
|
|
.icon-btn.primary:hover { background: #eef2ff; }
|
|
.icon-btn.danger:hover { color: #b91c1c; border-color: #fecaca; background: #fef2f2; }
|
|
.icon-btn[disabled] { opacity: 0.5; cursor: wait; }
|
|
|
|
.mp-empty, .mp-loading {
|
|
text-align: center; padding: 48px 16px;
|
|
color: var(--text-secondary, #6b7280); font-size: 13px;
|
|
}
|
|
.mp-empty .big { font-size: 15px; color: var(--text-primary, #111827); margin-bottom: 6px; font-weight: 500; }
|
|
|
|
/* Modal — identical to admin_users.html */
|
|
.modal-backdrop {
|
|
position: fixed; inset: 0; background: rgba(15, 23, 42, 0.55);
|
|
display: none; align-items: center; justify-content: center; z-index: 1000;
|
|
padding: 16px;
|
|
}
|
|
.modal-backdrop.is-open { display: flex; }
|
|
.modal-card {
|
|
background: var(--surface, #fff); border-radius: 12px;
|
|
padding: 24px; width: 100%; max-width: 480px;
|
|
box-shadow: 0 20px 60px rgba(0, 0, 0, 0.25);
|
|
max-height: 90vh; overflow-y: auto;
|
|
}
|
|
.modal-card h3 { margin: 0 0 6px; font-size: 17px; font-weight: 600; }
|
|
.modal-card p.sub { margin: 0 0 18px; font-size: 13px; color: var(--text-secondary, #6b7280); }
|
|
.modal-card label { display: block; font-size: 12px; font-weight: 500; color: var(--text-secondary, #6b7280); margin: 12px 0 6px; }
|
|
.modal-card .help { font-size: 11px; color: var(--text-secondary, #9ca3af); margin-top: 4px; }
|
|
.modal-card input[type="text"], .modal-card input[type="url"], .modal-card input[type="email"], .modal-card input[type="password"], .modal-card textarea {
|
|
width: 100%; padding: 9px 12px; border: 1px solid var(--border, #e5e7eb);
|
|
border-radius: 8px; font-size: 13px; box-sizing: border-box;
|
|
background: var(--surface, #fff); color: var(--text-primary, #111827);
|
|
font-family: inherit;
|
|
}
|
|
.modal-card textarea { min-height: 60px; resize: vertical; }
|
|
.modal-card input:focus, .modal-card textarea:focus { outline: 2px solid var(--primary, #6366f1); outline-offset: -1px; }
|
|
.modal-actions { display: flex; gap: 8px; justify-content: flex-end; margin-top: 20px; }
|
|
.modal-btn {
|
|
padding: 8px 16px; border-radius: 8px; font-size: 13px; font-weight: 500;
|
|
border: 1px solid var(--border, #e5e7eb); background: var(--surface, #fff);
|
|
cursor: pointer; transition: all 0.15s;
|
|
}
|
|
.modal-btn:hover { background: var(--border-light, #f9fafb); }
|
|
.modal-btn.primary { background: var(--primary, #6366f1); color: #fff; border-color: var(--primary, #6366f1); }
|
|
.modal-btn.primary:hover { filter: brightness(1.05); }
|
|
.modal-btn.danger { background: #dc2626; color: #fff; border-color: #dc2626; }
|
|
.modal-btn.danger:hover { filter: brightness(1.05); }
|
|
|
|
.sync-result {
|
|
margin: 12px 0;
|
|
padding: 12px; border-radius: 8px;
|
|
background: #f0fdf4; border: 1px solid #bbf7d0;
|
|
font-family: var(--font-mono, ui-monospace, monospace);
|
|
font-size: 12px; word-break: break-all;
|
|
}
|
|
.sync-result.err { background: #fef2f2; border-color: #fecaca; }
|
|
|
|
.mp-plugin-count {
|
|
display: inline-block; min-width: 22px; padding: 2px 8px;
|
|
border-radius: 999px; background: #ede9fe; color: #5b21b6;
|
|
font-size: 11px; font-weight: 600; text-align: center;
|
|
}
|
|
|
|
/* Details modal — plugin list */
|
|
.plugin-list {
|
|
margin: 12px 0 4px;
|
|
max-height: 60vh; overflow-y: auto;
|
|
border: 1px solid var(--border, #e5e7eb); border-radius: 8px;
|
|
background: var(--surface, #fff);
|
|
}
|
|
.plugin-list .empty { padding: 24px; text-align: center; color: var(--text-secondary, #6b7280); font-size: 13px; }
|
|
.plugin-item {
|
|
padding: 12px 14px;
|
|
border-bottom: 1px solid var(--border-light, #f3f4f6);
|
|
}
|
|
.plugin-item:last-child { border-bottom: none; }
|
|
.plugin-item-head {
|
|
display: flex; align-items: baseline; gap: 8px; flex-wrap: wrap;
|
|
}
|
|
.plugin-name { font-weight: 600; color: var(--text-primary, #111827); font-size: 13px; }
|
|
.plugin-version {
|
|
font-family: var(--font-mono, ui-monospace, monospace); font-size: 11px;
|
|
color: var(--text-secondary, #6b7280);
|
|
}
|
|
.plugin-source {
|
|
display: inline-block; padding: 1px 6px; border-radius: 4px;
|
|
background: #f3f4f6; color: #374151; font-size: 10px; font-weight: 500;
|
|
text-transform: uppercase; letter-spacing: 0.3px;
|
|
}
|
|
.plugin-desc {
|
|
margin-top: 4px; font-size: 12px; color: var(--text-secondary, #4b5563);
|
|
line-height: 1.45;
|
|
}
|
|
.plugin-meta {
|
|
margin-top: 6px; display: flex; gap: 12px; flex-wrap: wrap;
|
|
font-size: 11px; color: var(--text-secondary, #6b7280);
|
|
}
|
|
.plugin-meta a { color: var(--primary, #6366f1); text-decoration: none; }
|
|
.plugin-meta a:hover { text-decoration: underline; }
|
|
|
|
/* v39: SYSTEM pill + per-plugin toggle button. Pill uses the same
|
|
amber palette as the .origin-system chip on /admin/groups so the
|
|
"system" semantic reads consistently across the admin surface. */
|
|
.plugin-system-pill {
|
|
display: inline-block; padding: 1px 7px; border-radius: 999px;
|
|
background: #fef3c7; color: #92400e;
|
|
font-size: 10px; font-weight: 600;
|
|
text-transform: uppercase; letter-spacing: 0.4px;
|
|
}
|
|
.plugin-item-head .spacer { flex: 1; }
|
|
/* Toggle-style chip: same size/weight in both states, differentiated
|
|
only by tonal fill (outlined ↔ tinted). Off state reads as "click to
|
|
enable", on state reads as "currently on, click to revert". The
|
|
amber palette is shared with the SYSTEM pill + Required badge so
|
|
the org-policy semantic stays visually consistent across surfaces. */
|
|
.plugin-system-btn {
|
|
display: inline-flex; align-items: center; gap: 5px;
|
|
background: #fff; border: 1px solid #fbbf24;
|
|
border-radius: 6px; padding: 5px 11px; font-size: 11px;
|
|
font-weight: 600; cursor: pointer;
|
|
color: #92400e; white-space: nowrap;
|
|
transition: background 0.15s, border-color 0.15s, color 0.15s;
|
|
}
|
|
.plugin-system-btn svg { display: block; flex-shrink: 0; }
|
|
.plugin-system-btn:hover {
|
|
background: #fffbeb; border-color: #f59e0b; color: #78350f;
|
|
}
|
|
.plugin-system-btn.is-system {
|
|
background: #fef3c7; border-color: #fcd34d; color: #92400e;
|
|
}
|
|
.plugin-system-btn.is-system:hover {
|
|
background: #fde68a; border-color: #f59e0b; color: #78350f;
|
|
}
|
|
.plugin-system-btn[disabled] { opacity: 0.5; cursor: wait; }
|
|
|
|
/* Amber primary modal action — pairs with the system-confirm modal so
|
|
the OK button matches the trigger button's visual weight. */
|
|
.modal-btn.warning {
|
|
background: #f59e0b; color: #fff; border-color: #d97706;
|
|
}
|
|
.modal-btn.warning:hover { filter: brightness(1.06); }
|
|
|
|
/* Toast */
|
|
.toast-stack {
|
|
position: fixed; bottom: 24px; right: 24px; z-index: 2000;
|
|
display: flex; flex-direction: column; gap: 8px;
|
|
pointer-events: none;
|
|
}
|
|
.toast {
|
|
background: #111827; color: #fff; padding: 10px 16px;
|
|
border-radius: 8px; font-size: 13px; box-shadow: 0 10px 30px rgba(0, 0, 0, 0.25);
|
|
opacity: 0; transform: translateY(8px);
|
|
transition: opacity 0.2s, transform 0.2s;
|
|
pointer-events: auto; max-width: 380px;
|
|
}
|
|
.toast.show { opacity: 1; transform: translateY(0); }
|
|
.toast.success { background: #047857; }
|
|
.toast.error { background: #b91c1c; }
|
|
</style>
|
|
|
|
<div class="marketplaces-page">
|
|
<div class="marketplaces-toolbar">
|
|
<h2 class="marketplaces-title">Curated Marketplaces</h2>
|
|
<input id="mp-search" type="search" class="marketplaces-search" placeholder="Filter by slug, name, or URL…" autocomplete="off">
|
|
<a class="icon-btn" href="/marketplace/format-guide" target="_blank" rel="noopener"
|
|
title="Open the agnes-metadata.json format guide in a new tab">
|
|
Format guide →
|
|
</a>
|
|
<button class="modal-btn primary" id="open-create-btn">+ Add marketplace</button>
|
|
</div>
|
|
|
|
<div class="marketplaces-table-wrap">
|
|
<table class="marketplaces-table" id="marketplaces-table">
|
|
<thead>
|
|
<tr>
|
|
<th>Marketplace</th>
|
|
<th title="Person accountable for this marketplace's content">Curator</th>
|
|
<th>URL</th>
|
|
<th title="Plugins discovered in .claude-plugin/marketplace.json on last sync">Plugins</th>
|
|
<th>Last sync</th>
|
|
<th>Commit</th>
|
|
<th title="Auth token persisted to .env_overlay (not the DB)">Token</th>
|
|
<th style="text-align:right">Actions</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody id="mp-tbody"></tbody>
|
|
</table>
|
|
<div id="mp-loading" class="mp-loading">Loading marketplaces…</div>
|
|
<div id="mp-empty" class="mp-empty" style="display:none;">
|
|
<div class="big">No marketplaces registered</div>
|
|
<div>Click <strong>Add marketplace</strong> to register the first git repo.</div>
|
|
<div style="margin-top:8px; font-size:12px;">They are cloned to <code>$DATA_DIR/marketplaces/<slug>/</code> every night at 03:00 UTC and can be re-synced manually.</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Create modal -->
|
|
<div class="modal-backdrop" id="create-modal" role="dialog" aria-modal="true" aria-labelledby="create-modal-title">
|
|
<div class="modal-card">
|
|
<h3 id="create-modal-title">Add marketplace</h3>
|
|
<p class="sub">Register a git repository. It will be cloned into <code>$DATA_DIR/marketplaces/<slug>/</code> and fast-forwarded every night at 03:00 UTC.</p>
|
|
|
|
<label for="new-name">Display name</label>
|
|
<input id="new-name" type="text" placeholder="e.g. Acme Marketplace" required autocomplete="off">
|
|
|
|
<label for="new-slug">Slug (directory name)</label>
|
|
<input id="new-slug" type="text" placeholder="e.g. acme" required autocomplete="off" pattern="[a-z0-9][a-z0-9_-]{0,63}">
|
|
<div class="help">Lower-case alphanumerics, hyphens, and underscores. 1-64 chars, must start with a letter or digit.</div>
|
|
|
|
<label for="new-url">Git URL (https://)</label>
|
|
<input id="new-url" type="url" placeholder="https://github.com/org/repo.git" required autocomplete="off">
|
|
|
|
<label for="new-branch">Branch (optional)</label>
|
|
<input id="new-branch" type="text" placeholder="main (leave empty for remote HEAD)" autocomplete="off">
|
|
|
|
<label for="new-description">Description (optional)</label>
|
|
<textarea id="new-description" autocomplete="off"></textarea>
|
|
|
|
<label for="new-curator-name">Curator full name <span style="color:#b91c1c">*</span></label>
|
|
<input id="new-curator-name" type="text" placeholder="e.g. Jane Doe" required autocomplete="off">
|
|
<div class="help">The named person accountable for what's in this marketplace. Surfaced on plugin cards and detail pages.</div>
|
|
|
|
<label for="new-curator-email">Curator email <span style="color:#b91c1c">*</span></label>
|
|
<input id="new-curator-email" type="email" placeholder="e.g. jane.doe@example.com" required autocomplete="off">
|
|
|
|
<label for="new-token">GitHub PAT (optional — private repos only)</label>
|
|
<input id="new-token" type="password" placeholder="ghp_… or ghs_… (leave empty for public repos)" autocomplete="off">
|
|
<div class="help">Stored in <code>$DATA_DIR/state/.env_overlay</code> (chmod 600) on the data volume. Never written to the database or committed to git.</div>
|
|
|
|
<div class="modal-actions">
|
|
<button class="modal-btn" data-close-modal="create-modal">Cancel</button>
|
|
<button class="modal-btn primary" id="confirm-create-btn">Register</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Edit modal -->
|
|
<div class="modal-backdrop" id="edit-modal" role="dialog" aria-modal="true" aria-labelledby="edit-modal-title">
|
|
<div class="modal-card">
|
|
<h3 id="edit-modal-title">Edit marketplace</h3>
|
|
<p class="sub" id="edit-slug-label"></p>
|
|
|
|
<label for="edit-name">Display name</label>
|
|
<input id="edit-name" type="text" autocomplete="off">
|
|
|
|
<label for="edit-url">Git URL</label>
|
|
<input id="edit-url" type="url" autocomplete="off">
|
|
|
|
<label for="edit-branch">Branch</label>
|
|
<input id="edit-branch" type="text" placeholder="main (leave empty for remote HEAD)" autocomplete="off">
|
|
|
|
<label for="edit-description">Description</label>
|
|
<textarea id="edit-description" autocomplete="off"></textarea>
|
|
|
|
<label for="edit-curator-name">Curator full name</label>
|
|
<input id="edit-curator-name" type="text" placeholder="e.g. Jane Doe" autocomplete="off">
|
|
<div class="help">Leave unchanged to keep the current curator. Empty strings are ignored — to clear the curator, fill it in with the new owner instead.</div>
|
|
|
|
<label for="edit-curator-email">Curator email</label>
|
|
<input id="edit-curator-email" type="email" placeholder="e.g. jane.doe@example.com" autocomplete="off">
|
|
|
|
<label for="edit-token">GitHub PAT</label>
|
|
<input id="edit-token" type="password" placeholder="Leave empty to keep current. Type a new token to rotate." autocomplete="off">
|
|
<div class="help">
|
|
<label style="display:inline-flex; align-items:center; gap:6px; margin:6px 0 0; font-weight:400; color: var(--text-secondary, #6b7280);">
|
|
<input id="edit-clear-token" type="checkbox"> Remove the stored token (revert to public access)
|
|
</label>
|
|
</div>
|
|
|
|
<div class="modal-actions">
|
|
<button class="modal-btn" data-close-modal="edit-modal">Cancel</button>
|
|
<button class="modal-btn primary" id="confirm-edit-btn">Save</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Sync result modal -->
|
|
<div class="modal-backdrop" id="sync-modal" role="dialog" aria-modal="true" aria-labelledby="sync-title">
|
|
<div class="modal-card">
|
|
<h3 id="sync-title">Sync result</h3>
|
|
<p class="sub" id="sync-target"></p>
|
|
<div id="sync-result-body" class="sync-result"></div>
|
|
<div class="modal-actions">
|
|
<button class="modal-btn primary" data-close-modal="sync-modal">Close</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Marketplace details modal -->
|
|
<div class="modal-backdrop" id="details-modal" role="dialog" aria-modal="true" aria-labelledby="details-title">
|
|
<div class="modal-card" style="max-width:720px;">
|
|
<h3 id="details-title">Marketplace details</h3>
|
|
<p class="sub" id="details-sub"></p>
|
|
<div id="details-body" class="plugin-list"></div>
|
|
<div class="modal-actions">
|
|
<button class="modal-btn primary" data-close-modal="details-modal">Close</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Confirm dialog -->
|
|
<div class="modal-backdrop" id="confirm-modal" role="dialog" aria-modal="true" aria-labelledby="confirm-title">
|
|
<div class="modal-card">
|
|
<h3 id="confirm-title">Are you sure?</h3>
|
|
<p class="sub" id="confirm-text"></p>
|
|
<label style="display:flex; align-items:center; gap:8px; margin-top:6px; font-size:13px; color: var(--text-primary, #111827); font-weight:500;">
|
|
<input id="confirm-purge" type="checkbox" checked> Also delete the working copy from disk
|
|
</label>
|
|
<div class="modal-actions">
|
|
<button class="modal-btn" data-close-modal="confirm-modal">Cancel</button>
|
|
<button class="modal-btn danger" id="confirm-ok-btn">Delete</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- v39: confirm dialog for Mark/Unmark system. Replaces the native
|
|
confirm() because operators need a clear, scannable summary of
|
|
what fanout actually does to existing + future principals before
|
|
they hit OK. Title + body are populated dynamically so the same
|
|
modal handles both directions. -->
|
|
<div class="modal-backdrop" id="system-confirm-modal" role="dialog" aria-modal="true" aria-labelledby="system-confirm-title">
|
|
<div class="modal-card" style="max-width: 520px;">
|
|
<h3 id="system-confirm-title">Mark as system plugin?</h3>
|
|
<p class="sub" id="system-confirm-plugin" style="margin: 0 0 14px; font-size: 13px; color: var(--text-primary, #111827); font-weight: 500;"></p>
|
|
<div id="system-confirm-body" style="font-size: 13px; line-height: 1.55; color: var(--text-secondary, #374151);"></div>
|
|
<div class="modal-actions">
|
|
<button class="modal-btn" data-close-modal="system-confirm-modal">Cancel</button>
|
|
<button class="modal-btn warning" id="system-confirm-ok-btn">Confirm</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="toast-stack" id="toast-stack" aria-live="polite"></div>
|
|
|
|
<script>
|
|
const API = "/api/marketplaces";
|
|
|
|
function esc(s) {
|
|
const d = document.createElement("div");
|
|
d.textContent = s == null ? "" : String(s);
|
|
return d.innerHTML;
|
|
}
|
|
function fmtDate(s) { return s ? s.slice(0, 16).replace("T", " ") : "—"; }
|
|
function shortSha(s) { return s ? s.slice(0, 7) : "—"; }
|
|
|
|
// ── Toast ──
|
|
function toast(msg, kind = "") {
|
|
const el = document.createElement("div");
|
|
el.className = "toast " + kind;
|
|
el.textContent = msg;
|
|
document.getElementById("toast-stack").appendChild(el);
|
|
requestAnimationFrame(() => el.classList.add("show"));
|
|
setTimeout(() => {
|
|
el.classList.remove("show");
|
|
setTimeout(() => el.remove(), 250);
|
|
}, 3500);
|
|
}
|
|
|
|
// ── Modal helpers ──
|
|
function openModal(id) {
|
|
document.getElementById(id).classList.add("is-open");
|
|
const focusable = document.querySelector(`#${id} input, #${id} button.primary`);
|
|
focusable && focusable.focus();
|
|
}
|
|
function closeModal(id) {
|
|
document.getElementById(id).classList.remove("is-open");
|
|
}
|
|
document.querySelectorAll("[data-close-modal]").forEach(el =>
|
|
el.addEventListener("click", () => closeModal(el.dataset.closeModal)));
|
|
document.querySelectorAll(".modal-backdrop").forEach(el => {
|
|
el.addEventListener("click", e => { if (e.target === el) el.classList.remove("is-open"); });
|
|
});
|
|
document.addEventListener("keydown", e => {
|
|
if (e.key === "Escape") document.querySelectorAll(".modal-backdrop.is-open").forEach(m => m.classList.remove("is-open"));
|
|
});
|
|
|
|
// ── State ──
|
|
let allMarketplaces = [];
|
|
let filterText = "";
|
|
|
|
function renderMarketplaces() {
|
|
const tbody = document.getElementById("mp-tbody");
|
|
const loading = document.getElementById("mp-loading");
|
|
const empty = document.getElementById("mp-empty");
|
|
loading.style.display = "none";
|
|
|
|
const ft = filterText.trim().toLowerCase();
|
|
const filtered = ft
|
|
? allMarketplaces.filter(m => (m.id || "").toLowerCase().includes(ft)
|
|
|| (m.name || "").toLowerCase().includes(ft)
|
|
|| (m.url || "").toLowerCase().includes(ft))
|
|
: allMarketplaces;
|
|
|
|
if (allMarketplaces.length === 0) {
|
|
empty.style.display = "block";
|
|
tbody.innerHTML = "";
|
|
return;
|
|
}
|
|
empty.style.display = "none";
|
|
|
|
if (filtered.length === 0) {
|
|
tbody.innerHTML = `<tr><td colspan="8" class="mp-loading">No matches for "${esc(filterText)}"</td></tr>`;
|
|
return;
|
|
}
|
|
|
|
tbody.innerHTML = "";
|
|
for (const m of filtered) {
|
|
const tr = document.createElement("tr");
|
|
const lastSync = m.last_error
|
|
? `<span class="mp-err-badge" title="${esc(m.last_error)}">failed ${esc(fmtDate(m.last_synced_at))}</span>`
|
|
: (m.last_synced_at ? `<span class="mp-ok-dot">●</span> ${esc(fmtDate(m.last_synced_at))}` : `<span class="mp-muted">never</span>`);
|
|
tr.innerHTML = `
|
|
<td>
|
|
<div class="mp-cell">
|
|
<div class="mp-avatar">
|
|
<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M9 19c-5 1.5-5-2.5-7-3m14 6v-3.87a3.37 3.37 0 0 0-.94-2.61c3.14-.35 6.44-1.54 6.44-7A5.44 5.44 0 0 0 20 4.77 5.07 5.07 0 0 0 19.91 1S18.73.65 16 2.48a13.38 13.38 0 0 0-7 0C6.27.65 5.09 1 5.09 1A5.07 5.07 0 0 0 5 4.77a5.44 5.44 0 0 0-1.5 3.78c0 5.42 3.3 6.61 6.44 7A3.37 3.37 0 0 0 9 18.13V22"/></svg>
|
|
</div>
|
|
<div class="mp-meta">
|
|
<span class="slug">${esc(m.id)}</span>
|
|
<span class="name">${esc(m.name || "")}</span>
|
|
</div>
|
|
</div>
|
|
</td>
|
|
<td>${m.curator_name
|
|
? `<div class="mp-curator-cell"><span class="name">${esc(m.curator_name)}</span>${m.curator_email ? `<span class="email">${esc(m.curator_email)}</span>` : ""}</div>`
|
|
: `<span class="mp-muted" title="Open the Edit modal to fill in curator name + email">unset</span>`}</td>
|
|
<td><span class="mp-url" title="${esc(m.url)}${m.branch ? ` (branch: ${esc(m.branch)})` : ""}">${esc(m.url)}</span></td>
|
|
<td>${m.plugin_count > 0
|
|
? `<span class="mp-plugin-count">${m.plugin_count}</span>`
|
|
: `<span class="mp-muted">0</span>`}</td>
|
|
<td class="date-cell">${lastSync}</td>
|
|
<td class="mp-sha">${m.last_commit_sha ? esc(shortSha(m.last_commit_sha)) : `<span class="mp-muted">—</span>`}</td>
|
|
<td><span class="mp-token-dot ${m.has_token ? "has-token" : ""}" title="${m.has_token ? "Authenticated (PAT present in environment)" : "Public / no token"}"></span></td>
|
|
<td>
|
|
<div class="row-actions">
|
|
<button class="icon-btn primary" data-action="sync" data-id="${esc(m.id)}">Sync now</button>
|
|
<button class="icon-btn" data-action="details" data-id="${esc(m.id)}" data-name="${esc(m.name || m.id)}">Details</button>
|
|
<button class="icon-btn" data-action="edit" data-id="${esc(m.id)}">Edit</button>
|
|
<button class="icon-btn danger" data-action="delete" data-id="${esc(m.id)}" data-name="${esc(m.name || m.id)}">Delete</button>
|
|
</div>
|
|
</td>`;
|
|
tbody.appendChild(tr);
|
|
}
|
|
|
|
tbody.querySelectorAll('[data-action="sync"]').forEach(el =>
|
|
el.addEventListener("click", () => syncNow(el.dataset.id, el)));
|
|
tbody.querySelectorAll('[data-action="details"]').forEach(el =>
|
|
el.addEventListener("click", () => openDetails(el.dataset.id, el.dataset.name)));
|
|
tbody.querySelectorAll('[data-action="edit"]').forEach(el =>
|
|
el.addEventListener("click", () => openEdit(el.dataset.id)));
|
|
tbody.querySelectorAll('[data-action="delete"]').forEach(el =>
|
|
el.addEventListener("click", () => delMarketplace(el.dataset.id, el.dataset.name)));
|
|
}
|
|
|
|
async function loadMarketplaces() {
|
|
try {
|
|
const r = await fetch(API, { credentials: "include" });
|
|
if (!r.ok) throw new Error("HTTP " + r.status);
|
|
allMarketplaces = await r.json();
|
|
renderMarketplaces();
|
|
} catch (e) {
|
|
document.getElementById("mp-loading").textContent = "Failed to load marketplaces: " + e.message;
|
|
toast("Failed to load marketplaces", "error");
|
|
}
|
|
}
|
|
|
|
document.getElementById("mp-search").addEventListener("input", e => {
|
|
filterText = e.target.value;
|
|
renderMarketplaces();
|
|
});
|
|
|
|
// ── Sync now ──
|
|
async function syncNow(id, btn) {
|
|
if (btn) { btn.disabled = true; btn.textContent = "Syncing…"; }
|
|
let body, ok = false;
|
|
try {
|
|
const r = await fetch(`${API}/${encodeURIComponent(id)}/sync`, { method: "POST", credentials: "include" });
|
|
body = await r.json().catch(() => ({}));
|
|
ok = r.ok;
|
|
} catch (e) {
|
|
body = { error: e.message };
|
|
} finally {
|
|
if (btn) { btn.disabled = false; btn.textContent = "Sync now"; }
|
|
}
|
|
document.getElementById("sync-target").textContent = id;
|
|
const rb = document.getElementById("sync-result-body");
|
|
rb.classList.toggle("err", !ok);
|
|
if (ok) {
|
|
const plugLine = body.plugin_count != null ? `\nplugins: ${body.plugin_count}` : "";
|
|
rb.textContent = `action: ${body.action}\ncommit: ${body.commit}\npath: ${body.path}${plugLine}`;
|
|
toast("Sync OK", "success");
|
|
} else {
|
|
rb.textContent = body.detail || body.error || "Sync failed";
|
|
toast("Sync failed", "error");
|
|
}
|
|
openModal("sync-modal");
|
|
loadMarketplaces();
|
|
}
|
|
|
|
// ── Details ──
|
|
async function openDetails(id, name) {
|
|
const m = allMarketplaces.find(x => x.id === id);
|
|
document.getElementById("details-title").textContent = `${name || id}`;
|
|
const sub = document.getElementById("details-sub");
|
|
sub.innerHTML = `<code>${esc(id)}</code>${m && m.url ? ` · <span class="mp-url" title="${esc(m.url)}">${esc(m.url)}</span>` : ""}`;
|
|
const body = document.getElementById("details-body");
|
|
body.innerHTML = `<div class="empty">Loading plugins…</div>`;
|
|
openModal("details-modal");
|
|
try {
|
|
const r = await fetch(`${API}/${encodeURIComponent(id)}/plugins`, { credentials: "include" });
|
|
if (!r.ok) throw new Error("HTTP " + r.status);
|
|
const plugins = await r.json();
|
|
if (!plugins.length) {
|
|
body.innerHTML = `<div class="empty">No plugins found.<br>
|
|
Either the marketplace has never been synced, or its
|
|
<code>.claude-plugin/marketplace.json</code> does not list any plugins.</div>`;
|
|
return;
|
|
}
|
|
body.innerHTML = plugins.map(p => {
|
|
const src = p.source_type ? `<span class="plugin-source">${esc(p.source_type)}</span>` : "";
|
|
const ver = p.version ? `<span class="plugin-version">v${esc(p.version)}</span>` : "";
|
|
const desc = p.description ? `<div class="plugin-desc">${esc(p.description)}</div>` : "";
|
|
const meta = [];
|
|
if (p.author_name) meta.push(`by ${esc(p.author_name)}`);
|
|
if (p.category) meta.push(`category: ${esc(p.category)}`);
|
|
if (p.homepage) meta.push(`<a href="${esc(p.homepage)}" target="_blank" rel="noopener">homepage ↗</a>`);
|
|
const metaHtml = meta.length ? `<div class="plugin-meta">${meta.join(" · ")}</div>` : "";
|
|
const sysPill = p.is_system
|
|
? `<span class="plugin-system-pill" title="Mandatory for every user — managed below">SYSTEM</span>`
|
|
: "";
|
|
const sysBtnLabel = p.is_system ? "Unmark system" : "Mark as system";
|
|
const sysBtnTitle = p.is_system
|
|
? "Remove the system mark. Existing per-group grants and per-user subscriptions remain — admin can clean them up via /admin/access."
|
|
: "Make this plugin mandatory for every user. Adds RBAC grants for every group and subscriptions for every user.";
|
|
// Shield icon on both states — it's the semantic anchor for
|
|
// "org policy / system" and dropping it on unmark broke visual
|
|
// parity between the two toggle states.
|
|
const sysBtnIcon =
|
|
`<svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor"
|
|
stroke-width="2.4" stroke-linecap="round" stroke-linejoin="round" aria-hidden="true">
|
|
<path d="M12 2 4 6v6c0 5 3.4 9.5 8 10 4.6-.5 8-5 8-10V6l-8-4z"/>
|
|
</svg>`;
|
|
return `<div class="plugin-item">
|
|
<div class="plugin-item-head">
|
|
<span class="plugin-name">${esc(p.name)}</span>${ver}${src}${sysPill}
|
|
<span class="spacer"></span>
|
|
<button class="plugin-system-btn ${p.is_system ? "is-system" : ""}"
|
|
data-action="toggle-system"
|
|
data-marketplace="${esc(id)}"
|
|
data-plugin="${esc(p.name)}"
|
|
data-is-system="${p.is_system ? "1" : "0"}"
|
|
title="${esc(sysBtnTitle)}">${sysBtnIcon}${esc(sysBtnLabel)}</button>
|
|
</div>${desc}${metaHtml}
|
|
</div>`;
|
|
}).join("");
|
|
body.querySelectorAll('[data-action="toggle-system"]').forEach(btn =>
|
|
btn.addEventListener("click", () => toggleSystem(btn, id, name)));
|
|
} catch (e) {
|
|
body.innerHTML = `<div class="empty" style="color:#b91c1c;">Failed to load plugins: ${esc(e.message)}</div>`;
|
|
}
|
|
}
|
|
|
|
function toggleSystem(btn, marketplaceId, marketplaceName) {
|
|
// Two-step: render a populated confirm modal, then on OK perform the
|
|
// PUT/DELETE. The native confirm() lost the operator's eye in a tiny
|
|
// OS popup that didn't make the fanout consequences obvious.
|
|
const pluginName = btn.dataset.plugin;
|
|
const isSystem = btn.dataset.isSystem === "1";
|
|
const titleEl = document.getElementById("system-confirm-title");
|
|
const pluginEl = document.getElementById("system-confirm-plugin");
|
|
const bodyEl = document.getElementById("system-confirm-body");
|
|
const okBtn = document.getElementById("system-confirm-ok-btn");
|
|
|
|
if (isSystem) {
|
|
titleEl.textContent = "Unmark system plugin?";
|
|
pluginEl.textContent = pluginName;
|
|
bodyEl.innerHTML = `
|
|
<p style="margin: 0 0 10px;">This removes the <strong>system</strong> flag.
|
|
The plugin is no longer required.</p>
|
|
<ul style="margin: 0; padding-left: 20px;">
|
|
<li>Existing per-group RBAC grants <strong>stay</strong>.</li>
|
|
<li>Existing per-user subscriptions <strong>stay</strong>.</li>
|
|
<li>Operators can revoke either via <a href="/admin/access" style="color: var(--primary, #6366f1);">/admin/access</a> or by uninstalling on each user's stack.</li>
|
|
</ul>`;
|
|
okBtn.textContent = "Unmark system";
|
|
okBtn.classList.remove("warning");
|
|
okBtn.classList.add("primary");
|
|
} else {
|
|
titleEl.textContent = "Mark as system plugin?";
|
|
pluginEl.textContent = pluginName;
|
|
bodyEl.innerHTML = `
|
|
<p style="margin: 0 0 10px;">After confirming, <strong>every Agnes user</strong> will
|
|
receive this plugin as <strong>required</strong>:</p>
|
|
<ul style="margin: 0 0 10px; padding-left: 20px;">
|
|
<li>RBAC grants are added for <strong>every existing group</strong>.</li>
|
|
<li>Subscriptions are created for <strong>every existing user</strong>.</li>
|
|
<li>Every <strong>new user</strong> and <strong>new group</strong> inherits it automatically.</li>
|
|
<li>Users see it as <strong>Required</strong> on /marketplace and cannot uninstall it from /marketplace?tab=my.</li>
|
|
<li>Admins cannot revoke it from /admin/access while the system flag is on.</li>
|
|
</ul>
|
|
<p style="margin: 0; font-size: 12px; color: var(--text-secondary, #6b7280);">
|
|
Unmarking later does not auto-clean the materialized grants — they stay until manually removed.
|
|
</p>`;
|
|
okBtn.textContent = "Mark as system";
|
|
okBtn.classList.remove("primary");
|
|
okBtn.classList.add("warning");
|
|
}
|
|
|
|
// Single-shot handler — re-bind on every open so we don't leak across
|
|
// toggles and so the closure captures the current btn / ids.
|
|
const onConfirm = async () => {
|
|
okBtn.removeEventListener("click", onConfirm);
|
|
closeModal("system-confirm-modal");
|
|
await performToggleSystem(btn, marketplaceId, marketplaceName, isSystem);
|
|
};
|
|
okBtn.addEventListener("click", onConfirm);
|
|
// Cancel cleanup — strip the listener so the next open starts fresh.
|
|
const onCancel = (e) => {
|
|
if (e && e.target === okBtn) return;
|
|
okBtn.removeEventListener("click", onConfirm);
|
|
};
|
|
document.querySelectorAll('[data-close-modal="system-confirm-modal"]').forEach(el =>
|
|
el.addEventListener("click", onCancel, { once: true }));
|
|
|
|
openModal("system-confirm-modal");
|
|
}
|
|
|
|
async function performToggleSystem(btn, marketplaceId, marketplaceName, isSystem) {
|
|
const pluginName = btn.dataset.plugin;
|
|
const method = isSystem ? "DELETE" : "POST";
|
|
btn.disabled = true;
|
|
const prevHtml = btn.innerHTML;
|
|
btn.textContent = "Working…";
|
|
try {
|
|
const url = `${API}/${encodeURIComponent(marketplaceId)}/plugins/${encodeURIComponent(pluginName)}/system`;
|
|
const r = await fetch(url, { method, credentials: "include" });
|
|
if (!r.ok) {
|
|
const err = await r.json().catch(() => ({}));
|
|
toast(`Toggle failed: ${err.detail || r.status}`, "error");
|
|
btn.disabled = false;
|
|
btn.innerHTML = prevHtml;
|
|
return;
|
|
}
|
|
const result = await r.json();
|
|
if (!isSystem) {
|
|
toast(`Marked ${pluginName} as system (${result.affected_groups} groups, ${result.affected_users} users)`, "success");
|
|
} else {
|
|
toast(`Unmarked ${pluginName} from system`, "success");
|
|
}
|
|
// Reload the modal so the pill + button label refresh from a fresh fetch.
|
|
openDetails(marketplaceId, marketplaceName);
|
|
} catch (e) {
|
|
toast(`Network error: ${e.message}`, "error");
|
|
btn.disabled = false;
|
|
btn.innerHTML = prevHtml;
|
|
}
|
|
}
|
|
|
|
// ── Edit ──
|
|
function openEdit(id) {
|
|
const m = allMarketplaces.find(x => x.id === id);
|
|
if (!m) return;
|
|
document.getElementById("edit-slug-label").textContent = `Editing ${m.id}`;
|
|
document.getElementById("edit-name").value = m.name || "";
|
|
document.getElementById("edit-url").value = m.url || "";
|
|
document.getElementById("edit-branch").value = m.branch || "";
|
|
document.getElementById("edit-description").value = m.description || "";
|
|
document.getElementById("edit-curator-name").value = m.curator_name || "";
|
|
document.getElementById("edit-curator-email").value = m.curator_email || "";
|
|
document.getElementById("edit-token").value = "";
|
|
document.getElementById("edit-clear-token").checked = false;
|
|
|
|
const btn = document.getElementById("confirm-edit-btn");
|
|
btn.onclick = async () => {
|
|
const payload = {
|
|
name: document.getElementById("edit-name").value.trim() || null,
|
|
url: document.getElementById("edit-url").value.trim() || null,
|
|
branch: document.getElementById("edit-branch").value.trim(), // empty string cleared by API (None=untouched)
|
|
description: document.getElementById("edit-description").value,
|
|
curator_name: document.getElementById("edit-curator-name").value.trim() || null,
|
|
curator_email: document.getElementById("edit-curator-email").value.trim() || null,
|
|
};
|
|
// branch: null means untouched; explicit "" clears to HEAD. Treat empty input as clear only if user typed then erased.
|
|
// Simpler UX: always send current value (empty = HEAD).
|
|
if (payload.branch === "") payload.branch = null; // untouched
|
|
const tokenVal = document.getElementById("edit-token").value;
|
|
const clearTok = document.getElementById("edit-clear-token").checked;
|
|
if (clearTok) payload.token = "";
|
|
else if (tokenVal) payload.token = tokenVal;
|
|
const r = await fetch(`${API}/${encodeURIComponent(id)}`, {
|
|
method: "PATCH", credentials: "include",
|
|
headers: { "Content-Type": "application/json" },
|
|
body: JSON.stringify(payload),
|
|
});
|
|
if (!r.ok) { toast("Update failed: " + (await r.text()), "error"); return; }
|
|
closeModal("edit-modal");
|
|
toast("Marketplace updated", "success");
|
|
loadMarketplaces();
|
|
};
|
|
openModal("edit-modal");
|
|
}
|
|
|
|
// ── Delete ──
|
|
function delMarketplace(id, name) {
|
|
document.getElementById("confirm-text").textContent =
|
|
`Delete marketplace "${name}" (${id})? Unregisters it and stops future nightly syncs.`;
|
|
document.getElementById("confirm-purge").checked = true;
|
|
const okBtn = document.getElementById("confirm-ok-btn");
|
|
okBtn.onclick = async () => {
|
|
const purge = document.getElementById("confirm-purge").checked;
|
|
const r = await fetch(`${API}/${encodeURIComponent(id)}?purge=${purge}`, {
|
|
method: "DELETE", credentials: "include",
|
|
});
|
|
closeModal("confirm-modal");
|
|
if (!r.ok) { toast("Delete failed: " + (await r.text()), "error"); return; }
|
|
toast("Marketplace removed", "success");
|
|
loadMarketplaces();
|
|
};
|
|
openModal("confirm-modal");
|
|
}
|
|
|
|
// ── Create ──
|
|
document.getElementById("open-create-btn").addEventListener("click", () => {
|
|
document.getElementById("new-name").value = "";
|
|
document.getElementById("new-slug").value = "";
|
|
document.getElementById("new-url").value = "";
|
|
document.getElementById("new-branch").value = "";
|
|
document.getElementById("new-description").value = "";
|
|
document.getElementById("new-curator-name").value = "";
|
|
document.getElementById("new-curator-email").value = "";
|
|
document.getElementById("new-token").value = "";
|
|
openModal("create-modal");
|
|
});
|
|
document.getElementById("confirm-create-btn").addEventListener("click", async () => {
|
|
const curatorName = document.getElementById("new-curator-name").value.trim();
|
|
const curatorEmail = document.getElementById("new-curator-email").value.trim();
|
|
const payload = {
|
|
name: document.getElementById("new-name").value.trim(),
|
|
slug: document.getElementById("new-slug").value.trim().toLowerCase(),
|
|
url: document.getElementById("new-url").value.trim(),
|
|
branch: document.getElementById("new-branch").value.trim() || null,
|
|
description: document.getElementById("new-description").value.trim() || null,
|
|
curator_name: curatorName,
|
|
curator_email: curatorEmail,
|
|
token: document.getElementById("new-token").value || null,
|
|
};
|
|
if (!payload.name || !payload.slug || !payload.url) {
|
|
toast("Name, slug, and URL are required", "error");
|
|
return;
|
|
}
|
|
if (!curatorName || !curatorEmail) {
|
|
toast("Curator full name and email are required", "error");
|
|
return;
|
|
}
|
|
const r = await fetch(API, {
|
|
method: "POST", credentials: "include",
|
|
headers: { "Content-Type": "application/json" },
|
|
body: JSON.stringify(payload),
|
|
});
|
|
if (!r.ok) {
|
|
const err = await r.json().catch(() => ({}));
|
|
toast("Create failed: " + (err.detail || r.status), "error");
|
|
return;
|
|
}
|
|
closeModal("create-modal");
|
|
toast("Marketplace registered", "success");
|
|
loadMarketplaces();
|
|
});
|
|
|
|
// Auto-derive slug from name on first focus
|
|
document.getElementById("new-slug").addEventListener("focus", e => {
|
|
if (e.target.value) return;
|
|
const name = document.getElementById("new-name").value.trim().toLowerCase();
|
|
e.target.value = name.replace(/[^a-z0-9_-]+/g, "-").replace(/^-+|-+$/g, "").slice(0, 64);
|
|
});
|
|
|
|
loadMarketplaces();
|
|
</script>
|
|
{% endblock %}
|