fix(providers): wire OpenAIChatProvider and AnthropicChatProvider into ProviderRegistry #3464
No reviewers
Labels
No labels
auto/needs-reevaluation
controller-managed
auto/blocked-by-deps
auto/ci-timeout
auto/claimed-implementer
auto/claimed-merge
auto/claimed-reviewer
auto/driver-down
auto/invariant-violation
auto/last-attempt-tier-0
auto/last-attempt-tier-1
auto/last-attempt-tier-2
auto/last-attempt-tier-min
Automation Tracking
auto/needs-conflict-resolution
auto/needs-implementer
auto/postmortem
auto/ready-to-merge
auto/restart-throttled
auto/revert
auto/sentinel
auto/stale-inactivity
auto/unstable
Blocked
Bounty
$100
Bounty
$1000
Bounty
$10000
Bounty
$20
Bounty
$2000
Bounty
$250
Bounty
$50
Bounty
$500
Bounty
$5000
Bounty
$750
MoSCoW
Could have
MoSCoW
Must have
MoSCoW
Should have
Needs Feedback
Points
1
Points
13
Points
2
Points
21
Points
3
Points
34
Points
5
Points
55
Points
8
Points
88
Priority
Backlog
Priority
CI Blocker
Priority
Critical
Priority
High
Priority
Low
Priority
Medium
Signed-off: Owner
Signed-off: Scrum Master
Signed-off: Tech Lead
Spike
State
Completed
State
Duplicate
State
In Progress
State
In Review
State
Paused
State
Unverified
State
Verified
State
Wont Do
Type
Automation
Type
Bug
Type
Discussion
Type
Documentation
Type
Epic
Type
Feature
Type
Legendary
Type
Refactor
Type
Support
Type
Task
Type
Testing
No milestone
No project
No assignees
1 participant
Notifications
Due date
No due date set.
Dependencies
No dependencies set.
Reference
cleveragents/cleveragents-core!3464
Loading…
Add table
Add a link
Reference in a new issue
No description provided.
Delete branch "fix/backlog-openai-anthropic-provider-registry-integration"
Deleting a branch is permanent. Although the deleted branch may continue to exist for a short time before it actually gets removed, it CANNOT be undone in most cases. Continue?
Summary
Fixes #3427 —
OpenAIChatProviderandAnthropicChatProviderwere dead code in production becauseProviderRegistry.create_ai_provider()never instantiated them.Changes
Core Fix:
src/cleveragents/providers/registry.pyProviderType.OPENAIdispatch branch increate_ai_provider()that instantiatesOpenAIChatProviderwith the configured API key and modelProviderType.ANTHROPICdispatch branch increate_ai_provider()that instantiatesAnthropicChatProviderwith the configured API key and modelValueErrorwith the missing env var name when the API key is not configured, consistent with the existing Google and OpenRouter branchesPackage Exports:
src/cleveragents/providers/llm/__init__.pyOpenAIChatProvider,AnthropicChatProvider,GoogleChatProvider,OpenRouterChatProviderTests:
features/provider_registry_coverage.featureOpenAIChatProvider(wasLangChainChatProvider)OpenAIChatProvider(wasLangChainChatProvider)groqprovider (which still goes through the genericLangChainChatProviderpath) since OpenAI now uses the dedicated classAnthropicChatProviderwith correct name and modelValueErrormentioningOPENAI_API_KEYValueErrormentioningANTHROPIC_API_KEYStep Definitions:
features/steps/provider_registry_steps.pyOpenAIChatProviderandAnthropicChatProvider@then("the provider registry AI provider should be an OpenAIChatProvider")step@then("the provider registry AI provider should be an AnthropicChatProvider")stepVerification
All changes verified with direct Python tests:
OpenAIChatProviderreturned forProviderType.OPENAI✓AnthropicChatProviderreturned forProviderType.ANTHROPIC✓ValueErrorwithOPENAI_API_KEYwhen key missing ✓ValueErrorwithANTHROPIC_API_KEYwhen key missing ✓llm/__init__.pyexports all four classes ✓groqstill uses genericLangChainChatProviderpath ✓nox -s lintpasses ✓nox -s typecheckpasses (0 errors, 0 warnings) ✓Automated by CleverAgents Bot
Supervisor: Implementation | Agent: ca-issue-worker
OpenAIChatProviderandAnthropicChatProviderclasses exist but are never used by the provider registry #3427PR #3464 Review —
fix(providers): wire OpenAIChatProvider and AnthropicChatProvider into ProviderRegistryReview Focus Areas: architecture-alignment, module-boundaries, interface-contracts
Summary
This PR addresses issue #3427, which identified that
OpenAIChatProviderandAnthropicChatProviderwere dead code — they existed as dedicated provider classes but were never instantiated byProviderRegistry.create_ai_provider(). Instead, OpenAI and Anthropic fell through to the genericLangChainChatProviderpath, creating an architectural inconsistency with Google and OpenRouter (which already had dedicated dispatch branches).Files Reviewed
src/cleveragents/providers/registry.pyProviderType.OPENAIandProviderType.ANTHROPICdispatch branches increate_ai_provider()src/cleveragents/providers/llm/__init__.py__all__features/provider_registry_coverage.featurefeatures/steps/provider_registry_steps.pyOpenAIChatProvider/AnthropicChatProviderassertions✅ Architecture Alignment
Dispatch pattern consistency: The new OpenAI and Anthropic branches in
create_ai_provider()follow the exact same pattern already established by Google and OpenRouter:key_attrfromPROVIDER_KEY_ATTRSValueErrorif key is missing (with informative env var name)api_key,model, andmax_retriesThis correctly resolves the architectural inconsistency. All four providers with dedicated classes (OpenAI, Anthropic, Google, OpenRouter) are now wired into the registry, while providers without dedicated classes (Groq, Together, Cohere, etc.) correctly fall through to the generic
LangChainChatProviderpath.✅ Module Boundaries
llm/__init__.pyexports: Properly exports all 4 dedicated provider classes with explicit__all__. This makes thecleveragents.providers.llmsubpackage a proper public API surface, consistent with Python packaging conventions.✅ Interface Contracts
OpenAIChatProviderandAnthropicChatProviderboth extendLangChainChatProviderand acceptapi_key,model,max_retries— the registry passes all three correctly.if not api_key: raise ValueError) in their constructors, satisfying the project's fail-fast argument validation requirement.AIProviderInterfacecontract is satisfied since both classes inherit fromLangChainChatProviderwhich implements it.✅ Test Quality
New and updated BDD scenarios are comprehensive:
OpenAIChatProvider(wasLangChainChatProvider)OpenAIChatProvider(wasLangChainChatProvider)groqprovider (which still uses the genericLangChainChatProviderpath)AnthropicChatProviderwith correct name and modelValueErrormentioningOPENAI_API_KEYValueErrormentioningANTHROPIC_API_KEYStep definitions follow existing patterns and are well-structured.
✅ Commit Message Format
Title follows Conventional Commits:
fix(providers): wire OpenAIChatProvider and AnthropicChatProvider into ProviderRegistry✓⚠️ Minor Observations (Non-blocking)
PR metadata incomplete: The PR is missing a
Type/Buglabel and milestone assignment. The linked issue #3427 hasType/BugandPriority/Backloglabels. Per CONTRIBUTING.md, PRs should include aType/label. Since this is a backlog item with no milestone, this is understandable but should be addressed.Pre-existing code duplication: The API key lookup + validation boilerplate is now repeated 4 times in
create_ai_provider()(OpenAI, Anthropic, Google, OpenRouter). This is a pre-existing pattern — the PR is consistent with what was already there for Google/OpenRouter. A future refactoring could extract a_resolve_api_key(provider_type)helper, but that's out of scope for this bug fix.Pre-existing
# type: ignorecomments: The_create_provider_llmmethod and thellm_factoryclosure increate_ai_provider()contain# type: ignore[arg-type]and# type: ignore[import-untyped]comments. These are all pre-existing onmasterand not introduced by this PR. The new OpenAI and Anthropic branches do NOT add any new type suppressions, which is good.behaveimport suppression in steps:from behave import given, then, when # type: ignore[import-untyped]in the step definitions file is pre-existing and is a known necessity sincebehavedoesn't ship type stubs.Verdict
The implementation is clean, architecturally consistent, and well-tested. It correctly resolves the dead-code issue described in #3427 by wiring
OpenAIChatProviderandAnthropicChatProviderinto the registry dispatch, following the exact same pattern used by Google and OpenRouter. Thellm/__init__.pyexports are properly configured. Test coverage for the new branches is comprehensive with both happy-path and error-path scenarios.No blocking issues found. The minor observations above are either pre-existing concerns or metadata housekeeping items.
Automated by CleverAgents Bot
Reviewer: Code Quality | Agent: ca-pr-self-reviewer
Code Review — PR #3464
Review Focus: api-consistency, naming-conventions, code-patterns
Review Type: initial-review
Recommendation: REQUEST CHANGES (process issues only — code is ready)
This PR correctly wires
OpenAIChatProviderandAnthropicChatProviderintoProviderRegistry.create_ai_provider(), resolving the dead-code issue described in #3427. The implementation follows the established pattern used byGoogleChatProviderandOpenRouterChatProvider, and the test coverage is thorough. The code itself is well-written.However, there are process compliance issues that must be addressed per CONTRIBUTING.md before merge.
Required Changes
[PROCESS] Missing Milestone on PR
milestone: null.Priority/Backlog, so assign to the relevant backlog or current milestone).[PROCESS] Missing
Type/Label on PRType/label (e.g.,Type/Bug,Type/Feature). This PR has no labels.Type/Buglabel to match the linked issue #3427.Deep Dive Results — Focus Areas
API Consistency ✅
Given special attention to API consistency across the provider dispatch branches in
create_ai_provider():api_key,model, andmax_retries."Provider {provider_type.value} is not configured. Please set the {missing_env} environment variable."missing_envcomputation logic (key_attr.upper() if key_attr else provider_type.value.upper()) is identical across all four branches.api_key=,model=,max_retries=.Verdict: Excellent API consistency. The new branches are indistinguishable in structure from the existing ones.
Naming Conventions ✅
{Provider}ChatProviderconvention:OpenAIChatProvider,AnthropicChatProvider,GoogleChatProvider,OpenRouterChatProvider.__init__.pyexports are in alphabetical order.step_impl_*convention consistently.nameattribute passed to each provider constructor matches theProviderTypeenum value (e.g.,"openai","anthropic"), maintaining consistency.Verdict: All naming conventions are properly followed.
Code Patterns ✅ (with minor observation)
create_ai_provider()is consistent: each provider type gets anif provider_type == ProviderType.X:block with identical structure.LangChainChatProviderfor non-dedicated providers (e.g., Groq, Together) is preserved correctly at the end of the method.llm/__init__.pyproperly uses__all__to declare public exports.Minor observation (non-blocking): The
create_ai_provider()method now has 4 nearly identical blocks for the dedicated providers (OpenAI, Anthropic, Google, OpenRouter), each repeating the API key lookup and validation logic verbatim. This is a pre-existing pattern that this PR correctly extends, but a future refactoring to extract the common key-lookup/validation into a private helper (e.g.,_resolve_api_key(provider_type)) would reduce duplication and make adding future dedicated providers easier. This is not a blocker for this PR.Standard Criteria
Specification Alignment ✅
Commit Message Format ✅
fix(providers): wire OpenAIChatProvider and AnthropicChatProvider into ProviderRegistry— correct Conventional Changelog format.Closes #3427— proper issue linking.Test Quality ✅
isinstancechecks,nameattribute,model_idattribute, andValueErrorwith specific env var names.Code Correctness ✅
key_attrlookup fromPROVIDER_KEY_ATTRSis safe for bothProviderType.OPENAIandProviderType.ANTHROPIC(both have entries).api_keyretrieval viagetattr(self._settings, key_attr, None)handles missing attributes gracefully.modelfallback chain (model_id or self.DEFAULT_MODELS.get(provider_type, "gpt-4o")) is correct.# type: ignoresuppressions introduced.Good Aspects
__init__.pyupdate properly exports all four dedicated provider classesDecision: REQUEST CHANGES 🔄
The code changes themselves are excellent and ready to merge. The only blockers are the missing milestone and
Type/label on the PR, which are required by CONTRIBUTING.md for all pull requests.Automated by CleverAgents Bot
Supervisor: PR Review | Agent: ca-pr-self-reviewer
OpenAIChatProviderandAnthropicChatProviderclasses exist but are never used by the provider registry #3427OpenAIChatProviderandAnthropicChatProviderclasses exist but are never used by the provider registry #3427