mirror of
https://github.com/OpenHands/OpenHands.git
synced 2026-03-22 13:47:19 +08:00
refactor: introduce HTTPClient protocol for git service integrations (#10731)
Co-authored-by: openhands <openhands@all-hands.dev>
This commit is contained in:
@@ -24,7 +24,7 @@ async def test_github_service_token_handling():
|
||||
assert service.token.get_secret_value() == 'test-token'
|
||||
|
||||
# Test headers contain the token correctly
|
||||
headers = await service._get_github_headers()
|
||||
headers = await service._get_headers()
|
||||
assert headers['Authorization'] == 'Bearer test-token'
|
||||
assert headers['Accept'] == 'application/vnd.github.v3+json'
|
||||
|
||||
|
||||
309
tests/unit/integrations/protocols/test_http_client.py
Normal file
309
tests/unit/integrations/protocols/test_http_client.py
Normal file
@@ -0,0 +1,309 @@
|
||||
"""Unit tests for HTTPClient abstract base class (ABC)."""
|
||||
|
||||
from typing import Any
|
||||
from unittest.mock import AsyncMock, Mock
|
||||
|
||||
import httpx
|
||||
import pytest
|
||||
from pydantic import SecretStr
|
||||
|
||||
from openhands.integrations.protocols.http_client import HTTPClient
|
||||
from openhands.integrations.service_types import (
|
||||
AuthenticationError,
|
||||
RateLimitError,
|
||||
RequestMethod,
|
||||
ResourceNotFoundError,
|
||||
UnknownException,
|
||||
)
|
||||
|
||||
|
||||
class TestableHTTPClient(HTTPClient):
|
||||
"""Testable concrete implementation of HTTPClient for unit testing."""
|
||||
|
||||
def __init__(self, provider_name: str = 'test-provider'):
|
||||
self.token = SecretStr('test-token')
|
||||
self.refresh = False
|
||||
self.external_auth_id = None
|
||||
self.external_auth_token = None
|
||||
self.external_token_manager = False
|
||||
self.base_domain = None
|
||||
self._provider_name = provider_name
|
||||
|
||||
@property
|
||||
def provider(self) -> str:
|
||||
return self._provider_name
|
||||
|
||||
@provider.setter
|
||||
def provider(self, value: str) -> None:
|
||||
self._provider_name = value
|
||||
|
||||
async def get_latest_token(self) -> SecretStr | None:
|
||||
return self.token
|
||||
|
||||
async def _get_headers(self) -> dict[str, Any]:
|
||||
return {'Authorization': f'Bearer {self.token.get_secret_value()}'}
|
||||
|
||||
async def _make_request(
|
||||
self,
|
||||
url: str,
|
||||
params: dict | None = None,
|
||||
method: RequestMethod = RequestMethod.GET,
|
||||
):
|
||||
# Mock implementation for testing
|
||||
return {'test': 'data'}, {}
|
||||
|
||||
|
||||
@pytest.mark.asyncio
|
||||
class TestHTTPClient:
|
||||
"""Test cases for HTTPClient ABC."""
|
||||
|
||||
def setup_method(self):
|
||||
"""Set up test fixtures."""
|
||||
self.client = TestableHTTPClient()
|
||||
|
||||
def test_default_attributes(self):
|
||||
"""Test default attribute values."""
|
||||
assert isinstance(self.client.token, SecretStr)
|
||||
assert self.client.refresh is False
|
||||
assert self.client.external_auth_id is None
|
||||
assert self.client.external_auth_token is None
|
||||
assert self.client.external_token_manager is False
|
||||
assert self.client.base_domain is None
|
||||
|
||||
def test_provider_property(self):
|
||||
"""Test provider property."""
|
||||
assert self.client.provider == 'test-provider'
|
||||
|
||||
def test_has_token_expired_default_implementation(self):
|
||||
"""Test default _has_token_expired implementation."""
|
||||
# The TestableHTTPClient inherits the default implementation from the protocol
|
||||
client = TestableHTTPClient()
|
||||
|
||||
assert client._has_token_expired(401) is True
|
||||
assert client._has_token_expired(200) is False
|
||||
assert client._has_token_expired(404) is False
|
||||
assert client._has_token_expired(500) is False
|
||||
|
||||
async def test_execute_request_get(self):
|
||||
"""Test execute_request with GET method."""
|
||||
client = TestableHTTPClient()
|
||||
|
||||
mock_client = AsyncMock()
|
||||
mock_response = AsyncMock()
|
||||
mock_client.get.return_value = mock_response
|
||||
|
||||
url = 'https://api.example.com/user'
|
||||
headers = {'Authorization': 'Bearer token'}
|
||||
params = {'per_page': 10}
|
||||
|
||||
result = await client.execute_request(
|
||||
mock_client, url, headers, params, RequestMethod.GET
|
||||
)
|
||||
|
||||
assert result == mock_response
|
||||
mock_client.get.assert_called_once_with(url, headers=headers, params=params)
|
||||
|
||||
async def test_execute_request_post(self):
|
||||
"""Test execute_request with POST method."""
|
||||
client = TestableHTTPClient()
|
||||
|
||||
mock_client = AsyncMock()
|
||||
mock_response = AsyncMock()
|
||||
mock_client.post.return_value = mock_response
|
||||
|
||||
url = 'https://api.example.com/issues'
|
||||
headers = {'Authorization': 'Bearer token'}
|
||||
params = {'title': 'Test Issue'}
|
||||
|
||||
result = await client.execute_request(
|
||||
mock_client, url, headers, params, RequestMethod.POST
|
||||
)
|
||||
|
||||
assert result == mock_response
|
||||
mock_client.post.assert_called_once_with(url, headers=headers, json=params)
|
||||
|
||||
def test_handle_http_status_error_401(self):
|
||||
"""Test handling of 401 HTTP status error."""
|
||||
client = TestableHTTPClient('github')
|
||||
|
||||
mock_response = Mock()
|
||||
mock_response.status_code = 401
|
||||
|
||||
error = httpx.HTTPStatusError(
|
||||
message='401 Unauthorized', request=Mock(), response=mock_response
|
||||
)
|
||||
|
||||
result = client.handle_http_status_error(error)
|
||||
assert isinstance(result, AuthenticationError)
|
||||
assert 'Invalid github token' in str(result)
|
||||
|
||||
def test_handle_http_status_error_404(self):
|
||||
"""Test handling of 404 HTTP status error."""
|
||||
client = TestableHTTPClient()
|
||||
client.provider = 'gitlab'
|
||||
|
||||
mock_response = Mock()
|
||||
mock_response.status_code = 404
|
||||
|
||||
error = httpx.HTTPStatusError(
|
||||
message='404 Not Found', request=Mock(), response=mock_response
|
||||
)
|
||||
|
||||
result = client.handle_http_status_error(error)
|
||||
assert isinstance(result, ResourceNotFoundError)
|
||||
assert 'Resource not found on gitlab API' in str(result)
|
||||
|
||||
def test_handle_http_status_error_429(self):
|
||||
"""Test handling of 429 HTTP status error."""
|
||||
client = TestableHTTPClient()
|
||||
client.provider = 'bitbucket'
|
||||
|
||||
mock_response = Mock()
|
||||
mock_response.status_code = 429
|
||||
|
||||
error = httpx.HTTPStatusError(
|
||||
message='429 Too Many Requests', request=Mock(), response=mock_response
|
||||
)
|
||||
|
||||
result = client.handle_http_status_error(error)
|
||||
assert isinstance(result, RateLimitError)
|
||||
assert 'bitbucket API rate limit exceeded' in str(result)
|
||||
|
||||
def test_handle_http_status_error_other(self):
|
||||
"""Test handling of other HTTP status errors."""
|
||||
client = TestableHTTPClient()
|
||||
client.provider = 'test-provider'
|
||||
|
||||
mock_response = Mock()
|
||||
mock_response.status_code = 500
|
||||
|
||||
error = httpx.HTTPStatusError(
|
||||
message='500 Internal Server Error', request=Mock(), response=mock_response
|
||||
)
|
||||
|
||||
result = client.handle_http_status_error(error)
|
||||
assert isinstance(result, UnknownException)
|
||||
assert 'Unknown error' in str(result)
|
||||
|
||||
def test_handle_http_error(self):
|
||||
"""Test handling of general HTTP errors."""
|
||||
client = TestableHTTPClient()
|
||||
client.provider = 'test-provider'
|
||||
|
||||
error = httpx.ConnectError('Connection failed')
|
||||
|
||||
result = client.handle_http_error(error)
|
||||
assert isinstance(result, UnknownException)
|
||||
assert 'HTTP error ConnectError' in str(result)
|
||||
|
||||
def test_handle_http_error_with_different_error_types(self):
|
||||
"""Test handling of different HTTP error types."""
|
||||
client = TestableHTTPClient()
|
||||
client.provider = 'test-provider'
|
||||
|
||||
# Test with different error types
|
||||
errors = [
|
||||
httpx.ConnectError('Connection failed'),
|
||||
httpx.TimeoutException('Request timed out'),
|
||||
httpx.ReadTimeout('Read timeout'),
|
||||
httpx.WriteTimeout('Write timeout'),
|
||||
]
|
||||
|
||||
for error in errors:
|
||||
result = client.handle_http_error(error)
|
||||
assert isinstance(result, UnknownException)
|
||||
assert f'HTTP error {type(error).__name__}' in str(result)
|
||||
|
||||
def test_runtime_checkable(self):
|
||||
"""Test that HTTPClient is runtime checkable."""
|
||||
from openhands.integrations.protocols.http_client import HTTPClient
|
||||
|
||||
# Test that our testable client implements the protocol
|
||||
assert isinstance(self.client, HTTPClient)
|
||||
|
||||
# Test that a class without the required methods doesn't implement the protocol
|
||||
class IncompleteClient:
|
||||
pass
|
||||
|
||||
incomplete = IncompleteClient()
|
||||
assert not isinstance(incomplete, HTTPClient)
|
||||
|
||||
def test_protocol_attributes_exist(self):
|
||||
"""Test that protocol defines expected attributes."""
|
||||
client = TestableHTTPClient()
|
||||
|
||||
# Test default attribute values from protocol
|
||||
assert hasattr(client, 'token')
|
||||
assert hasattr(client, 'refresh')
|
||||
assert hasattr(client, 'external_auth_id')
|
||||
assert hasattr(client, 'external_auth_token')
|
||||
assert hasattr(client, 'external_token_manager')
|
||||
assert hasattr(client, 'base_domain')
|
||||
|
||||
# Test TestableHTTPClient values
|
||||
assert client.token == SecretStr('test-token')
|
||||
assert client.refresh is False
|
||||
assert client.external_auth_id is None
|
||||
assert client.external_auth_token is None
|
||||
assert client.external_token_manager is False
|
||||
assert client.base_domain is None
|
||||
|
||||
def test_protocol_methods_exist(self):
|
||||
"""Test that protocol defines expected methods."""
|
||||
client = TestableHTTPClient()
|
||||
|
||||
# Test that methods exist
|
||||
assert hasattr(client, 'get_latest_token')
|
||||
assert hasattr(client, '_get_headers')
|
||||
assert hasattr(client, '_make_request')
|
||||
assert hasattr(client, '_has_token_expired')
|
||||
assert hasattr(client, 'execute_request')
|
||||
assert hasattr(client, 'handle_http_status_error')
|
||||
assert hasattr(client, 'handle_http_error')
|
||||
assert hasattr(client, 'provider')
|
||||
|
||||
def test_protocol_concrete_methods_work(self):
|
||||
"""Test that concrete protocol methods work correctly."""
|
||||
client = TestableHTTPClient()
|
||||
|
||||
# These methods should work since TestableHTTPClient implements them
|
||||
assert client.provider == 'test-provider'
|
||||
|
||||
# Test that the default implementations from the protocol are available
|
||||
assert hasattr(client, '_has_token_expired')
|
||||
assert hasattr(client, 'execute_request')
|
||||
assert hasattr(client, 'handle_http_status_error')
|
||||
assert hasattr(client, 'handle_http_error')
|
||||
|
||||
def test_provider_specific_error_messages(self):
|
||||
"""Test that error messages are provider-specific."""
|
||||
providers = ['github', 'gitlab', 'bitbucket']
|
||||
|
||||
for provider in providers:
|
||||
client = TestableHTTPClient()
|
||||
client.provider = provider
|
||||
|
||||
# Test 401 error
|
||||
mock_response = Mock()
|
||||
mock_response.status_code = 401
|
||||
error = httpx.HTTPStatusError(
|
||||
message='401 Unauthorized', request=Mock(), response=mock_response
|
||||
)
|
||||
result = client.handle_http_status_error(error)
|
||||
assert f'Invalid {provider} token' in str(result)
|
||||
|
||||
# Test 404 error
|
||||
mock_response.status_code = 404
|
||||
error = httpx.HTTPStatusError(
|
||||
message='404 Not Found', request=Mock(), response=mock_response
|
||||
)
|
||||
result = client.handle_http_status_error(error)
|
||||
assert f'Resource not found on {provider} API' in str(result)
|
||||
|
||||
# Test 429 error
|
||||
mock_response.status_code = 429
|
||||
error = httpx.HTTPStatusError(
|
||||
message='429 Too Many Requests', request=Mock(), response=mock_response
|
||||
)
|
||||
result = client.handle_http_status_error(error)
|
||||
assert f'{provider} API rate limit exceeded' in str(result)
|
||||
Reference in New Issue
Block a user