From 0250ab85f625a8d86079c96586d506f06102c6bc Mon Sep 17 00:00:00 2001 From: Thomas Poignant Date: Tue, 30 Apr 2024 12:42:51 +0200 Subject: [PATCH] feat(python of provider): Stop using provider.provider (#1772) * Stop using provider.provider as mentionned in https://github.com/open-feature/python-sdk/issues/319 Signed-off-by: Thomas Poignant * Remove usage of status Signed-off-by: Thomas Poignant * remove unused init Signed-off-by: Thomas Poignant --------- Signed-off-by: Thomas Poignant --- .../gofeatureflag_python_provider/provider.py | 53 +------------------ .../provider_status.py | 9 ---- .../response_flag_evaluation.py | 2 +- .../test_gofeatureflag_python_provider.py | 18 ------- .../test_websocket_cache_invalidation.py | 2 - 5 files changed, 2 insertions(+), 82 deletions(-) delete mode 100644 openfeature/providers/python-provider/gofeatureflag_python_provider/provider_status.py diff --git a/openfeature/providers/python-provider/gofeatureflag_python_provider/provider.py b/openfeature/providers/python-provider/gofeatureflag_python_provider/provider.py index 4b0962e0924..69fab278cc4 100644 --- a/openfeature/providers/python-provider/gofeatureflag_python_provider/provider.py +++ b/openfeature/providers/python-provider/gofeatureflag_python_provider/provider.py @@ -18,13 +18,12 @@ from openfeature.flag_evaluation import FlagResolutionDetails, Reason from openfeature.hook import Hook from openfeature.provider.metadata import Metadata -from openfeature.provider.provider import AbstractProvider +from openfeature.provider import AbstractProvider from pydantic import PrivateAttr, ValidationError from gofeatureflag_python_provider.data_collector_hook import DataCollectorHook from gofeatureflag_python_provider.metadata import GoFeatureFlagMetadata from gofeatureflag_python_provider.options import BaseModel, GoFeatureFlagOptions -from gofeatureflag_python_provider.provider_status import ProviderStatus from gofeatureflag_python_provider.request_flag_evaluation import ( RequestFlagEvaluation, convert_evaluation_context, @@ -46,7 +45,6 @@ class GoFeatureFlagProvider(BaseModel, AbstractProvider, metaclass=CombinedMetac options: GoFeatureFlagOptions _http_client: urllib3.PoolManager = PrivateAttr() _cache: pylru.lrucache = PrivateAttr() - _status: ProviderStatus = PrivateAttr(ProviderStatus.NOT_READY) _data_collector_hook: Optional[DataCollectorHook] = PrivateAttr() _ws: websocket.WebSocketApp = PrivateAttr() _ws_thread: Thread = PrivateAttr() @@ -75,18 +73,8 @@ def __init__(self, **data): self._ws = websocket.WebSocketApp( self._build_websocket_uri(), on_message=self._websocket_message_handler, - on_open=self._websocket_open_handler, - on_close=self._websocket_close_handler, - on_error=self._websocket_error_handler, ) - def get_status(self) -> ProviderStatus: - """ - get_status returns the status of the provider - :return: the status of the provider - """ - return self._status - def initialize(self, evaluation_context: EvaluationContext) -> None: """ initialize is called when the provider is initialized. @@ -99,8 +87,6 @@ def initialize(self, evaluation_context: EvaluationContext) -> None: if self.options.disable_cache_invalidation is False: self._ws_thread = Thread(target=self.run_websocket) self._ws_thread.start() - else: - self._status = ProviderStatus.READY def shutdown(self): if self.options.disable_cache_invalidation is False: @@ -114,8 +100,6 @@ def shutdown(self): self._data_collector_hook.shutdown() self._data_collector_hook = None - self._status = ProviderStatus.NOT_READY - def get_metadata(self) -> Metadata: return GoFeatureFlagMetadata() @@ -192,14 +176,6 @@ def generic_go_feature_flag_resolver( :return: a FlagResolutionDetails object containing the response for the SDK. """ try: - if self._status != ProviderStatus.READY: - return FlagResolutionDetails[original_type]( - value=default_value, - reason=Reason.ERROR, - error_code=ErrorCode.PROVIDER_NOT_READY, - error_message="GO Feature Flag provider is not ready", - ) - goff_evaluation_context = convert_evaluation_context(evaluation_context) goff_request = RequestFlagEvaluation( user=goff_evaluation_context, @@ -310,32 +286,5 @@ def _websocket_message_handler(self, wsapp, message) -> None: # when we receive a message from go-feature-flag server, we clear the cache. self._cache.clear() - def _websocket_open_handler(self, ws_app) -> None: - """ - websocket_open_handler is the handler called when the websocket is open - :param ws app: the websocket app - :return: None - """ - self._status = ProviderStatus.READY - - def _websocket_error_handler(self, ws_app, error) -> None: - """ - websocket_error_handler is the handler called when we receive an error from the GO Feature Flag server - :param ws_app: the websocket app - :param error: error received - :return: None - """ - self._status = ProviderStatus.ERROR - - def _websocket_close_handler(self, ws_app, close_status_code, close_msg) -> None: - """ - websocket_close_handler is the handler called when the websocket is closed - :param wsapp: the websocket app - :param close_status_code: the status code of the close - :param close_msg: the message of the close - :return: None - """ - self._status = ProviderStatus.STALE - def __hash__(self): return id(self) diff --git a/openfeature/providers/python-provider/gofeatureflag_python_provider/provider_status.py b/openfeature/providers/python-provider/gofeatureflag_python_provider/provider_status.py deleted file mode 100644 index 8d8d031c05f..00000000000 --- a/openfeature/providers/python-provider/gofeatureflag_python_provider/provider_status.py +++ /dev/null @@ -1,9 +0,0 @@ -from enum import Enum - - -# ProviderStatus is an enum that represents the status of a provider -class ProviderStatus(Enum): - NOT_READY = 1 - READY = 2 - STALE = 3 - ERROR = 4 diff --git a/openfeature/providers/python-provider/gofeatureflag_python_provider/response_flag_evaluation.py b/openfeature/providers/python-provider/gofeatureflag_python_provider/response_flag_evaluation.py index 1bb7039ed9f..e03fb039042 100644 --- a/openfeature/providers/python-provider/gofeatureflag_python_provider/response_flag_evaluation.py +++ b/openfeature/providers/python-provider/gofeatureflag_python_provider/response_flag_evaluation.py @@ -1,4 +1,4 @@ -from typing import Optional, Generic, Union, TypeVar +from typing import Optional, Union, TypeVar from gofeatureflag_python_provider.options import BaseModel diff --git a/openfeature/providers/python-provider/tests/test_gofeatureflag_python_provider.py b/openfeature/providers/python-provider/tests/test_gofeatureflag_python_provider.py index 2803c7730a2..94207c22fcd 100644 --- a/openfeature/providers/python-provider/tests/test_gofeatureflag_python_provider.py +++ b/openfeature/providers/python-provider/tests/test_gofeatureflag_python_provider.py @@ -12,7 +12,6 @@ from gofeatureflag_python_provider.options import GoFeatureFlagOptions from gofeatureflag_python_provider.provider import GoFeatureFlagProvider -from gofeatureflag_python_provider.provider_status import ProviderStatus _default_evaluation_ctx = EvaluationContext( targeting_key="d45e303a-38c2-11ed-a261-0242ac120002", @@ -43,7 +42,6 @@ def _generic_test( ), ) api.set_provider(goff_provider) - wait_provider_ready(goff_provider) client = api.get_client(domain="test-client") if evaluationType == "bool": @@ -148,7 +146,6 @@ def test_should_return_an_error_if_endpoint_not_available(mock_request): ) ) api.set_provider(goff_provider) - wait_provider_ready(goff_provider) client = api.get_client(domain="test-client") res = client.get_boolean_details( flag_key=flag_key, @@ -451,7 +448,6 @@ def test_should_resolve_from_cache_if_multiple_call_to_the_same_flag_with_same_c ) ) api.set_provider(goff_provider) - wait_provider_ready(goff_provider) client = api.get_client(domain="test-client") got = client.get_boolean_details( @@ -506,7 +502,6 @@ def test_should_call_data_collector_multiple_times_with_cached_event_waiting_ttl ) ) api.set_provider(goff_provider) - wait_provider_ready(goff_provider) client = api.get_client(domain="test-client") got = client.get_boolean_details( @@ -563,7 +558,6 @@ def test_should_not_call_data_collector_if_not_having_cache(mock_request: Mock): ) api.set_provider(goff_provider) - wait_provider_ready(goff_provider) client = api.get_client(domain="test-client") client.get_boolean_details( @@ -575,18 +569,6 @@ def test_should_not_call_data_collector_if_not_having_cache(mock_request: Mock): assert mock_request.call_count == 1 -def wait_provider_ready(provider: GoFeatureFlagProvider): - # check the provider get_status method until it returns ProviderStatus.READY or, we waited more than 5 seconds - start = time.time() - while provider.get_status() != ProviderStatus.READY: - time.sleep(0.1) - if time.time() - start > 5: - break - - if provider.get_status() != ProviderStatus.READY: - raise Exception("Provider is not ready") - - def _read_mock_file(flag_key: str) -> str: # This hacky if is here to make test run inside pycharm and from the root of the project if os.getcwd().endswith("/tests"): diff --git a/openfeature/providers/python-provider/tests/test_websocket_cache_invalidation.py b/openfeature/providers/python-provider/tests/test_websocket_cache_invalidation.py index e7fb24a628c..5c504cc1cde 100644 --- a/openfeature/providers/python-provider/tests/test_websocket_cache_invalidation.py +++ b/openfeature/providers/python-provider/tests/test_websocket_cache_invalidation.py @@ -11,7 +11,6 @@ from gofeatureflag_python_provider.options import GoFeatureFlagOptions from gofeatureflag_python_provider.provider import GoFeatureFlagProvider from tests.test_gofeatureflag_python_provider import ( - wait_provider_ready, _default_evaluation_ctx, ) @@ -54,7 +53,6 @@ def test_test_websocket_cache_invalidation(goff): ) ) api.set_provider(goff_provider) - wait_provider_ready(goff_provider) client = api.get_client(domain="test-client") want = FlagEvaluationDetails(