Skip to content

Opencode supports OpenAI SDK and Anthropic SDK API. Favorites ids are…#408

Open
eli-l wants to merge 5 commits into
Gitlawb:mainfrom
eli-l:feat/opencode-self-discovery
Open

Opencode supports OpenAI SDK and Anthropic SDK API. Favorites ids are…#408
eli-l wants to merge 5 commits into
Gitlawb:mainfrom
eli-l:feat/opencode-self-discovery

Conversation

@eli-l

@eli-l eli-l commented Jul 2, 2026

Copy link
Copy Markdown
Contributor

Summary

  1. Introduce distinction for Opencode Go models, as per documentation some of these shall be used via Anthropic-like connector (SDK) as per documentation https://opencode.ai/docs/go/#endpoints (Minimax and Qwen family)
    These are still preserved over OpenAI-like connector (SDK) as these are working (undocumented). It provides clear indication how models are used and at least - a built-in fallback in case side effect (undocumented) OpenAI-like access stop working for these models.
    /models API does not provide details of these, so filtering had to be added (hardcoded)

  2. Opencode provides /model endpoint, that allows self-discovery. It sounds reasonable to preserve a list of provided models and make these available to be chosen once discovered, rather than adding models one by one.
    Configuring models as provider-model configuration entry requires separate naming and prevents effective key management. Re-keying basically means all entries must be updated.

  3. Multiple providers might provide the same models. In such setup current approach to favourite entries becomes unambiguous. This PR introduces / format, making clear which provider will be used for the favourite model. Failing to pick proper provider may cause unwanted costs for the users.

Linked issue

#428
#429

Checklist

  • The linked issue already has the issue-approved label.
  • go build ./..., go vet ./..., and go test ./... pass locally.
  • gofmt clean.
  • Tests added/updated for the change (and run under -race where relevant).
  • UI changes include screenshots or a short recording where possible.

Summary by CodeRabbit

  • New Features
    • Persist the full live-discovered model list (including per-model overrides) across first-run and provider setup.
    • Added a new Anthropic-compatible provider option and expanded its curated model set.
    • Apply provider-specific model filtering to catalog and discovery results (including Anthropic-focused allowlisting).
  • Bug Fixes
    • If discovered-model persistence fails, setup continues (with a warning).
    • Model favorites and picker behavior are now provider-qualified; model identifiers render consistently.
    • Saved provider details refresh immediately after wizard completion.
  • Tests
    • Updated/added coverage for persistence, filtering, and picker/favorite rendering.

Copilot AI review requested due to automatic review settings July 2, 2026 19:18

Copilot AI left a comment

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Pull request overview

This PR expands provider/model handling to support additional OpenCode provider variants, makes model favorites provider-qualified (<provider>/<model>), and persists live-discovered model lists into config so they can be reused across sessions.

Changes:

  • Add new provider catalog entries for OpenCode-Go OpenAI SDK and Anthropic SDK variants, plus curated model lists and provider-specific model filtering.
  • Persist live-discovered models into config (ProviderProfile.Models) and thread discovered models through setup flows.
  • Update TUI model picker and provider wizard UI behavior to prefer raw model IDs and provider-qualified favorites.

Reviewed changes

Copilot reviewed 16 out of 16 changed files in this pull request and generated 4 comments.

Show a summary per file
File Description
internal/tui/provider_wizard.go Persist discovered models after saving provider; refresh savedProviders; adjust model label/secondary text rendering; add mergeSavedProvider.
internal/tui/provider_wizard_test.go Update assertions to match new model label/secondary text behavior.
internal/tui/picker.go Provider-qualified favorite keys, updated picker labeling, and favorite toggling keyed by provider/model.
internal/tui/picker_test.go Update picker tests for provider-qualified favorites and raw ID rendering.
internal/tui/options.go Add SetupSelection.Models to carry discovered models through setup.
internal/tui/onboarding.go Include discovered models in setup save request.
internal/providermodeldiscovery/discovery.go Filter discovered/catalog models through provider-specific allow/block logic.
internal/providermodelcatalog/filter.go Add provider-aware model ID allow/block and filtering helpers.
internal/providermodelcatalog/catalog.go Add curated models for new OpenCode-Go providers; apply provider-specific filtering.
internal/providercatalog/catalog.go Register new OpenCode-Go OpenAI/Anthropic SDK provider descriptors.
internal/providercatalog/catalog_test.go Update expected catalog IDs and per-transport ordering.
internal/config/writer.go Add persistence/merge logic for discovered models; add profile-splitting helper; add dedupe helper.
internal/config/writer_test.go Add tests for discovered model persistence/merge behavior and validation.
internal/config/types.go Introduce DiscoveredModel; add ProviderProfile.Models and unmarshal support.
internal/config/types_test.go Add JSON round-trip tests for DiscoveredModel and ProviderProfile.Models.
internal/cli/setup.go Persist discovered models after provider setup (non-fatal on failure).

