mirror of
https://github.com/AgentSeal/codeburn.git
synced 2026-05-16 19:44:14 +00:00
Some checks are pending
CI / semgrep (push) Waiting to run
* Rewrite GNOME extension UI with branded popover matching macOS menubar Combines PR #212's modular architecture (DataClient, GSettings, Libadwaita prefs) with the custom St widget UI from feat/tauri-menubar-win-linux. Adds: branded header, horizontal agent tabs, hero typography, period/insight pills, 19-day token histogram, 6 content views (Activity, Trend, Forecast, Pulse, Stats, Plan), currency switcher with FX conversion, findings CTA, budget alerts, theme detection, payload caching with TTL. * Add Main.panel.addToStatusArea call to extension entry point * Align activity/model rows as table with separators, gear icon for prefs * Add table column headers, oneshot placeholder, currency picker dropdown * Enhance GNOME extension with scrollable UI, dark mode, charts, and performance fixes - Add vertical scroll for popup content and horizontal scroll for 6+ provider tabs - Add token histogram chart with hover tooltips showing date, in/out tokens, cost - Add skeleton loading animation with stale-while-revalidate caching (5min TTL) - Add dark/light theme support with force-dark-mode setting - Add exact costs toggle for full decimal values - Add right-aligned columns for cost, turns, oneshot, and model data - Remove unsupported St CSS properties (text-align, letter-spacing) - Fix post-destroy crash: guard async callbacks, abort Soup session on teardown - Fix dataClient double-resolve race with settled guard - Expand PATH resolution for volta, bun, cargo, asdf, fnm, pnpm - Reduce CLI timeout from 45s to 15s and cache augmented PATH - Remove unused imports (Pango, Main) and dead constants - Show 10 top activities, remove Plan pill
161 lines
3.9 KiB
JavaScript
161 lines
3.9 KiB
JavaScript
import GLib from 'gi://GLib';
|
|
import Gio from 'gi://Gio';
|
|
|
|
const TIMEOUT_SECONDS = 15;
|
|
const SAFE_ARG_RE = /^[A-Za-z0-9 ._/\-]+$/;
|
|
|
|
function buildAdditionalPaths() {
|
|
const home = GLib.get_home_dir();
|
|
return [
|
|
'/usr/local/bin',
|
|
`${home}/.local/bin`,
|
|
`${home}/.npm-global/bin`,
|
|
`${home}/.volta/bin`,
|
|
`${home}/.bun/bin`,
|
|
`${home}/.cargo/bin`,
|
|
`${home}/.asdf/shims`,
|
|
`${home}/.local/share/fnm/aliases/default/bin`,
|
|
`${home}/.local/share/pnpm`,
|
|
];
|
|
}
|
|
|
|
export class DataClient {
|
|
_cache = new Map();
|
|
_inFlight = null;
|
|
_codeburnPath;
|
|
_augmentedPath;
|
|
|
|
constructor(codeburnPath) {
|
|
this._codeburnPath = codeburnPath || '';
|
|
this._augmentedPath = this._buildAugmentedPath();
|
|
}
|
|
|
|
setCodeburnPath(path) {
|
|
this._codeburnPath = path || '';
|
|
}
|
|
|
|
cancelInFlight() {
|
|
if (this._inFlight) {
|
|
this._inFlight.cancellable.cancel();
|
|
this._inFlight = null;
|
|
}
|
|
}
|
|
|
|
getCached(period, provider) {
|
|
const key = `${period}:${provider}`;
|
|
return this._cache.get(key) ?? null;
|
|
}
|
|
|
|
async fetch(period, provider) {
|
|
this.cancelInFlight();
|
|
|
|
const cancellable = new Gio.Cancellable();
|
|
this._inFlight = { cancellable };
|
|
|
|
try {
|
|
const payload = await this._spawn(period, provider, cancellable);
|
|
const key = `${period}:${provider}`;
|
|
this._cache.set(key, payload);
|
|
return payload;
|
|
} finally {
|
|
if (this._inFlight?.cancellable === cancellable)
|
|
this._inFlight = null;
|
|
}
|
|
}
|
|
|
|
_buildArgv(period, provider) {
|
|
let base;
|
|
if (this._codeburnPath && SAFE_ARG_RE.test(this._codeburnPath)) {
|
|
base = this._codeburnPath.split(' ').filter(s => s.length > 0);
|
|
} else {
|
|
base = ['codeburn'];
|
|
}
|
|
|
|
const args = [
|
|
...base,
|
|
'status',
|
|
'--format', 'menubar-json',
|
|
'--period', period,
|
|
'--no-optimize',
|
|
];
|
|
|
|
if (provider && provider !== 'all')
|
|
args.push('--provider', provider);
|
|
|
|
return args;
|
|
}
|
|
|
|
_buildAugmentedPath() {
|
|
const currentPath = GLib.getenv('PATH') || '/usr/bin:/bin';
|
|
const parts = currentPath.split(':');
|
|
for (const extra of buildAdditionalPaths()) {
|
|
if (!parts.includes(extra))
|
|
parts.push(extra);
|
|
}
|
|
return parts.join(':');
|
|
}
|
|
|
|
_spawn(period, provider, cancellable) {
|
|
return new Promise((resolve, reject) => {
|
|
const argv = this._buildArgv(period, provider);
|
|
let settled = false;
|
|
|
|
const settle = (fn, value) => {
|
|
if (settled) return;
|
|
settled = true;
|
|
fn(value);
|
|
};
|
|
|
|
let proc;
|
|
try {
|
|
const launcher = Gio.SubprocessLauncher.new(
|
|
Gio.SubprocessFlags.STDOUT_PIPE | Gio.SubprocessFlags.STDERR_PIPE
|
|
);
|
|
launcher.setenv('PATH', this._augmentedPath, true);
|
|
proc = launcher.spawnv(argv);
|
|
} catch (e) {
|
|
settle(reject, new Error(`CLI not found: ${e.message}`));
|
|
return;
|
|
}
|
|
|
|
let timeoutId = GLib.timeout_add_seconds(GLib.PRIORITY_DEFAULT, TIMEOUT_SECONDS, () => {
|
|
timeoutId = 0;
|
|
proc.force_exit();
|
|
settle(reject, new Error('CLI timeout'));
|
|
return GLib.SOURCE_REMOVE;
|
|
});
|
|
|
|
proc.communicate_utf8_async(null, cancellable, (_proc, res) => {
|
|
if (timeoutId) {
|
|
GLib.Source.remove(timeoutId);
|
|
timeoutId = 0;
|
|
}
|
|
|
|
try {
|
|
const [, stdout, stderr] = _proc.communicate_utf8_finish(res);
|
|
|
|
if (!_proc.get_successful()) {
|
|
const msg = stderr?.trim() || 'CLI exited with error';
|
|
settle(reject, new Error(msg));
|
|
return;
|
|
}
|
|
|
|
if (!stdout || stdout.trim().length === 0) {
|
|
settle(reject, new Error('CLI returned empty output'));
|
|
return;
|
|
}
|
|
|
|
const payload = JSON.parse(stdout);
|
|
settle(resolve, payload);
|
|
} catch (e) {
|
|
settle(reject, e);
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
destroy() {
|
|
this.cancelInFlight();
|
|
this._cache.clear();
|
|
}
|
|
}
|