diff --git a/.github/PULL_REQUEST_TEMPLATE.md b/.github/PULL_REQUEST_TEMPLATE.md
new file mode 100644
index 0000000..9af748a
--- /dev/null
+++ b/.github/PULL_REQUEST_TEMPLATE.md
@@ -0,0 +1,18 @@
+## Summary
+
+
+
+## Testing
+
+- [ ] I have tested this locally against real data (not just unit tests)
+- [ ] `npm test` passes
+- [ ] `npm run build` succeeds
+
+### For new providers only:
+
+- [ ] I installed the tool and generated real sessions by using it
+- [ ] `npm run dev -- today` shows correct costs and session counts for this provider
+- [ ] `npm run dev -- models --provider ` shows correct model names and pricing
+- [ ] Screenshot or terminal output attached below proving it works with real data
+
+
diff --git a/.github/workflows/release-menubar.yml b/.github/workflows/release-menubar.yml
index 990d473..b2cf949 100644
--- a/.github/workflows/release-menubar.yml
+++ b/.github/workflows/release-menubar.yml
@@ -45,7 +45,9 @@ jobs:
uses: actions/upload-artifact@v4
with:
name: CodeBurnMenubar-${{ steps.version.outputs.value }}
- path: mac/.build/dist/CodeBurnMenubar-*.zip
+ path: |
+ mac/.build/dist/CodeBurnMenubar-${{ steps.version.outputs.value }}.zip
+ mac/.build/dist/CodeBurnMenubar-${{ steps.version.outputs.value }}.zip.sha256
if-no-files-found: error
- name: Create / update GitHub Release
@@ -66,6 +68,6 @@ jobs:
and macOS shows "cannot verify developer", right-click the app in Finder and
pick Open to whitelist it once.
files: |
- mac/.build/dist/CodeBurnMenubar-*.zip
- mac/.build/dist/CodeBurnMenubar-*.zip.sha256
+ mac/.build/dist/CodeBurnMenubar-${{ steps.version.outputs.value }}.zip
+ mac/.build/dist/CodeBurnMenubar-${{ steps.version.outputs.value }}.zip.sha256
fail_on_unmatched_files: true
diff --git a/CHANGELOG.md b/CHANGELOG.md
index f5b69b7..d8c1163 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -2,16 +2,15 @@
## Unreleased
+### Added (CLI)
+- **IBM Bob provider.** Discovers IBM Bob IDE task history, reuses the
+ Cline-family parser for token/cost records, extracts model tags and
+ workspace-based project names from session data. Closes #248.
+
### Fixed (CLI)
-- **Claude 1-hour cache writes use the correct price.** Claude Code records
- 5-minute and 1-hour prompt-cache writes separately in
- `usage.cache_creation`. CodeBurn now prices the 1-hour portion at 2x base
- input cost (1.6x the LiteLLM 5-minute cache-write rate) while preserving the
- existing legacy fallback when only `cache_creation_input_tokens` is present.
- Daily cache version bumped to v6 so previously cached under-reported costs
- are recomputed from raw sessions.
- This fixes under-reporting for plan-mode and long agent sessions that rely on
- 1-hour cache writes. Closes #276.
+- **Claude 1-hour cache write pricing.** 1-hour cache writes are now priced
+ at 2x base input (previously used the 5-minute 1.25x rate for all writes).
+ Daily cache bumped to v6 so stale totals are recomputed. Closes #276.
## 0.9.8 - 2026-05-10
diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
index 84b21f4..aebe0f2 100644
--- a/CONTRIBUTING.md
+++ b/CONTRIBUTING.md
@@ -84,6 +84,23 @@ The `.github/workflows/block-claude-coauthor.yml` workflow rejects any PR whose
If a flagged PR rejects on this check, the workflow prints the exact rebase command to fix it.
+## Before You Start
+
+**Comment on the issue first.** Before writing code for a feature or new provider, leave a comment on the relevant issue saying what you plan to do. Wait for a maintainer to confirm the approach. Unsolicited PRs that duplicate work already in progress or take an incompatible approach will be closed.
+
+**One PR at a time.** We will not review a second PR from you until the first is merged or closed. This keeps the review queue manageable and ensures each contribution gets proper attention.
+
+## Adding a New Provider
+
+New providers have the highest bar because broken parsing silently produces wrong data for users. Before opening a PR:
+
+1. **Install the tool and use it.** Generate real sessions by actually coding with the provider. We do this ourselves for every provider we ship.
+2. **Test against real data.** Run `npm run dev -- today` and `npm run dev -- models` with your real sessions and confirm the output looks correct — costs are non-zero, model names resolve, session counts match what you see in the tool.
+3. **Include proof in the PR.** Attach a screenshot or terminal output showing codeburn correctly parsing your real sessions. PRs for new providers without evidence of local testing will not be reviewed.
+4. **Do not rely on AI-generated guesses about storage paths or schemas.** Tools change their data formats between versions. The only way to know the current schema is to install the tool and inspect the actual files on disk.
+
+PRs that add a provider based solely on online documentation or AI-generated code, without evidence of testing against real data, will be closed.
+
## Pull Requests
1. Fork or branch from `main`.
diff --git a/README.md b/README.md
index b370022..9db2a1f 100644
--- a/README.md
+++ b/README.md
@@ -13,7 +13,7 @@
-CodeBurn tracks token usage, cost, and performance across **18 AI coding tools**. It breaks down spending by task type, model, tool, project, and provider so you can see exactly where your budget goes.
+CodeBurn tracks token usage, cost, and performance across **19 AI coding tools**. It breaks down spending by task type, model, tool, project, and provider so you can see exactly where your budget goes.
Everything runs locally. No wrapper, no proxy, no API keys. CodeBurn reads session data directly from disk and prices every call using [LiteLLM](https://github.com/BerriAI/litellm).
@@ -104,6 +104,7 @@ Arrow keys switch between Today, 7 Days, 30 Days, Month, and 6 Months (use `--fr
| | cursor-agent | Yes | [cursor-agent.md](docs/providers/cursor-agent.md) |
| | Gemini CLI | Yes | [gemini.md](docs/providers/gemini.md) |
| | GitHub Copilot | Yes | [copilot.md](docs/providers/copilot.md) |
+| | IBM Bob | Yes | [ibm-bob.md](docs/providers/ibm-bob.md) |
| | Kiro | Yes | [kiro.md](docs/providers/kiro.md) |
| | OpenCode | Yes | [opencode.md](docs/providers/opencode.md) |
| | OpenClaw | Yes | [openclaw.md](docs/providers/openclaw.md) |
@@ -119,7 +120,7 @@ Arrow keys switch between Today, 7 Days, 30 Days, Month, and 6 Months (use `--fr
Each provider doc lists the exact data location, storage format, and known quirks. Linux and Windows paths are detected automatically. If a path has changed or is wrong, please [open an issue](https://github.com/getagentseal/codeburn/issues).
-Provider logos are trademarks of their respective owners. The icon set was sourced from [tokscale](https://github.com/junhoyeo/tokscale) (MIT) plus official vendor assets, used under nominative fair use for the purpose of identifying supported tools.
+Provider logos are trademarks of their respective owners. The icon set was sourced from [tokscale](https://github.com/junhoyeo/tokscale) (MIT), official vendor assets, and simple provider identifiers, used under nominative fair use for the purpose of identifying supported tools.
CodeBurn auto-detects which AI coding tools you use. If multiple providers have session data on disk, press `p` in the dashboard to toggle between them.
@@ -378,6 +379,8 @@ These are starting points, not verdicts. A 60% cache hit on a single experimenta
**OpenClaw** stores agent sessions as JSONL at `~/.openclaw/agents/*.jsonl`. Also checks legacy paths `.clawdbot`, `.moltbot`, `.moldbot`. Token usage comes from assistant message `usage` blocks; model from `modelId` or `message.model` fields.
+**IBM Bob** stores IDE task history in `User/globalStorage/ibm.bob-code/tasks//` under the IBM Bob application data directory. CodeBurn reads `ui_messages.json` for API request token/cost records and `api_conversation_history.json` for the selected model, with support for both GA (`IBM Bob`) and preview (`Bob-IDE`) app data folders.
+
**Roo Code / KiloCode** are Cline-family VS Code extensions. CodeBurn reads `ui_messages.json` from each task directory in VS Code's `globalStorage`, filtering `type: "say"` entries with `say: "api_req_started"` to extract token counts.
CodeBurn deduplicates messages (by API message ID for Claude, by cumulative token cross-check for Codex, by conversation/timestamp for Cursor, by session ID for Gemini, by session+message ID for OpenCode, by responseId for Pi/OMP), filters by date range per entry, and classifies each turn.
diff --git a/assets/providers/ibm-bob.svg b/assets/providers/ibm-bob.svg
new file mode 100644
index 0000000..ab76047
--- /dev/null
+++ b/assets/providers/ibm-bob.svg
@@ -0,0 +1,6 @@
+
diff --git a/docs/architecture.md b/docs/architecture.md
index 9b1ea14..c3a8c25 100644
--- a/docs/architecture.md
+++ b/docs/architecture.md
@@ -128,14 +128,14 @@ type Provider = {
}
```
-`src/providers/index.ts` registers eighteen providers across two tiers:
+`src/providers/index.ts` registers nineteen providers across two tiers:
-- **Eager**: `claude`, `codex`, `copilot`, `droid`, `gemini`, `kilo-code`, `kiro`, `openclaw`, `pi`, `omp`, `qwen`, `roo-code`. Imported at module load.
+- **Eager**: `claude`, `codex`, `copilot`, `droid`, `gemini`, `ibm-bob`, `kilo-code`, `kiro`, `openclaw`, `pi`, `omp`, `qwen`, `roo-code`. Imported at module load.
- **Lazy**: `antigravity`, `goose`, `cursor`, `opencode`, `cursor-agent`, `crush`. Imported via dynamic `import()` so the heavy dependencies (SQLite, protobuf) do not touch users who do not have those tools installed.
Both lists hit the same `getAllProviders()` aggregator. A failed lazy import is silent and excludes that provider from the run.
-`src/providers/vscode-cline-parser.ts` is a shared helper consumed by `kilo-code` and `roo-code`. It is not registered as a provider on its own.
+`src/providers/vscode-cline-parser.ts` is a shared helper consumed by `ibm-bob`, `kilo-code`, and `roo-code`. It is not registered as a provider on its own.
For the per-provider data location, storage format, parser quirks, and test coverage, see `docs/providers/`.
diff --git a/docs/providers/README.md b/docs/providers/README.md
index 05f43db..600bd60 100644
--- a/docs/providers/README.md
+++ b/docs/providers/README.md
@@ -15,6 +15,7 @@ For the architectural picture, see `../architecture.md`.
| [Copilot](copilot.md) | JSONL | `src/providers/copilot.ts` | `tests/providers/copilot.test.ts` |
| [Droid](droid.md) | JSONL | `src/providers/droid.ts` | `tests/providers/droid.test.ts` |
| [Gemini](gemini.md) | JSON / JSONL | `src/providers/gemini.ts` | none |
+| [IBM Bob](ibm-bob.md) | JSON | `src/providers/ibm-bob.ts` | `tests/providers/ibm-bob.test.ts` |
| [KiloCode](kilo-code.md) | JSON | `src/providers/kilo-code.ts` | `tests/providers/kilo-code.test.ts` |
| [Kiro](kiro.md) | JSON | `src/providers/kiro.ts` | `tests/providers/kiro.test.ts` |
| [OpenClaw](openclaw.md) | JSONL | `src/providers/openclaw.ts` | `tests/providers/openclaw.test.ts` |
@@ -38,7 +39,7 @@ For the architectural picture, see `../architecture.md`.
| Helper | Used by | Source |
|---|---|---|
-| [vscode-cline-parser](vscode-cline-parser.md) | `kilo-code`, `roo-code` | `src/providers/vscode-cline-parser.ts` |
+| [vscode-cline-parser](vscode-cline-parser.md) | `ibm-bob`, `kilo-code`, `roo-code` | `src/providers/vscode-cline-parser.ts` |
## File Format
diff --git a/docs/providers/ibm-bob.md b/docs/providers/ibm-bob.md
new file mode 100644
index 0000000..c9d4373
--- /dev/null
+++ b/docs/providers/ibm-bob.md
@@ -0,0 +1,55 @@
+# IBM Bob
+
+IBM Bob IDE task history.
+
+- **Source:** `src/providers/ibm-bob.ts`
+- **Loading:** eager (`src/providers/index.ts`)
+- **Test:** `tests/providers/ibm-bob.test.ts`
+
+## Where It Reads From
+
+IBM Bob stores IDE task history below `User/globalStorage/ibm.bob-code/tasks/` in the application data directory.
+
+Default paths checked:
+
+| Platform | Paths |
+|---|---|
+| macOS | `~/Library/Application Support/IBM Bob/User/globalStorage/ibm.bob-code/`, `~/Library/Application Support/Bob-IDE/User/globalStorage/ibm.bob-code/` |
+| Windows | `%APPDATA%/IBM Bob/User/globalStorage/ibm.bob-code/`, `%APPDATA%/Bob-IDE/User/globalStorage/ibm.bob-code/` |
+| Linux | `$XDG_CONFIG_HOME/IBM Bob/User/globalStorage/ibm.bob-code/`, `$XDG_CONFIG_HOME/Bob-IDE/User/globalStorage/ibm.bob-code/` with `~/.config` fallback |
+
+The `Bob-IDE` paths cover the preview-era app name that some installs used before the GA `IBM Bob` directory.
+
+## Storage Format
+
+Each task is a directory under `tasks//` and must contain `ui_messages.json`.
+
+CodeBurn parses the same Cline-family UI event format used by Roo Code and KiloCode:
+
+- `ui_messages.json` entries with `type: "say"` and `say: "api_req_started"` contain serialized token/cost metrics.
+- `ui_messages.json` user text entries seed the turn's first user message.
+- `api_conversation_history.json` is optional and is used to extract the selected model from `...` environment details when present.
+- `task_metadata.json` may exist upstream, but CodeBurn does not need it for usage math today.
+
+If no model tag is present, the parser uses `ibm-bob-auto`, which is priced through the same conservative Sonnet fallback used for Cline-family auto modes.
+
+## Caching
+
+None at the provider level.
+
+## Deduplication
+
+Per `::` via `vscode-cline-parser.ts`.
+
+## Quirks
+
+- IBM Bob has shipped under both `IBM Bob` and `Bob-IDE` application data folder names.
+- This provider intentionally covers the IDE task-history format. Bob Shell's `~/.bob` checkpoint data is a separate storage surface and is not parsed until we have a stable usage schema fixture.
+- The shared Cline parser does not currently extract individual tool names from UI messages, so tool breakdowns are empty for IBM Bob just like Roo Code and KiloCode.
+
+## When Fixing A Bug Here
+
+1. Check whether the install uses `IBM Bob` or `Bob-IDE` as the application data directory.
+2. Confirm the task folder still contains `ui_messages.json` and `api_conversation_history.json`.
+3. If the UI message schema changed, add a focused fixture to `tests/providers/ibm-bob.test.ts`.
+4. If the change also affects Roo Code or KiloCode, update `src/providers/vscode-cline-parser.ts` and run all three provider test files.
diff --git a/docs/providers/vscode-cline-parser.md b/docs/providers/vscode-cline-parser.md
index 5b6bdfa..ea68eae 100644
--- a/docs/providers/vscode-cline-parser.md
+++ b/docs/providers/vscode-cline-parser.md
@@ -1,17 +1,18 @@
# vscode-cline-parser (Shared Helper)
-Shared discovery and parsing for VS Code extensions descended from Cline.
+Shared discovery and parsing for Cline-family task folders.
- **Source:** `src/providers/vscode-cline-parser.ts`
-- **Loading:** not a provider; imported by `kilo-code.ts` and `roo-code.ts`.
-- **Test:** none directly. Coverage comes from `tests/providers/kilo-code.test.ts` and `tests/providers/roo-code.test.ts`.
+- **Loading:** not a provider; imported by `ibm-bob.ts`, `kilo-code.ts`, and `roo-code.ts`.
+- **Test:** none directly. Coverage comes from `tests/providers/ibm-bob.test.ts`, `tests/providers/kilo-code.test.ts`, and `tests/providers/roo-code.test.ts`.
## What it does
Two responsibilities:
-1. `discoverClineTasks(extensionId)` walks VS Code's `globalStorage//tasks/` directories and returns one source per task that has a `ui_messages.json` file (`vscode-cline-parser.ts:25-50`).
-2. `createClineParser` reads each task's `ui_messages.json` and `api_conversation_history.json`, extracts model, tools, and token counts, and yields `ParsedProviderCall` objects.
+1. `discoverClineTasks(extensionId)` walks VS Code's `globalStorage//tasks/` directories and returns one source per task that has a `ui_messages.json` file.
+2. `discoverClineTasksInBaseDirs(baseDirs)` does the same for non-VS Code apps with compatible task storage, such as IBM Bob.
+3. `createClineParser` reads each task's `ui_messages.json` and `api_conversation_history.json`, extracts model and token counts, and yields `ParsedProviderCall` objects.
## Storage layout
@@ -25,25 +26,25 @@ Per task directory:
## Model resolution
-The model is extracted from `api_conversation_history.json` by searching user message content blocks for a `...` tag (`vscode-cline-parser.ts:54-72`). Falls back to `cline-auto` if no tag is found.
+The model is extracted from `api_conversation_history.json` by searching user message content blocks for a `...` tag. Falls back to the provider-supplied auto model (`cline-auto` by default) if no tag is found.
## Token extraction
-From `api_req_started` entries inside `ui_messages.json`. Each such entry's `text` field is JSON-parsed; the parsed object holds `tokensIn`, `tokensOut`, `cacheReads`, `cacheWrites`, and (optionally) `cost` (`vscode-cline-parser.ts:119-134`).
+From `api_req_started` entries inside `ui_messages.json`. Each such entry's `text` field is JSON-parsed; the parsed object holds `tokensIn`, `tokensOut`, `cacheReads`, `cacheWrites`, and (optionally) `cost`.
-If `cost` is present, it is used directly. If not, `calculateCost` from `src/models.ts` computes it from tokens (`vscode-cline-parser.ts:139`).
+If `cost` is present, it is used directly. If not, `calculateCost` from `src/models.ts` computes it from tokens.
## Deduplication
-Per `::` where `index` is the position of the `api_req_started` entry within `ui_messages.json` (`vscode-cline-parser.ts:109`).
+Per `::` where `index` is the position of the `api_req_started` entry within `ui_messages.json`.
## Quirks
-- Only the **first** user message is emitted as `userMessage` in the `ParsedProviderCall` (`vscode-cline-parser.ts:157`). Subsequent user turns are accounted but not surfaced.
+- Only the **first** user message is emitted as `userMessage` in the `ParsedProviderCall`. Subsequent user turns are accounted but not surfaced.
- The model regex looks inside content blocks, not at top-level fields. Some Cline-derivative extensions emit the model elsewhere; if you add support for one, branch on extension ID rather than rewriting the regex.
## When fixing a bug here
-1. A change here ripples to **both** KiloCode and Roo Code. Run both test files (`tests/providers/kilo-code.test.ts` and `tests/providers/roo-code.test.ts`) before opening a PR.
+1. A change here ripples to IBM Bob, KiloCode, and Roo Code. Run all three provider test files before opening a PR.
2. If you find that one of the two extensions emits a different shape, branch on the extension ID parameter that the discovery function already takes; do not duplicate the parser.
-3. If you add support for a third Cline-derivative extension, register it as a thin wrapper file in the same shape as `kilo-code.ts` and `roo-code.ts`.
+3. If you add support for another Cline-family task store, register it as a thin wrapper file in the same shape as `ibm-bob.ts`, `kilo-code.ts`, and `roo-code.ts`.
diff --git a/mac/Scripts/package-app.sh b/mac/Scripts/package-app.sh
index 5de94ed..ee0dc06 100755
--- a/mac/Scripts/package-app.sh
+++ b/mac/Scripts/package-app.sh
@@ -96,7 +96,7 @@ codesign --verify --deep --strict "${BUNDLE}" 2>/dev/null || echo " (signature
ZIP_NAME="CodeBurnMenubar-${VERSION}.zip"
ZIP_PATH="${DIST_DIR}/${ZIP_NAME}"
echo "▸ Packaging ${ZIP_NAME}..."
-(cd "${DIST_DIR}" && /usr/bin/ditto -c -k --keepParent "${BUNDLE_NAME}" "${ZIP_NAME}")
+(cd "${DIST_DIR}" && COPYFILE_DISABLE=1 /usr/bin/ditto -c -k --norsrc --keepParent "${BUNDLE_NAME}" "${ZIP_NAME}")
CHECKSUM_NAME="${ZIP_NAME}.sha256"
CHECKSUM_PATH="${DIST_DIR}/${CHECKSUM_NAME}"
diff --git a/mac/Sources/CodeBurnMenubar/AppStore.swift b/mac/Sources/CodeBurnMenubar/AppStore.swift
index 00b27e8..ec5fdfa 100644
--- a/mac/Sources/CodeBurnMenubar/AppStore.swift
+++ b/mac/Sources/CodeBurnMenubar/AppStore.swift
@@ -140,6 +140,17 @@ final class AppStore {
inFlightKeys.removeAll()
}
+ func resetRefreshState(clearCache: Bool = false) {
+ switchTask?.cancel()
+ switchTask = nil
+ resetLoadingState()
+ attemptedKeys.removeAll()
+ lastErrorByKey.removeAll()
+ if clearCache {
+ cache.removeAll()
+ }
+ }
+
private let loadingWatchdogSeconds: TimeInterval = 60
@discardableResult
@@ -725,6 +736,7 @@ enum ProviderFilter: String, CaseIterable, Identifiable {
case copilot = "Copilot"
case droid = "Droid"
case gemini = "Gemini"
+ case ibmBob = "IBM Bob"
case kiro = "Kiro"
case kiloCode = "KiloCode"
case openclaw = "OpenClaw"
@@ -742,6 +754,7 @@ enum ProviderFilter: String, CaseIterable, Identifiable {
case .cursor: ["cursor", "cursor agent"]
case .rooCode: ["roo-code", "roo code"]
case .kiloCode: ["kilo-code", "kilocode"]
+ case .ibmBob: ["ibm-bob", "ibm bob"]
case .openclaw: ["openclaw"]
default: [rawValue.lowercased()]
}
@@ -756,6 +769,7 @@ enum ProviderFilter: String, CaseIterable, Identifiable {
case .copilot: "copilot"
case .droid: "droid"
case .gemini: "gemini"
+ case .ibmBob: "ibm-bob"
case .kiloCode: "kilo-code"
case .kiro: "kiro"
case .openclaw: "openclaw"
diff --git a/mac/Sources/CodeBurnMenubar/CodeBurnApp.swift b/mac/Sources/CodeBurnMenubar/CodeBurnApp.swift
index 5868258..a58d044 100644
--- a/mac/Sources/CodeBurnMenubar/CodeBurnApp.swift
+++ b/mac/Sources/CodeBurnMenubar/CodeBurnApp.swift
@@ -6,6 +6,7 @@ private let refreshIntervalSeconds: UInt64 = 30
private let nanosPerSecond: UInt64 = 1_000_000_000
private let refreshIntervalNanos: UInt64 = refreshIntervalSeconds * nanosPerSecond
private let forceRefreshWatchdogSeconds: TimeInterval = 90
+private let interactiveQuotaRefreshFloorSeconds: TimeInterval = 30
private let statusItemWidth: CGFloat = NSStatusItem.variableLength
private let popoverWidth: CGFloat = 360
private let popoverHeight: CGFloat = 660
@@ -39,6 +40,8 @@ final class AppDelegate: NSObject, NSApplicationDelegate, NSPopoverDelegate {
private var forceRefreshTask: Task?
private var forceRefreshStartedAt: Date?
private var forceRefreshGeneration: UInt64 = 0
+ private var manualRefreshTask: Task?
+ private var manualRefreshGeneration: UInt64 = 0
func applicationWillFinishLaunching(_ notification: Notification) {
// Set accessory policy before the app's focus chain forms. On macOS Tahoe
@@ -95,6 +98,10 @@ final class AppDelegate: NSObject, NSApplicationDelegate, NSPopoverDelegate {
self?.forceRefreshTask = nil
self?.forceRefreshStartedAt = nil
self?.forceRefreshGeneration &+= 1
+ self?.manualRefreshTask?.cancel()
+ self?.manualRefreshTask = nil
+ self?.manualRefreshGeneration &+= 1
+ self?.store.resetLoadingState()
self?.refreshLoopTask?.cancel()
self?.refreshLoopTask = nil
}
@@ -110,9 +117,7 @@ final class AppDelegate: NSObject, NSApplicationDelegate, NSPopoverDelegate {
queue: .main
) { [weak self] _ in
Task { @MainActor in
- self?.store.resetLoadingState()
- self?.forceRefresh()
- if self?.refreshLoopTask == nil { self?.startRefreshLoop() }
+ self?.recoverRefreshPipelineAfterInterruption(resetLoading: true)
}
}
@@ -121,7 +126,9 @@ final class AppDelegate: NSObject, NSApplicationDelegate, NSPopoverDelegate {
object: nil,
queue: .main
) { [weak self] _ in
- Task { @MainActor in self?.forceRefresh() }
+ Task { @MainActor in
+ self?.recoverRefreshPipelineAfterInterruption(resetLoading: true)
+ }
}
}
@@ -131,10 +138,24 @@ final class AppDelegate: NSObject, NSApplicationDelegate, NSPopoverDelegate {
object: nil,
queue: .main
) { [weak self] _ in
- Task { @MainActor in self?.forceRefresh() }
+ Task { @MainActor in
+ self?.recoverRefreshPipelineAfterInterruption(resetLoading: false)
+ }
}
}
+ private func recoverRefreshPipelineAfterInterruption(resetLoading: Bool) {
+ if resetLoading {
+ store.resetLoadingState()
+ } else {
+ _ = store.clearStaleLoadingIfNeeded()
+ }
+ if refreshLoopTask == nil {
+ startRefreshLoop()
+ }
+ forceRefresh()
+ }
+
private func installLaunchAgentIfNeeded() {
let fm = FileManager.default
let agentName = "com.codeburn.refresh.plist"
@@ -232,6 +253,7 @@ final class AppDelegate: NSObject, NSApplicationDelegate, NSPopoverDelegate {
private func forceRefresh() {
let now = Date()
_ = clearStaleForceRefreshIfNeeded(now: now)
+ guard forceRefreshTask == nil else { return }
guard now.timeIntervalSince(lastRefreshTime) > 5 else { return }
lastRefreshTime = now
forceRefreshStartedAt = now
@@ -241,7 +263,8 @@ final class AppDelegate: NSObject, NSApplicationDelegate, NSPopoverDelegate {
forceRefreshTask = Task {
async let main: Void = store.refresh(includeOptimize: false, force: true, showLoading: true)
async let today: Void = store.refreshQuietly(period: .today)
- _ = await (main, today)
+ async let quotas: Bool = refreshLiveQuotaProgressIfDue()
+ _ = await (main, today, quotas)
refreshStatusButton()
await MainActor.run { [weak self] in
guard let self, self.forceRefreshGeneration == generation else { return }
@@ -275,6 +298,51 @@ final class AppDelegate: NSObject, NSApplicationDelegate, NSPopoverDelegate {
}
fileprivate var lastSubscriptionRefreshAt: Date?
+ fileprivate var lastCodexRefreshAt: Date?
+
+ @discardableResult
+ private func refreshLiveQuotaProgressIfDue(force: Bool = false) async -> Bool {
+ let cadence = SubscriptionRefreshCadence.current
+ if !force && cadence == .manual { return false }
+
+ let now = Date()
+ let threshold = force ? 0 : TimeInterval(cadence.rawValue)
+ let shouldRefreshClaude = force || now.timeIntervalSince(lastSubscriptionRefreshAt ?? .distantPast) >= threshold
+ let shouldRefreshCodex = force || now.timeIntervalSince(lastCodexRefreshAt ?? .distantPast) >= threshold
+ guard shouldRefreshClaude || shouldRefreshCodex else { return false }
+
+ switch (shouldRefreshClaude, shouldRefreshCodex) {
+ case (true, true):
+ async let claude = store.refreshSubscriptionReportingSuccess()
+ async let codex = store.refreshCodexReportingSuccess()
+ if await claude { lastSubscriptionRefreshAt = Date() }
+ if await codex { lastCodexRefreshAt = Date() }
+ case (true, false):
+ if await store.refreshSubscriptionReportingSuccess() {
+ lastSubscriptionRefreshAt = Date()
+ }
+ case (false, true):
+ if await store.refreshCodexReportingSuccess() {
+ lastCodexRefreshAt = Date()
+ }
+ case (false, false):
+ break
+ }
+ return true
+ }
+
+ private func refreshLiveQuotaProgressForPopoverOpen() {
+ let now = Date()
+ let claudeElapsed = now.timeIntervalSince(lastSubscriptionRefreshAt ?? .distantPast)
+ let codexElapsed = now.timeIntervalSince(lastCodexRefreshAt ?? .distantPast)
+ guard claudeElapsed >= interactiveQuotaRefreshFloorSeconds ||
+ codexElapsed >= interactiveQuotaRefreshFloorSeconds else { return }
+
+ Task { [weak self] in
+ guard let self else { return }
+ _ = await self.refreshLiveQuotaProgressIfDue(force: true)
+ }
+ }
private func startRefreshLoop() {
refreshLoopTask?.cancel()
@@ -282,10 +350,7 @@ final class AppDelegate: NSObject, NSApplicationDelegate, NSPopoverDelegate {
// Provider refreshes only run when the user has explicitly connected.
// Each refresh is a no-op until its corresponding bootstrap flag is set.
if let self {
- async let claude = self.store.refreshSubscriptionReportingSuccess()
- async let codex = self.store.refreshCodexReportingSuccess()
- if await claude { self.lastSubscriptionRefreshAt = Date() }
- if await codex { self.lastCodexRefreshAt = Date() }
+ await self.refreshLiveQuotaProgressIfDue(force: true)
}
while !Task.isCancelled {
guard let self else { return }
@@ -311,39 +376,50 @@ final class AppDelegate: NSObject, NSApplicationDelegate, NSPopoverDelegate {
// (not last attempt) so an intermittent failure doesn't reset
// the timer. Each provider has its own anchor so a Codex 429
// doesn't delay a due Claude refresh.
- let cadence = SubscriptionRefreshCadence.current
- if cadence != .manual {
- let claudeElapsed = Date().timeIntervalSince(self.lastSubscriptionRefreshAt ?? .distantPast)
- if claudeElapsed >= TimeInterval(cadence.rawValue) {
- let succeeded = await self.store.refreshSubscriptionReportingSuccess()
- if succeeded { self.lastSubscriptionRefreshAt = Date() }
- }
- let codexElapsed = Date().timeIntervalSince(self.lastCodexRefreshAt ?? .distantPast)
- if codexElapsed >= TimeInterval(cadence.rawValue) {
- let succeeded = await self.store.refreshCodexReportingSuccess()
- if succeeded { self.lastCodexRefreshAt = Date() }
- }
- }
+ await self.refreshLiveQuotaProgressIfDue()
try? await Task.sleep(nanoseconds: refreshIntervalNanos)
}
}
}
- fileprivate var lastCodexRefreshAt: Date?
-
@MainActor
func refreshSubscriptionNow() {
- Task { [weak self] in
+ manualRefreshTask?.cancel()
+ manualRefreshGeneration &+= 1
+ let generation = manualRefreshGeneration
+ forceRefreshTask?.cancel()
+ forceRefreshTask = nil
+ forceRefreshStartedAt = nil
+ forceRefreshGeneration &+= 1
+ pendingRefreshWork?.cancel()
+ pendingRefreshWork = nil
+ refreshLoopTask?.cancel()
+ refreshLoopTask = nil
+ store.resetRefreshState(clearCache: true)
+ lastRefreshTime = .distantPast
+ refreshStatusButton()
+
+ manualRefreshTask = Task { [weak self] in
guard let self else { return }
// "Refresh Now" should refresh the menubar payload AND every
- // connected provider's live quota — the user's intent is "make
+ // connected provider's live quota. The user's intent is "make
// this match reality right now."
+ let needsTodayTotal = self.store.selectedPeriod != .today || self.store.selectedProvider != .all
async let payload: Void = self.store.refresh(includeOptimize: false, force: true, showLoading: true)
- async let claude: Bool = self.store.refreshSubscriptionReportingSuccess()
- async let codex: Bool = self.store.refreshCodexReportingSuccess()
+ async let quotas: Bool = self.refreshLiveQuotaProgressIfDue(force: true)
+ if needsTodayTotal {
+ await self.store.refreshQuietly(period: .today)
+ }
_ = await payload
- if await claude { self.lastSubscriptionRefreshAt = Date() }
- if await codex { self.lastCodexRefreshAt = Date() }
+ guard self.manualRefreshGeneration == generation, !Task.isCancelled else { return }
+ self.lastRefreshTime = Date()
+ self.refreshStatusButton()
+ _ = await quotas
+ guard self.manualRefreshGeneration == generation, !Task.isCancelled else { return }
+ self.manualRefreshTask = nil
+ if self.refreshLoopTask == nil {
+ self.startRefreshLoop()
+ }
}
}
@@ -541,6 +617,7 @@ final class AppDelegate: NSObject, NSApplicationDelegate, NSPopoverDelegate {
window.collectionBehavior.insert(.canJoinAllSpaces)
window.makeKeyAndOrderFront(nil)
}
+ refreshLiveQuotaProgressForPopoverOpen()
}
}
diff --git a/mac/Sources/CodeBurnMenubar/Data/UpdateChecker.swift b/mac/Sources/CodeBurnMenubar/Data/UpdateChecker.swift
index 6ad0a90..ce575b6 100644
--- a/mac/Sources/CodeBurnMenubar/Data/UpdateChecker.swift
+++ b/mac/Sources/CodeBurnMenubar/Data/UpdateChecker.swift
@@ -46,7 +46,7 @@ final class UpdateChecker {
let (data, _) = try await URLSession.shared.data(for: request)
let release = try JSONDecoder().decode(GitHubRelease.self, from: data)
guard let asset = release.assets.first(where: {
- $0.name.hasPrefix("CodeBurnMenubar-") && $0.name.hasSuffix(".zip")
+ $0.name.hasPrefix("CodeBurnMenubar-v") && $0.name.hasSuffix(".zip")
}) else { return }
let version = asset.name
diff --git a/mac/Sources/CodeBurnMenubar/Views/AgentTabStrip.swift b/mac/Sources/CodeBurnMenubar/Views/AgentTabStrip.swift
index 6561cc9..df47c46 100644
--- a/mac/Sources/CodeBurnMenubar/Views/AgentTabStrip.swift
+++ b/mac/Sources/CodeBurnMenubar/Views/AgentTabStrip.swift
@@ -345,6 +345,7 @@ extension ProviderFilter {
case .copilot: return Color(red: 0x6D/255.0, green: 0x8F/255.0, blue: 0xA6/255.0)
case .droid: return Color(red: 0x7C/255.0, green: 0x3A/255.0, blue: 0xED/255.0)
case .gemini: return Color(red: 0x44/255.0, green: 0x85/255.0, blue: 0xF4/255.0)
+ case .ibmBob: return Color(red: 0x0F/255.0, green: 0x62/255.0, blue: 0xFE/255.0)
case .kiloCode: return Color(red: 0x00/255.0, green: 0x96/255.0, blue: 0x88/255.0)
case .kiro: return Color(red: 0x4A/255.0, green: 0x9E/255.0, blue: 0xC4/255.0)
case .openclaw: return Color(red: 0xDA/255.0, green: 0x70/255.0, blue: 0x56/255.0)
diff --git a/package.json b/package.json
index a58098d..b831b30 100644
--- a/package.json
+++ b/package.json
@@ -21,6 +21,7 @@
"claude-code",
"cursor",
"codex",
+ "ibm-bob",
"opencode",
"pi",
"ai-coding",
diff --git a/src/dashboard.tsx b/src/dashboard.tsx
index b46dbcc..e666b18 100644
--- a/src/dashboard.tsx
+++ b/src/dashboard.tsx
@@ -52,6 +52,7 @@ const PROVIDER_COLORS: Record = {
claude: '#FF8C42',
codex: '#5BF5A0',
cursor: '#00B4D8',
+ 'ibm-bob': '#0F62FE',
opencode: '#A78BFA',
pi: '#F472B6',
all: '#FF8C42',
@@ -513,6 +514,7 @@ const PROVIDER_DISPLAY_NAMES: Record = {
claude: 'Claude',
codex: 'Codex',
cursor: 'Cursor',
+ 'ibm-bob': 'IBM Bob',
opencode: 'OpenCode',
pi: 'Pi',
}
diff --git a/src/menubar-installer.ts b/src/menubar-installer.ts
index 397a81c..051c12c 100644
--- a/src/menubar-installer.ts
+++ b/src/menubar-installer.ts
@@ -11,17 +11,28 @@ import { Readable } from 'node:stream'
/// newest tagged release; we filter its assets list for our zipped .app bundle.
const RELEASE_API = 'https://api.github.com/repos/getagentseal/codeburn/releases/latest'
const APP_BUNDLE_NAME = 'CodeBurnMenubar.app'
-const ASSET_PATTERN = /^CodeBurnMenubar-.*\.zip$/
-const CHECKSUM_PATTERN = /^CodeBurnMenubar-.*\.zip\.sha256$/
+const VERSIONED_ASSET_PATTERN = /^CodeBurnMenubar-v.+\.zip$/
const APP_PROCESS_NAME = 'CodeBurnMenubar'
const SUPPORTED_OS = 'darwin'
const MIN_MACOS_MAJOR = 14
export type InstallResult = { installedPath: string; launched: boolean }
-type ReleaseAsset = { name: string; browser_download_url: string }
-type ReleaseResponse = { tag_name: string; assets: ReleaseAsset[] }
-type ResolvedAssets = { zip: ReleaseAsset; checksum: ReleaseAsset | null }
+export type ReleaseAsset = { name: string; browser_download_url: string }
+export type ReleaseResponse = { tag_name: string; assets: ReleaseAsset[] }
+export type ResolvedAssets = { zip: ReleaseAsset; checksum: ReleaseAsset | null }
+
+export function resolveMenubarReleaseAssets(release: ReleaseResponse): ResolvedAssets {
+ const zip = release.assets.find(a => VERSIONED_ASSET_PATTERN.test(a.name))
+ if (!zip) {
+ throw new Error(
+ `No ${APP_BUNDLE_NAME} versioned zip found in release ${release.tag_name}. ` +
+ `Check https://github.com/getagentseal/codeburn/releases.`
+ )
+ }
+ const checksum = release.assets.find(a => a.name === `${zip.name}.sha256`) ?? null
+ return { zip, checksum }
+}
function userApplicationsDir(): string {
return join(homedir(), 'Applications')
@@ -71,15 +82,7 @@ async function fetchLatestReleaseAssets(): Promise {
throw new Error(`GitHub release lookup failed: HTTP ${response.status}`)
}
const body = await response.json() as ReleaseResponse
- const zip = body.assets.find(a => ASSET_PATTERN.test(a.name))
- if (!zip) {
- throw new Error(
- `No ${APP_BUNDLE_NAME} zip found in release ${body.tag_name}. ` +
- `Check https://github.com/getagentseal/codeburn/releases.`
- )
- }
- const checksum = body.assets.find(a => CHECKSUM_PATTERN.test(a.name)) ?? null
- return { zip, checksum }
+ return resolveMenubarReleaseAssets(body)
}
async function verifyChecksum(archivePath: string, checksumUrl: string): Promise {
@@ -179,7 +182,7 @@ export async function installMenubarApp(options: { force?: boolean } = {}): Prom
}
console.log('Unpacking...')
- await runCommand('/usr/bin/unzip', ['-q', archivePath, '-d', stagingDir])
+ await runCommand('/usr/bin/ditto', ['-x', '-k', archivePath, stagingDir])
const unpackedApp = join(stagingDir, APP_BUNDLE_NAME)
if (!(await exists(unpackedApp))) {
diff --git a/src/models.ts b/src/models.ts
index 24c85b1..1070cdb 100644
--- a/src/models.ts
+++ b/src/models.ts
@@ -167,6 +167,7 @@ const BUILTIN_ALIASES: Record = {
'copilot-auto': 'claude-sonnet-4-5',
'copilot-openai-auto': 'gpt-5.3-codex',
'copilot-anthropic-auto': 'claude-sonnet-4-5',
+ 'ibm-bob-auto': 'claude-sonnet-4-5',
'kiro-auto': 'claude-sonnet-4-5',
'cline-auto': 'claude-sonnet-4-5',
'openclaw-auto': 'claude-sonnet-4-5',
@@ -357,6 +358,7 @@ const autoModelNames: Record = {
'copilot-auto': 'Copilot (auto)',
'copilot-openai-auto': 'Copilot (OpenAI)',
'copilot-anthropic-auto': 'Copilot (Anthropic)',
+ 'ibm-bob-auto': 'IBM Bob (auto)',
'kiro-auto': 'Kiro (auto)',
'cline-auto': 'Cline (auto)',
'openclaw-auto': 'OpenClaw (auto)',
diff --git a/src/parser.ts b/src/parser.ts
index 8ef8b0d..3bb602e 100644
--- a/src/parser.ts
+++ b/src/parser.ts
@@ -574,7 +574,7 @@ async function parseProviderSources(
const provider = await getProvider(providerName)
if (!provider) return []
- const sessionMap = new Map()
+ const sessionMap = new Map()
try {
for (const source of sources) {
@@ -598,13 +598,15 @@ async function parseProviderSources(
const turn = providerCallToTurn(call)
const classified = classifyTurn(turn)
- const key = `${providerName}:${call.sessionId}:${source.project}`
+ const project = call.project ?? source.project
+ const key = `${providerName}:${call.sessionId}:${project}`
const existing = sessionMap.get(key)
if (existing) {
existing.turns.push(classified)
+ if (!existing.projectPath && call.projectPath) existing.projectPath = call.projectPath
} else {
- sessionMap.set(key, { project: source.project, turns: [classified] })
+ sessionMap.set(key, { project, projectPath: call.projectPath, turns: [classified] })
}
}
}
@@ -616,22 +618,26 @@ async function parseProviderSources(
}
}
- const projectMap = new Map()
- for (const [key, { project, turns }] of sessionMap) {
+ const projectMap = new Map()
+ for (const [key, { project, projectPath, turns }] of sessionMap) {
const sessionId = key.split(':')[1] ?? key
const session = buildSessionSummary(sessionId, project, turns)
if (session.apiCalls > 0) {
- const existing = projectMap.get(project) ?? []
- existing.push(session)
- projectMap.set(project, existing)
+ const existing = projectMap.get(project)
+ if (existing) {
+ existing.sessions.push(session)
+ if (!existing.projectPath && projectPath) existing.projectPath = projectPath
+ } else {
+ projectMap.set(project, { projectPath, sessions: [session] })
+ }
}
}
const projects: ProjectSummary[] = []
- for (const [dirName, sessions] of projectMap) {
+ for (const [dirName, { projectPath, sessions }] of projectMap) {
projects.push({
project: dirName,
- projectPath: unsanitizePath(dirName),
+ projectPath: projectPath ?? unsanitizePath(dirName),
sessions,
totalCostUSD: sessions.reduce((s, sess) => s + sess.totalCostUSD, 0),
totalApiCalls: sessions.reduce((s, sess) => s + sess.apiCalls, 0),
diff --git a/src/providers/ibm-bob.ts b/src/providers/ibm-bob.ts
new file mode 100644
index 0000000..5aec0f6
--- /dev/null
+++ b/src/providers/ibm-bob.ts
@@ -0,0 +1,59 @@
+import { join } from 'path'
+import { homedir } from 'os'
+
+import { getShortModelName } from '../models.js'
+import { discoverClineTasksInBaseDirs, createClineParser } from './vscode-cline-parser.js'
+import type { Provider, SessionSource, SessionParser } from './types.js'
+
+const PROVIDER_NAME = 'ibm-bob'
+const DISPLAY_NAME = 'IBM Bob'
+const EXTENSION_ID = 'ibm.bob-code'
+const FALLBACK_MODEL = 'ibm-bob-auto'
+
+export function getIBMBobGlobalStorageDirs(): string[] {
+ const home = homedir()
+ if (process.platform === 'darwin') {
+ return [
+ join(home, 'Library', 'Application Support', 'IBM Bob', 'User', 'globalStorage', EXTENSION_ID),
+ join(home, 'Library', 'Application Support', 'Bob-IDE', 'User', 'globalStorage', EXTENSION_ID),
+ ]
+ }
+ if (process.platform === 'win32') {
+ const appData = process.env['APPDATA'] ?? join(home, 'AppData', 'Roaming')
+ return [
+ join(appData, 'IBM Bob', 'User', 'globalStorage', EXTENSION_ID),
+ join(appData, 'Bob-IDE', 'User', 'globalStorage', EXTENSION_ID),
+ ]
+ }
+ const configHome = process.env['XDG_CONFIG_HOME'] ?? join(home, '.config')
+ return [
+ join(configHome, 'IBM Bob', 'User', 'globalStorage', EXTENSION_ID),
+ join(configHome, 'Bob-IDE', 'User', 'globalStorage', EXTENSION_ID),
+ ]
+}
+
+export function createIBMBobProvider(overrideDir?: string): Provider {
+ return {
+ name: PROVIDER_NAME,
+ displayName: DISPLAY_NAME,
+
+ modelDisplayName(model: string): string {
+ return getShortModelName(model)
+ },
+
+ toolDisplayName(rawTool: string): string {
+ return rawTool
+ },
+
+ async discoverSessions(): Promise {
+ const dirs = overrideDir ? [overrideDir] : getIBMBobGlobalStorageDirs()
+ return discoverClineTasksInBaseDirs(dirs, PROVIDER_NAME, DISPLAY_NAME)
+ },
+
+ createSessionParser(source: SessionSource, seenKeys: Set): SessionParser {
+ return createClineParser(source, seenKeys, PROVIDER_NAME, FALLBACK_MODEL)
+ },
+ }
+}
+
+export const ibmBob = createIBMBobProvider()
diff --git a/src/providers/index.ts b/src/providers/index.ts
index 38ed490..551d3a2 100644
--- a/src/providers/index.ts
+++ b/src/providers/index.ts
@@ -3,6 +3,7 @@ import { codex } from './codex.js'
import { copilot } from './copilot.js'
import { droid } from './droid.js'
import { gemini } from './gemini.js'
+import { ibmBob } from './ibm-bob.js'
import { kiloCode } from './kilo-code.js'
import { kiro } from './kiro.js'
import { openclaw } from './openclaw.js'
@@ -101,7 +102,7 @@ async function loadCrush(): Promise {
}
}
-const coreProviders: Provider[] = [claude, codex, copilot, droid, gemini, kiloCode, kiro, openclaw, pi, omp, qwen, rooCode]
+const coreProviders: Provider[] = [claude, codex, copilot, droid, gemini, ibmBob, kiloCode, kiro, openclaw, pi, omp, qwen, rooCode]
export async function getAllProviders(): Promise {
const [ag, gs, cursor, opencode, cursorAgent, crush] = await Promise.all([loadAntigravity(), loadGoose(), loadCursor(), loadOpenCode(), loadCursorAgent(), loadCrush()])
diff --git a/src/providers/types.ts b/src/providers/types.ts
index 4e9a98a..90d5e1c 100644
--- a/src/providers/types.ts
+++ b/src/providers/types.ts
@@ -27,6 +27,8 @@ export type ParsedProviderCall = {
deduplicationKey: string
userMessage: string
sessionId: string
+ project?: string
+ projectPath?: string
}
export type Provider = {
diff --git a/src/providers/vscode-cline-parser.ts b/src/providers/vscode-cline-parser.ts
index d1d26c0..ffad939 100644
--- a/src/providers/vscode-cline-parser.ts
+++ b/src/providers/vscode-cline-parser.ts
@@ -24,6 +24,23 @@ export function getVSCodeGlobalStoragePath(extensionId: string): string {
export async function discoverClineTasks(extensionId: string, providerName: string, displayName: string, overrideDir?: string): Promise {
const baseDir = overrideDir ?? getVSCodeGlobalStoragePath(extensionId)
+ return discoverClineTasksInBaseDirs([baseDir], providerName, displayName)
+}
+
+export async function discoverClineTasksInBaseDirs(baseDirs: string[], providerName: string, displayName: string): Promise {
+ const sources: SessionSource[] = []
+ const seen = new Set()
+ for (const baseDir of baseDirs) {
+ for (const source of await discoverClineTasksInBaseDir(baseDir, providerName, displayName)) {
+ if (seen.has(source.path)) continue
+ seen.add(source.path)
+ sources.push(source)
+ }
+ }
+ return sources
+}
+
+async function discoverClineTasksInBaseDir(baseDir: string, providerName: string, displayName: string): Promise {
const tasksDir = join(baseDir, 'tasks')
const sources: SessionSource[] = []
@@ -50,28 +67,43 @@ export async function discoverClineTasks(extensionId: string, providerName: stri
}
const MODEL_TAG_RE = /([^<]+)<\/model>/
+const WORKSPACE_DIR_RE = /Current Workspace Directory \(([^)]+)\)/
-function extractModelFromHistory(taskDir: string): Promise {
+type HistoryMeta = { model: string; workspace: string | null }
+
+function extractHistoryMeta(taskDir: string, fallbackModel: string): Promise {
return readFile(join(taskDir, 'api_conversation_history.json'), 'utf-8')
.then(raw => {
const msgs = JSON.parse(raw) as Array<{ role?: string; content?: Array<{ text?: string }> }>
- if (!Array.isArray(msgs)) return 'cline-auto'
+ if (!Array.isArray(msgs)) return { model: fallbackModel, workspace: null }
+ let model: string | null = null
+ let workspace: string | null = null
for (const msg of msgs) {
if (msg.role !== 'user' || !Array.isArray(msg.content)) continue
for (const block of msg.content) {
- const match = typeof block.text === 'string' && MODEL_TAG_RE.exec(block.text)
- if (match) {
- const raw = match[1]
- return raw.includes('/') ? raw.split('/').pop()! : raw
+ if (typeof block.text !== 'string') continue
+ if (!model) {
+ const mm = MODEL_TAG_RE.exec(block.text)
+ if (mm) model = mm[1].includes('/') ? mm[1].split('/').pop()! : mm[1]
}
+ if (!workspace) {
+ const wm = WORKSPACE_DIR_RE.exec(block.text)
+ if (wm) workspace = wm[1]
+ }
+ if (model && workspace) break
}
+ if (model && workspace) break
}
- return 'cline-auto'
+ return { model: model ?? fallbackModel, workspace }
})
- .catch(() => 'cline-auto')
+ .catch(() => ({ model: fallbackModel, workspace: null }))
}
-export function createClineParser(source: SessionSource, seenKeys: Set, providerName: string): SessionParser {
+function workspaceToProject(workspace: string): string {
+ return basename(workspace) || workspace
+}
+
+export function createClineParser(source: SessionSource, seenKeys: Set, providerName: string, fallbackModel = 'cline-auto'): SessionParser {
return {
async *parse(): AsyncGenerator {
const taskDir = source.path
@@ -93,7 +125,10 @@ export function createClineParser(source: SessionSource, seenKeys: Set,
if (!Array.isArray(uiMessages)) return
- const model = await extractModelFromHistory(taskDir)
+ const meta = await extractHistoryMeta(taskDir, fallbackModel)
+ const model = meta.model
+ const project = meta.workspace ? workspaceToProject(meta.workspace) : undefined
+ const projectPath = meta.workspace ?? undefined
let userMessage = ''
for (const msg of uiMessages) {
@@ -156,6 +191,8 @@ export function createClineParser(source: SessionSource, seenKeys: Set,
deduplicationKey: dedupKey,
userMessage: index === 0 ? userMessage : '',
sessionId: taskId,
+ project,
+ projectPath,
}
}
},
diff --git a/tests/menubar-installer.test.ts b/tests/menubar-installer.test.ts
new file mode 100644
index 0000000..44f73cc
--- /dev/null
+++ b/tests/menubar-installer.test.ts
@@ -0,0 +1,37 @@
+import { describe, expect, it } from 'vitest'
+import { resolveMenubarReleaseAssets, type ReleaseResponse } from '../src/menubar-installer.js'
+
+function asset(name: string) {
+ return { name, browser_download_url: `https://example.test/${name}` }
+}
+
+describe('resolveMenubarReleaseAssets', () => {
+ it('ignores dev zips and pairs the checksum with the versioned zip', () => {
+ const release: ReleaseResponse = {
+ tag_name: 'mac-v0.9.8',
+ assets: [
+ asset('CodeBurnMenubar-dev.zip'),
+ asset('CodeBurnMenubar-dev.zip.sha256'),
+ asset('CodeBurnMenubar-v0.9.8.zip'),
+ asset('CodeBurnMenubar-v0.9.8.zip.sha256'),
+ ],
+ }
+
+ const resolved = resolveMenubarReleaseAssets(release)
+
+ expect(resolved.zip.name).toBe('CodeBurnMenubar-v0.9.8.zip')
+ expect(resolved.checksum?.name).toBe('CodeBurnMenubar-v0.9.8.zip.sha256')
+ })
+
+ it('fails when a release only contains dev assets', () => {
+ const release: ReleaseResponse = {
+ tag_name: 'mac-v0.9.8',
+ assets: [
+ asset('CodeBurnMenubar-dev.zip'),
+ asset('CodeBurnMenubar-dev.zip.sha256'),
+ ],
+ }
+
+ expect(() => resolveMenubarReleaseAssets(release)).toThrow(/versioned zip/)
+ })
+})
diff --git a/tests/provider-registry.test.ts b/tests/provider-registry.test.ts
index 4497946..2dc1dfc 100644
--- a/tests/provider-registry.test.ts
+++ b/tests/provider-registry.test.ts
@@ -3,7 +3,7 @@ import { providers, getAllProviders } from '../src/providers/index.js'
describe('provider registry', () => {
it('has core providers registered synchronously', () => {
- expect(providers.map(p => p.name)).toEqual(['claude', 'codex', 'copilot', 'droid', 'gemini', 'kilo-code', 'kiro', 'openclaw', 'pi', 'omp', 'qwen', 'roo-code'])
+ expect(providers.map(p => p.name)).toEqual(['claude', 'codex', 'copilot', 'droid', 'gemini', 'ibm-bob', 'kilo-code', 'kiro', 'openclaw', 'pi', 'omp', 'qwen', 'roo-code'])
})
it('includes sqlite providers after async load', async () => {
diff --git a/tests/providers/ibm-bob.test.ts b/tests/providers/ibm-bob.test.ts
new file mode 100644
index 0000000..d61f92e
--- /dev/null
+++ b/tests/providers/ibm-bob.test.ts
@@ -0,0 +1,164 @@
+import { describe, it, expect, beforeEach, afterEach } from 'vitest'
+import { mkdtemp, mkdir, writeFile, rm } from 'fs/promises'
+import { join } from 'path'
+import { tmpdir } from 'os'
+
+import { ibmBob, createIBMBobProvider } from '../../src/providers/ibm-bob.js'
+import type { ParsedProviderCall } from '../../src/providers/types.js'
+
+let tmpDir: string
+
+function makeUiMessages(opts: {
+ tokensIn?: number
+ tokensOut?: number
+ cacheReads?: number
+ cacheWrites?: number
+ cost?: number
+ userMessage?: string
+ ts?: number
+}): string {
+ const messages: unknown[] = []
+
+ if (opts.userMessage) {
+ messages.push({ type: 'say', say: 'user_feedback', text: opts.userMessage, ts: 1_700_000_000_000 })
+ }
+
+ const apiData: Record = {
+ tokensIn: opts.tokensIn ?? 100,
+ tokensOut: opts.tokensOut ?? 50,
+ cacheReads: opts.cacheReads ?? 0,
+ cacheWrites: opts.cacheWrites ?? 0,
+ }
+ if (opts.cost !== undefined) apiData.cost = opts.cost
+
+ messages.push({
+ type: 'say',
+ say: 'api_req_started',
+ text: JSON.stringify(apiData),
+ ts: opts.ts ?? 1_700_000_001_000,
+ })
+
+ return JSON.stringify(messages)
+}
+
+function makeApiHistory(model?: string): string {
+ const modelTag = model ? `${model}` : ''
+ return JSON.stringify([
+ { role: 'user', content: [{ type: 'text', text: `hello\n\n${modelTag}\n` }] },
+ { role: 'assistant', content: [{ type: 'text', text: 'response' }] },
+ ])
+}
+
+describe('ibm-bob provider - discovery and parsing', () => {
+ beforeEach(async () => {
+ tmpDir = await mkdtemp(join(tmpdir(), 'ibm-bob-test-'))
+ })
+
+ afterEach(async () => {
+ await rm(tmpDir, { recursive: true, force: true })
+ })
+
+ it('discovers IBM Bob task directories with ui_messages.json', async () => {
+ const task1 = join(tmpDir, 'tasks', 'task-a')
+ const task2 = join(tmpDir, 'tasks', 'task-b')
+ await mkdir(task1, { recursive: true })
+ await mkdir(task2, { recursive: true })
+ await writeFile(join(task1, 'ui_messages.json'), '[]')
+ await writeFile(join(task2, 'ui_messages.json'), '[]')
+
+ const provider = createIBMBobProvider(tmpDir)
+ const sessions = await provider.discoverSessions()
+
+ expect(sessions).toHaveLength(2)
+ expect(sessions.every(s => s.provider === 'ibm-bob')).toBe(true)
+ expect(sessions.every(s => s.project === 'IBM Bob')).toBe(true)
+ })
+
+ it('skips tasks without ui_messages.json', async () => {
+ const task = join(tmpDir, 'tasks', 'task-no-ui')
+ await mkdir(task, { recursive: true })
+ await writeFile(join(task, 'api_conversation_history.json'), '[]')
+
+ const provider = createIBMBobProvider(tmpDir)
+ const sessions = await provider.discoverSessions()
+
+ expect(sessions).toHaveLength(0)
+ })
+
+ it('parses token usage and provider cost from Bob ui messages', async () => {
+ const taskDir = join(tmpDir, 'tasks', 'task-001')
+ await mkdir(taskDir, { recursive: true })
+ await writeFile(join(taskDir, 'ui_messages.json'), makeUiMessages({
+ tokensIn: 250,
+ tokensOut: 125,
+ cacheReads: 60,
+ cacheWrites: 30,
+ cost: 0.08,
+ userMessage: 'modernize this class',
+ }))
+ await writeFile(join(taskDir, 'api_conversation_history.json'), makeApiHistory('anthropic/claude-sonnet-4-6'))
+
+ const source = { path: taskDir, project: 'IBM Bob', provider: 'ibm-bob' }
+ const calls: ParsedProviderCall[] = []
+ for await (const call of ibmBob.createSessionParser(source, new Set()).parse()) calls.push(call)
+
+ expect(calls).toHaveLength(1)
+ expect(calls[0]!).toMatchObject({
+ provider: 'ibm-bob',
+ model: 'claude-sonnet-4-6',
+ inputTokens: 250,
+ outputTokens: 125,
+ cacheReadInputTokens: 60,
+ cacheCreationInputTokens: 30,
+ costUSD: 0.08,
+ userMessage: 'modernize this class',
+ sessionId: 'task-001',
+ })
+ expect(calls[0]!.deduplicationKey).toBe('ibm-bob:task-001:0')
+ })
+
+ it('falls back to IBM Bob auto model when history has no model tag', async () => {
+ const taskDir = join(tmpDir, 'tasks', 'task-002')
+ await mkdir(taskDir, { recursive: true })
+ await writeFile(join(taskDir, 'ui_messages.json'), makeUiMessages({ tokensIn: 100, tokensOut: 50 }))
+ await writeFile(join(taskDir, 'api_conversation_history.json'), makeApiHistory())
+
+ const source = { path: taskDir, project: 'IBM Bob', provider: 'ibm-bob' }
+ const calls: ParsedProviderCall[] = []
+ for await (const call of ibmBob.createSessionParser(source, new Set()).parse()) calls.push(call)
+
+ expect(calls).toHaveLength(1)
+ expect(calls[0]!.model).toBe('ibm-bob-auto')
+ expect(calls[0]!.costUSD).toBeGreaterThan(0)
+ })
+
+ it('deduplicates across parser runs', async () => {
+ const taskDir = join(tmpDir, 'tasks', 'task-003')
+ await mkdir(taskDir, { recursive: true })
+ await writeFile(join(taskDir, 'ui_messages.json'), makeUiMessages({ tokensIn: 100, tokensOut: 50 }))
+
+ const source = { path: taskDir, project: 'IBM Bob', provider: 'ibm-bob' }
+ const seenKeys = new Set()
+
+ const calls1: ParsedProviderCall[] = []
+ for await (const call of ibmBob.createSessionParser(source, seenKeys).parse()) calls1.push(call)
+
+ const calls2: ParsedProviderCall[] = []
+ for await (const call of ibmBob.createSessionParser(source, seenKeys).parse()) calls2.push(call)
+
+ expect(calls1).toHaveLength(1)
+ expect(calls2).toHaveLength(0)
+ })
+})
+
+describe('ibm-bob provider - metadata', () => {
+ it('has correct name and displayName', () => {
+ expect(ibmBob.name).toBe('ibm-bob')
+ expect(ibmBob.displayName).toBe('IBM Bob')
+ })
+
+ it('uses shared short model display names', () => {
+ expect(ibmBob.modelDisplayName('ibm-bob-auto')).toBe('IBM Bob (auto)')
+ expect(ibmBob.modelDisplayName('claude-sonnet-4-6')).toBe('Sonnet 4.6')
+ })
+})