💡 Add Copilot custom instructions for smarter, more guided reviews. Learn how to get started.

Comment thread internal/tui/picker.go Outdated
Comment thread internal/config/writer.go Outdated
Comment thread internal/config/writer.go
Comment on lines +238 to +249
func dedupDiscoveredModels(models []DiscoveredModel) []DiscoveredModel {
seen := map[string]bool{}
out := make([]DiscoveredModel, 0, len(models))
for _, m := range models {
if seen[m.ID] {
continue
}
seen[m.ID] = true
out = append(out, m)
}
return out
}
Comment thread internal/tui/provider_wizard.go Outdated
@coderabbitai

coderabbitai Bot commented Jul 2, 2026

Copy link
Copy Markdown

Review Change Stack

Walkthrough

Adds discovered-model persistence in config and setup flows, extends provider and model catalogs with OpenCode-Go entries and filtering, and updates the TUI picker to use provider-qualified favorite keys and raw model IDs.

Changes

Discovered model persistence and provider-aware cataloging

Layer / File(s) Summary
Discovered model type and profile field
internal/config/types.go, internal/config/types_test.go
Adds DiscoveredModel, extends ProviderProfile.Models, and updates JSON unmarshalling plus round-trip coverage.
Config persistence for discovered models
internal/config/writer.go, internal/config/writer_test.go
Adds SetProviderDiscoveredModels and deduplication logic that preserves APIModel overrides, with tests for merge and validation cases.
New provider descriptors and model filtering
internal/providercatalog/catalog.go, internal/providercatalog/catalog_test.go, internal/providermodelcatalog/catalog.go, internal/providermodelcatalog/filter.go, internal/providermodeldiscovery/discovery.go
Adds a new provider descriptor, a curated model set, provider-aware filtering helpers, and filtered discovery/catalog results.
Setup and wizard persistence flow
internal/cli/setup.go, internal/tui/onboarding.go, internal/tui/options.go, internal/tui/provider_wizard.go, internal/tui/provider_wizard_test.go
Persists discovered models from setup and wizard flows, refreshes saved providers, and updates model rendering expectations.
Provider-qualified favorites and raw model labels
internal/tui/picker.go, internal/tui/picker_test.go
Switches picker labels to raw IDs and favorites to provider-qualified keys, with matching test updates.

Estimated code review effort: 4 (Complex) | ~60 minutes

Possibly related PRs

  • Gitlawb/zero#161: Both PRs touch the same provider onboarding flow, especially internal/tui/provider_wizard.go and related save behavior.
  • Gitlawb/zero#190: Both PRs extend saveSetupProvider in internal/cli/setup.go with provider-profile persistence changes.
  • Gitlawb/zero#388: Both PRs extend internal/providercatalog/catalog.go and catalog_test.go with additional provider descriptors.

Suggested reviewers: Vasanthdev2004, gnanam1990, kevincodex1

🚥 Pre-merge checks | ✅ 5
✅ Passed checks (5 passed)
Check name Status Explanation
Description Check ✅ Passed Check skipped - CodeRabbit’s high-level summary is enabled.
Title check ✅ Passed The title is related to the main changes: connector support and provider-qualified favorite IDs.
Docstring Coverage ✅ Passed No functions found in the changed files to evaluate docstring coverage. Skipping docstring coverage check.
Linked Issues check ✅ Passed Check skipped because no linked issues were found for this pull request.
Out of Scope Changes check ✅ Passed Check skipped because no linked issues were found for this pull request.
✨ Finishing Touches
🧪 Generate unit tests (beta)
  • Create PR with unit tests

Thanks for using CodeRabbit! It's free for OSS, and your support helps us grow. If you like it, consider giving us a shout-out.

❤️ Share

Comment @coderabbitai help to get the list of available commands.

@coderabbitai coderabbitai Bot left a comment

Copy link
Copy Markdown

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 2

Caution

Some comments are outside the diff and can’t be posted inline due to platform limitations.

⚠️ Outside diff range comments (1)
internal/tui/provider_wizard.go (1)

906-934: 🩺 Stability & Availability | 🟠 Major | ⚡ Quick win

Non-fatal warning is set but immediately discarded — never reaches the user.

wizard.err is set on line 930 when SetProviderDiscoveredModels fails, but applyProviderWizard unconditionally executes m.providerWizard = nil at line 958 before returning, in the same call and with no render step in between. Since wizard is m.providerWizard (a pointer, confirmed by the nil-check at function entry), setting it to nil discards the entire wizard state — including the just-set err — before it can ever be displayed. The intended non-fatal warning is effectively silent, defeating the purpose of setting it at all.

