Feature Request: Configurable JsonClient Class for Dependency Injection
Problem
When testing Campus services that use campus_python.Campus, we need to replace the default CampusRequest with a test-compatible version that routes to Flask test clients instead of making real HTTP requests.
Current Workaround: Monkey-Patching
Currently, we have to monkey-patch the CampusRequest class:
import campus_python
from tests.flask_test import TestCampusRequest
# Replace CampusRequest globally
campus_python.json_client.CampusRequest = TestCampusRequest
campus_python.CampusRequest = TestCampusRequest # Also patch module reference
Problems with this approach:
- ❌ Fragile - requires patching multiple module-level references
- ❌ Hard to debug - changes global state
- ❌ Confusing - not obvious that
CampusRequest has been replaced
- ❌ Brittle - may break if campus-api-python internals change
Proposed Solution
Add a configurable class attribute to allow dependency injection of the JsonClient class:
class Campus:
"""Unified Campus client interface."""
# Configurable JsonClient class
json_client_class: type[JsonClient] = CampusRequest
@property
def auth(self) -> AuthRoot:
if not hasattr(self, "_auth"):
# Use json_client_class instead of hardcoded CampusRequest
self._auth = AuthRoot(
json_client=self.json_client_class(
base_url=base_url,
timeout=self.timeout,
)
)
return self._auth
@property
def api(self) -> ApiRoot:
if not hasattr(self, "_api"):
self._api = ApiRoot(
json_client=self.json_client_class(
base_url=base_url,
timeout=self.timeout,
)
)
return self._api
Usage in Tests
import campus_python
from tests.flask_test import TestCampusRequest
def setup():
# Configure campus_python to use test client
campus_python.Campus.json_client_class = TestCampusRequest
# Now all Campus instances use TestCampusRequest
campus = campus_python.Campus(timeout=60)
campus.auth.root.authenticate(...) # Uses Flask test clients!
Benefits
- ✅ Clean dependency injection - No monkey-patching required
- ✅ Explicit configuration - Clear what JsonClient is being used
- ✅ Backward compatible - Defaults to
CampusRequest
- ✅ Test-friendly - Easy to inject test doubles
- ✅ Flexible - Allows custom JsonClient implementations for:
- Testing (Flask test clients)
- Mocking (for unit tests)
- Custom HTTP backends (async, retry logic, etc.)
Implementation Details
Changes Required
File: campus_python/__init__.py
-
Add class attribute:
class Campus:
json_client_class: type[JsonClient] = CampusRequest
-
Replace hardcoded CampusRequest(...) with self.json_client_class(...):
- In
auth property (line ~81)
- In
api property (line ~107)
Example Custom JsonClient
from campus_python.json_client.interface import JsonClient, JsonResponse
class CustomJsonClient(JsonClient):
"""Custom JsonClient with special behavior."""
def __init__(self, base_url: str | None = None, **kwargs):
self.base_url = base_url or ""
# ... custom initialization ...
def get(self, path: str, query: dict | None = None) -> JsonResponse:
# ... custom implementation ...
pass
# ... implement other methods ...
# Use it
campus_python.Campus.json_client_class = CustomJsonClient
Backward Compatibility
✅ Fully backward compatible - Default value is CampusRequest, so existing code continues to work without changes.
Related
Alternatives Considered
-
Constructor parameter (Campus(json_client_class=...))
- ❌ Doesn't work for services that instantiate
Campus() internally (campus.auth, campus.api)
-
Global function (set_json_client_class())
- ❌ More verbose than class attribute
- ❌ Requires additional function to maintain
-
Keep monkey-patching
Implementation Checklist
Feature Request: Configurable JsonClient Class for Dependency Injection
Problem
When testing Campus services that use
campus_python.Campus, we need to replace the defaultCampusRequestwith a test-compatible version that routes to Flask test clients instead of making real HTTP requests.Current Workaround: Monkey-Patching
Currently, we have to monkey-patch the
CampusRequestclass:Problems with this approach:
CampusRequesthas been replacedProposed Solution
Add a configurable class attribute to allow dependency injection of the JsonClient class:
Usage in Tests
Benefits
CampusRequestImplementation Details
Changes Required
File:
campus_python/__init__.pyAdd class attribute:
Replace hardcoded
CampusRequest(...)withself.json_client_class(...):authproperty (line ~81)apiproperty (line ~107)Example Custom JsonClient
Backward Compatibility
✅ Fully backward compatible - Default value is
CampusRequest, so existing code continues to work without changes.Related
tests/flask_test/campus_request.pyAlternatives Considered
Constructor parameter (
Campus(json_client_class=...))Campus()internally (campus.auth, campus.api)Global function (
set_json_client_class())Keep monkey-patching
Implementation Checklist
json_client_classclass attribute toCampusauthproperty to useself.json_client_classapiproperty to useself.json_client_class