Store the warning somewhere that survives past m.providerWizard = nil (e.g. a persistent status/toast field on model), or surface it before nilling the wizard.

🐛 Illustrative fix (adjust to the model's actual status-message mechanism)
 			if len(discovered) > 0 {
 				if _, err := config.SetProviderDiscoveredModels(m.userConfigPath, profile.Name, discovered); err != nil {
-					// Non-fatal: the provider was already saved. Log but don't block.
-					// (Redact the path in case it contains identifying info.)
-					wizard.err = strings.TrimSpace("saved provider but could not persist models")
+					// Non-fatal: the provider was already saved. Surface via a field
+					// that outlives the wizard, since m.providerWizard is nilled below.
+					m.postSetupWarning = "saved provider but could not persist models"
 				}
 			}
🤖 Prompt for AI Agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

In `@internal/tui/provider_wizard.go` around lines 906 - 934, The non-fatal
warning assigned in applyProviderWizard is lost because m.providerWizard is
cleared before the UI can render it. Update the applyProviderWizard flow in
provider_wizard.go so the SetProviderDiscoveredModels failure message is stored
in a model-level status/toast field or otherwise surfaced before
m.providerWizard is set to nil. Keep the existing warning text generation near
the wizard.err assignment, but move delivery of that message to a state that
survives beyond the wizard pointer reset.
🧹 Nitpick comments (3)
internal/providermodelcatalog/catalog.go (1)

169-178: 📐 Maintainability & Code Quality | 🔵 Trivial | ⚡ Quick win

Consolidate the repeated FilterModelsForProvider wrapping.

All three return branches now separately call FilterModelsForProvider(provider.ID, dedupeModels(...)). Collapsing this into a single filtered return at the end reduces duplication and removes the risk that a future new branch forgets to wrap the result in the filter.

♻️ Proposed refactor
 func Models(provider providercatalog.Descriptor) []Model {
-	if models, ok := curatedModels[provider.ID]; ok {
-		return FilterModelsForProvider(provider.ID, dedupeModels(provider.DefaultModel, models))
-	}
-	models := registryModels(provider)
-	if len(models) > 0 {
-		return FilterModelsForProvider(provider.ID, dedupeModels(provider.DefaultModel, models))
-	}
-	return FilterModelsForProvider(provider.ID, dedupeModels(provider.DefaultModel, nil))
+	models, ok := curatedModels[provider.ID]
+	if !ok {
+		models = registryModels(provider)
+	}
+	return FilterModelsForProvider(provider.ID, dedupeModels(provider.DefaultModel, models))
 }
🤖 Prompt for AI Agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

In `@internal/providermodelcatalog/catalog.go` around lines 169 - 178, The Models
function repeats the same FilterModelsForProvider(provider.ID,
dedupeModels(...)) wrapping in every branch, so refactor it to compute the
models slice once from curatedModels or registryModels and then apply the filter
in a single return at the end. Keep using provider.DefaultModel with
dedupeModels and preserve the existing fallback to nil when no registry models
are found, but centralize the final FilterModelsForProvider call to avoid
duplicated logic in Models.
internal/providermodelcatalog/filter.go (1)

3-24: 📐 Maintainability & Code Quality | 🔵 Trivial | ⚡ Quick win

Hardcoded provider allow-list logic will get harder to maintain as more providers need filtering.

opencode-go-anthropic is special-cased directly by ID string inside the switch. If another provider needs a similar allow-list later, this pattern requires another hardcoded case here, disconnected from the provider's own descriptor/catalog entry. Consider making the allow-list rule part of the provider data (e.g., an AllowedModelSubstrings/AllowedModelPattern field on the catalog descriptor or curated model set) so the constraint lives next to the provider definition instead of in a separate switch statement.

Also, this new exported logic (ModelIDAllowedForProvider/FilterModelsForProvider) doesn't appear to have its own dedicated unit tests in this diff — only catalog_test.go's ID/order checks exercise it indirectly. A few direct test cases (e.g., asserting qwen/minimax pass and other models are rejected for opencode-go-anthropic, and that all models pass through for unrelated providers) would guard against regressions in this filter contract.

🤖 Prompt for AI Agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

In `@internal/providermodelcatalog/filter.go` around lines 3 - 24, The hardcoded
provider-specific allow-list in ModelIDAllowedForProvider should be moved out of
the switch and into the provider’s catalog data so filtering rules live with the
provider descriptor instead of being special-cased by normalized provider ID.
Update the provider catalog/descriptors to carry an allow-list field (or
equivalent curated model rule) and have ModelIDAllowedForProvider consult that
data rather than strings like opencode-go-anthropic directly. Also add dedicated
unit tests for this filter contract covering qwen/minimax acceptance, rejection
of other models for opencode-go-anthropic, and passthrough behavior for
unrelated providers.
internal/config/writer.go (1)

184-235: 📐 Maintainability & Code Quality | 🔵 Trivial | ⚡ Quick win

No test coverage for PersistTwoProfiles/persistProfileGroup.

writer_test.go only adds tests for SetProviderDiscoveredModels; the new profile-splitting logic (classifier partitioning, single-vs-both-format no-op behavior, API key vs. APIKeyEnv branch) has no tests in this diff, despite being the mechanism that implements the PR's core "OpenAI SDK + Anthropic SDK" split.

🤖 Prompt for AI Agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

In `@internal/config/writer.go` around lines 184 - 235, Add tests for
PersistTwoProfiles and persistProfileGroup to cover the new split logic: verify
the classifier partitions models into OpenAI-compatible vs Anthropic-compatible
groups, that the function is a no-op when only one format is present, and that
both profiles are persisted only when both groups exist. Use the existing
symbols PersistTwoProfiles, persistProfileGroup, and the provider profile
helpers to assert the APIKey vs APIKeyEnv branch and the correct profile
names/format values are written.
🤖 Prompt for all review comments with AI agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

Inline comments:
In `@internal/cli/setup.go`:
- Around line 271-279: The empty error branch in the SetProviderDiscoveredModels
call is triggering SA9003 and hides useful diagnostics. Update the
selection.Models persistence block in setup.go so the err from
config.SetProviderDiscoveredModels is actually handled, ideally by logging a
non-fatal warning with profile.Name and configPath while still allowing setup to
continue. Keep the existing non-blocking behavior, but replace the empty if body
with an explicit use of err so failures are visible.

In `@internal/tui/picker.go`:
- Around line 350-369: The dedup logic in the picker assembly is using the wrong
key and never applies to catalog items, so the same provider+model can be
emitted twice. Update the recent/catalog merge in the function that builds the
picker list to use modelFavoriteKey(item) consistently for both seen checks and
writes, and apply the same dedup check before appending catalog entries while
still allowing cross-provider duplicates. Add a regression test around the
picker logic to cover an item appearing in both recent and catalog for the same
provider.

---

Outside diff comments:
In `@internal/tui/provider_wizard.go`:
- Around line 906-934: The non-fatal warning assigned in applyProviderWizard is
lost because m.providerWizard is cleared before the UI can render it. Update the
applyProviderWizard flow in provider_wizard.go so the
SetProviderDiscoveredModels failure message is stored in a model-level
status/toast field or otherwise surfaced before m.providerWizard is set to nil.
Keep the existing warning text generation near the wizard.err assignment, but
move delivery of that message to a state that survives beyond the wizard pointer
reset.

---

Nitpick comments:
In `@internal/config/writer.go`:
- Around line 184-235: Add tests for PersistTwoProfiles and persistProfileGroup
to cover the new split logic: verify the classifier partitions models into
OpenAI-compatible vs Anthropic-compatible groups, that the function is a no-op
when only one format is present, and that both profiles are persisted only when
both groups exist. Use the existing symbols PersistTwoProfiles,
persistProfileGroup, and the provider profile helpers to assert the APIKey vs
APIKeyEnv branch and the correct profile names/format values are written.

In `@internal/providermodelcatalog/catalog.go`:
- Around line 169-178: The Models function repeats the same
FilterModelsForProvider(provider.ID, dedupeModels(...)) wrapping in every
branch, so refactor it to compute the models slice once from curatedModels or
registryModels and then apply the filter in a single return at the end. Keep
using provider.DefaultModel with dedupeModels and preserve the existing fallback
to nil when no registry models are found, but centralize the final
FilterModelsForProvider call to avoid duplicated logic in Models.

In `@internal/providermodelcatalog/filter.go`:
- Around line 3-24: The hardcoded provider-specific allow-list in
ModelIDAllowedForProvider should be moved out of the switch and into the
provider’s catalog data so filtering rules live with the provider descriptor
instead of being special-cased by normalized provider ID. Update the provider
catalog/descriptors to carry an allow-list field (or equivalent curated model
rule) and have ModelIDAllowedForProvider consult that data rather than strings
like opencode-go-anthropic directly. Also add dedicated unit tests for this
filter contract covering qwen/minimax acceptance, rejection of other models for
opencode-go-anthropic, and passthrough behavior for unrelated providers.
🪄 Autofix (Beta)

Fix all unresolved CodeRabbit comments on this PR:

  • Push a commit to this branch (recommended)
  • Create a new PR with the fixes

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: fd6fb509-077f-48f2-a054-4b6dcd4ef28b

📥 Commits

Reviewing files that changed from the base of the PR and between cdf9d83 and 5457147.

📒 Files selected for processing (16)
  • internal/cli/setup.go
  • internal/config/types.go
  • internal/config/types_test.go
  • internal/config/writer.go
  • internal/config/writer_test.go
  • internal/providercatalog/catalog.go
  • internal/providercatalog/catalog_test.go
  • internal/providermodelcatalog/catalog.go
  • internal/providermodelcatalog/filter.go
  • internal/providermodeldiscovery/discovery.go
  • internal/tui/onboarding.go
  • internal/tui/options.go
  • internal/tui/picker.go
  • internal/tui/picker_test.go
  • internal/tui/provider_wizard.go
  • internal/tui/provider_wizard_test.go

Comment thread internal/cli/setup.go
Comment thread internal/tui/picker.go

@gnanam1990 gnanam1990 left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for this — the request path and existing provider-ID resolution are safe, but two things need addressing before merge.

1. Saved favorites are silently invalidated on upgrade (no migration). This re-keys favorites from bare <model> to <provider>/<model> (modelFavoriteKey in picker.go). Existing users' favorites are stored bare, so after upgrade m.favoriteModels[fk] misses — every favorite silently drops out of the Favorites group and lingers as orphaned config. The tell is that picker_test.go:416 had to change the stored favorite to the new key format to stay green. Please add a bare-key fallback in the lookup (or a one-time migration on load) so existing favorites survive.

2. The advertised SDK-split looks like inert plumbing. PersistTwoProfiles (config/writer.go) has no callers, the IsAnthropicModel classifier it expects isn't in the tree, and the new ProviderProfile.Models field is written but never read. If shipping it unwired is intentional, please note that in the PR description — as-is the title oversells what's actually active. (It being inert does mean it can't misroute existing providers.)

Minor: opencode-go-openai reuses the identical DefaultBaseURL as the existing opencode-go (catalog.go) — resolves fine today by registration order, but it's a latent footgun if descriptors get reordered.

build / vet / test are otherwise green. Happy to re-review once the favorites migration is in.

@anandh8x anandh8x left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the PR. Per CONTRIBUTING.md, community pull requests must be tied to an approved issue with the issue-approved label. This PR:

  • Has no linked issue (Fixes # is blank)
  • Has the checklist entirely unchecked
  • Has no description of what changed and why

Please open an issue first describing the problem you want to solve, and once it gets the issue-approved label, send the PR referencing it. The linked issue also helps scope the change — right now this PR mixes catalog entries, favorites re-keying, model persistence, and a dead PersistTwoProfiles, which is hard to review as one change.

@eli-l

eli-l commented Jul 3, 2026

Copy link
Copy Markdown
Contributor Author

This PR addresses a few problems:

  1. Opencode Go Qwen and Minimax model shall use Anthropic like SDK according to docs Opencode undocumented API #428 . Current access via OpenAI-like SDK is undocumented.
  2. Multi-model configuration per provider #429 - current config.json data model does not allow to store auto-discovered models. Additionally model picker favourites are based on non-unique identifiers.

FYI @gnanam1990 it does not make sense to migrate old identifiers, as config might already contain multiple providers for the same model and it won't be possible to determine which provider was used for favourited item. The only viable solution I see here is to drop favourite records entirely, to prevent dangling items.

Minor: opencode-go-openai reuses the identical DefaultBaseURL as the existing opencode-go - treat it as renaming opencode-go into the opencode-go-openai, due to the fact opencode-go-anthropic was introduced to provide access based on official docs https://opencode.ai/docs/go/#endpoints, requiring Anthropic-like client for minimax and qwen family.
The Anthropic-like requests are supported ONLY for qwen and minimax.
OpenAI-like requests - are supported for ALL models, yet docs says explicitly to use Anthropic SDK for mentioned models.
My take here is to add both ways to access minimax and qwen family, so in the future if openai-like accessed is removed, it would be possible to keep using opencode provider.

@kevincodex1

Copy link
Copy Markdown
Contributor

hello @eli-l thank you so much for your contributions. kindly rebase to main and fix conflicts

@Vasanthdev2004 Vasanthdev2004 left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Request changes — good ideas, but a silent favorites regression, inert plumbing, and an unscoped bundle

Thanks @eli-l — the provider-qualified favorites idea (#429) is a real improvement, and documenting the Anthropic-SDK transport for the OpenCode-Go Minimax/Qwen models (#428) is sensible. I ran go build, go vet, and the touched-package + TUI picker/wizard tests on Windows — all green. But as bundled this isn't mergeable yet, and I'd ask for a split.

🔴 Blocker — favorites silently break on upgrade (no migration)

picker.go now keys favorites via modelFavoriteKey(item) = <provider>/<model>, and every recent/catalog item now carries OwnerProvider = m.providerName. But existing favorites are stored bare (e.g. deepseek-v4-pro) and loaded as bare map keys, so after upgrade m.favoriteModels["opencode-go/deepseek-v4-pro"] misses — every pre-existing favorite drops out of the Favorites group and lingers as orphaned config. The modelFavoriteKey doc comment claims legacy favorites still match, but that branch only fires when OwnerProvider is empty, which never happens for a real model. The tell: picker_test.go:416 had to rewrite the stored favorite to the new prefixed form to stay green. Fix: add a bare-key fallback in the lookup, or a one-time migration on config load. (Also found independently by @gnanam1990.)

🟡 Major — the "SDK-split" persistence half is inert

config.DiscoveredModel, ProviderProfile.Models, SetProviderDiscoveredModels, and the threading through setup/onboarding/wizard persist a discovered-model list — but nothing in internal/providers/factory.go or internal/modelregistry ever reads it (the type comment itself says APIModel is "not yet wired into the factory"). So it's new config surface + merge/dedup/sort logic for zero active behavior, and the PR title oversells it. Either wire a consumer or drop it and reintroduce it alongside the factory change that uses it. (Credit for already removing the dead PersistTwoProfiles.)

🟡 Major — process / scope

Per CONTRIBUTING.md this needs a linked issue-approved issue (checklist unchecked; @anandh8x already raised this), and the branch is CONFLICTING — a rebase is required before CI is meaningful. Please split into: (1) provider-qualified favorites with migration, (2) the new OpenCode-Go SDK-variant catalog entries gated on the approved issue, (3) persistence deferred until a consumer lands.

Minor

  • opencode-go-openai reuses the identical DefaultBaseURL as the merged opencode-go — resolves by registration order today, latent footgun.
  • The qwen/minimax substring allow-list in filter.go is special-cased by provider-ID string with no direct tests; a per-descriptor allow-list field would keep the rule next to the provider.

CI note: fork PR, so Go workflows are maintainer-gated and haven't run — only CodeRabbit did; I verified build/vet/tests locally on Windows. Happy to re-review once the favorites migration is in and the scope is trimmed.

@eli-l eli-l force-pushed the feat/opencode-self-discovery branch from 900b285 to 0b48a9f Compare July 3, 2026 17:39
@eli-l

eli-l commented Jul 3, 2026

Copy link
Copy Markdown
Contributor Author

There is nothing to migrate, as existing favourites are ambiguous and I am not making up the migration, that guesses the provider. My suggestion was to drop the existing favourites if these matches old format. This has not response.

Overall feels like AI gen slop review. I strongly suggest human to take a look on this and agree on decisions or otherwise drop, as I have no intention arguing with bots.

@gnanam1990 gnanam1990 left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

VERDICT: needs-changes

REGRESSION RISK: medium

Re-review of new commits since my prior review (61e3f2320b48a9fd). The new commits address review feedback (fix PR comments, remove unused methods) but introduce a catalog test failure that CI has not caught because the CI checks are in action_required state (not yet approved to run on the latest commits).

BUILD / TEST

  • go build ./... — pass
  • go vet ./internal/cli/... ./internal/config/... ./internal/providercatalog/... ./internal/tui/... — pass
  • gofmt -l on changed files — clean
  • go test ./internal/config/... — pass
  • go test ./internal/providercatalog/... — FAIL. TestListByTransport fails at catalog_test.go:310: the expected TransportOpenAICompat list does not include "opencode-go", but the PR adds openAICompat("opencode-go", ...) to catalog.go:142. The test's expected list at line 297 includes "opencode" and "opencode-go-openai" but not "opencode-go". The PR must add "opencode-go" to the expected list in the test.
  • CI has not run on the latest commit (0b48a9fd) — the workflow runs are in action_required state. This failure would be caught by the Ubuntu smoke job once CI is approved.

FINDINGS

  1. Blocker — internal/providercatalog/catalog_test.go:297 — missing "opencode-go" in expected TransportOpenAICompat list. The PR adds openAICompat("opencode-go", ...) to catalog.go:142 but does not add "opencode-go" to the expected list in the TestListByTransport test. The test fails with ListByTransport("openai-compatible") IDs = [..., "opencode-go", ...], want [..., (no "opencode-go"), ...]. Fix: add "opencode-go" to the expected list at line 297, between "opencode" and "atomic-chat".

BOTTOM LINE

The review feedback cleanup is good, but the catalog test is red because a new provider ID was added without updating the expected test list; CI hasn't caught it yet because the runs are pending approval.

@coderabbitai coderabbitai Bot left a comment

Copy link
Copy Markdown

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🧹 Nitpick comments (1)
internal/providermodelcatalog/catalog.go (1)

159-164: 📐 Maintainability & Code Quality | 🔵 Trivial | ⚡ Quick win

Add a direct test case for opencode-go-anthropic filtering.

ModelIDAllowedForProvider singles out opencode-go-anthropic for the qwen/minimax restriction — the core behavior this PR introduces — but the provider-scoped test table shown in catalog_test.go covers ollama-cloud, ollama, groq, chatgpt, mistral, minimaxi-cn, zai-cn, and doesn't appear to include opencode-go-anthropic itself. Adding a case asserting Models(opencode-go-anthropic-descriptor) includes MiniMax-M3/qwen-plus and excludes something like gpt-4.1/claude-sonnet-4.5 would directly pin down this new gating rule.

🤖 Prompt for AI Agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

In `@internal/providermodelcatalog/catalog.go` around lines 159 - 164, Add a
direct test for the new `opencode-go-anthropic` filtering path so the core
gating rule is covered explicitly. Update `catalog_test.go` to include a case
for `opencode-go-anthropic` using the provider descriptor and assert that
`Models(opencode-go-anthropic-descriptor)` returns the allowed qwen/minimax
entries such as `MiniMax-M3` and `qwen-plus`, while excluding disallowed models
like `gpt-4.1` and `claude-sonnet-4.5`. Use the existing
`ModelIDAllowedForProvider` and `Models` behavior to keep the test aligned with
the provider-scoped filtering logic.
🤖 Prompt for all review comments with AI agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

Nitpick comments:
In `@internal/providermodelcatalog/catalog.go`:
- Around line 159-164: Add a direct test for the new `opencode-go-anthropic`
filtering path so the core gating rule is covered explicitly. Update
`catalog_test.go` to include a case for `opencode-go-anthropic` using the
provider descriptor and assert that `Models(opencode-go-anthropic-descriptor)`
returns the allowed qwen/minimax entries such as `MiniMax-M3` and `qwen-plus`,
while excluding disallowed models like `gpt-4.1` and `claude-sonnet-4.5`. Use
the existing `ModelIDAllowedForProvider` and `Models` behavior to keep the test
aligned with the provider-scoped filtering logic.

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: 0fd77691-8969-49c6-903a-cecd0c7d4dfb

📥 Commits

Reviewing files that changed from the base of the PR and between 0b48a9f and ab945ae.

📒 Files selected for processing (3)
  • internal/providercatalog/catalog.go
  • internal/providercatalog/catalog_test.go
  • internal/providermodelcatalog/catalog.go
💤 Files with no reviewable changes (1)
  • internal/providercatalog/catalog.go
✅ Files skipped from review due to trivial changes (1)
  • internal/providercatalog/catalog_test.go

@jatmn jatmn left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I found a few issues that need to be addressed before this is ready.

Findings

  • [P1] Use the documented OpenCode Go model IDs
    internal/providermodelcatalog/catalog.go:159
    The new Anthropic-compatible OpenCode Go catalog entries use MiniMax-M3, MiniMax-M2.7, qwen-plus, and qwen-max as the model IDs that will be sent to the Anthropic-compatible runtime. The current OpenCode Go endpoint documentation lists the API model IDs as lowercase/versioned IDs such as minimax-m3, minimax-m2.7, qwen3.7-max, qwen3.7-plus, and qwen3.6-plus for the /v1/messages endpoint. With the current hardcoded IDs, selecting the new provider sends names the service does not document, so the advertised SDK split can fail before a request reaches the intended model. Please store the documented API IDs in the catalog and keep display names separate if friendly labels are needed.

  • [P1] Keep existing favorites working after re-keying
    internal/tui/picker.go:373
    The picker now looks up favorites with modelFavoriteKey(item), which becomes <provider>/<model> for real recent/catalog rows because those rows get OwnerProvider set. Existing configs still store bare model IDs, though: favoriteModelSet preserves "qwen3-coder:480b" as-is, and the updated test only covers the new "ollama-cloud/qwen3-coder:480b" format. After upgrade, every existing bare favorite misses the lookup and disappears from the Favorites group. Please add a bare-key fallback or a one-time config migration so existing favorites survive the new provider-qualified format.

  • [P2] Wire persisted discovered models back into model selection
    internal/tui/provider_wizard_models.go:11
    This PR writes live-discovered models into ProviderProfile.Models, but the saved list is not read when the provider is used later. The wizard still rebuilds options from providermodelcatalog.Models(provider), and provider runtime resolution ignores profile.Models, so a model discovered and persisted during setup is lost after restart unless it is also in the curated catalog. That does not match the PR goal of preserving the live-discovered list for later selection. Please either consume the persisted models in the model picker/wizard/runtime path, or remove/defer this config surface until the consumer lands.

  • [P2] Preserve live-discovered models that are not already in the catalog
    internal/providermodeldiscovery/discovery.go:392
    The self-discovery path still drops any live /models ID that is not already present in the catalog whenever catalogModels is non-empty: mergeLiveModels only appends a live model when it finds the same ID in byID, and otherwise continues at line 409. Since setup persists the result of DiscoverCatalog, newly exposed OpenCode models that are not in the hardcoded/remote catalog never reach ProviderProfile.Models at all. Please merge allowed live models into the result instead of using the catalog as a strict allow-list for the very providers this PR is trying to self-discover.

  • [P2] Reuse the existing OpenCode API key env var
    internal/providercatalog/catalog.go:144
    The existing OpenCode Zen and OpenCode Go descriptors both use OPENCODE_API_KEY, but the new Anthropic-compatible OpenCode Go descriptor advertises only OPENCODE_GO_ANTHROPIC_API_KEY. Setup and the provider wizard only check the descriptor's AuthEnvVars, so users who already configured the documented OpenCode key will be forced to re-enter or duplicate the same key for the Anthropic SDK variant. Please include OPENCODE_API_KEY as the primary or fallback env var for this descriptor unless the service actually requires a different credential.

  • [P2] Link approved issues before merging this community PR
    CONTRIBUTING.md:60
    The PR links #428 and #429, but both issues are currently labeled enhancement only and do not have the required issue-approved label. The contribution policy says community PRs must link an existing approved issue and stay focused on that approved scope. Please get the linked issue or issues approved first, then trim the PR to that approved scope if needed.

@Vasanthdev2004

Copy link
Copy Markdown
Collaborator

@eli-l — you're right to be annoyed, and the favorites point especially deserved a real answer instead of the same request three times. Let me give you actual decisions on all of it.

Favorites — your call is the right one. You're correct that a bare favorite is ambiguous: there's no honest way to migrate deepseek-v4-pro into <provider>/<model> without guessing which provider it meant, and guessing is worse than not migrating. So don't build a guessing migration — that ask was wrong. Dropping the old-format entries is the right move. The one thing I'd ask is that it be explicit rather than silent: when config loads, detect a bare-key favorite, drop it from the active set and remove the orphaned record (the real bug today is that they vanish from the UI and linger in config forever). A one-line "favorites were reset for the new provider-qualified format" notice would be a nice touch, but even a clean drop is fine. That closes the blocker the way you proposed.

Scope — I want two of the three things here, and the third isn't ready. The provider-qualified favorites (#429) and the Anthropic-SDK OpenCode-Go entries (#428) are both genuinely wanted — I've marked both issues issue-approved, so that gate is cleared and you're no longer blocked on it. The piece I'd pull out is the discovered-model persistence: right now ProviderProfile.Models is written but nothing reads it back (the wizard still rebuilds from the curated catalog and runtime resolution ignores it), and mergeLiveModels drops any live model that isn't already in the catalog — so end to end, the "preserve what we discovered" goal doesn't actually hold yet. That's not a knock on the idea; it just needs its consumer wired in the same change to be real, which makes it its own PR. Split it out and the two solid features can land now.

The rest that's genuinely real — short, and push back if I've got it wrong:

  • Model IDs: jatmn read the OpenCode Go docs and the /v1/messages IDs look lowercase/versioned (minimax-m3, qwen3.7-plus, …) rather than MiniMax-M3 / qwen-plus. If that's what the docs say, send those and keep friendly labels separate — otherwise the Anthropic route fails before it reaches a model. If you've tested the current IDs against the live endpoint and they work, say so and I'll drop this one.
  • The new descriptor advertises only OPENCODE_GO_ANTHROPIC_API_KEY. Unless the Anthropic endpoint truly needs a separate credential, fall back to the existing OPENCODE_API_KEY so people don't re-enter the same key.
  • catalog_test.go needs "opencode-go" added to the expected TransportOpenAICompat list, and the branch needs a rebase — it's conflicting right now, so CI isn't meaningful until that's done.

None of that is a knock on the ideas — the favorites disambiguation is a real improvement and the Anthropic-SDK fallback is sensible. Do the favorites drop, trim the persistence out, fix the IDs/env/test, rebase, and I'll take it the rest of the way myself. Thanks for pushing back — that was the right instinct, and there's a human on it now.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

7 participants