openfeature-provider-flagd 0.2.0__py3-none-any.whl → 0.2.2__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -3,6 +3,8 @@ import os
3
3
  import typing
4
4
  from enum import Enum
5
5
 
6
+ import grpc
7
+
6
8
 
7
9
  class ResolverType(Enum):
8
10
  RPC = "rpc"
@@ -45,9 +47,11 @@ ENV_VAR_RETRY_BACKOFF_MS = "FLAGD_RETRY_BACKOFF_MS"
45
47
  ENV_VAR_RETRY_BACKOFF_MAX_MS = "FLAGD_RETRY_BACKOFF_MAX_MS"
46
48
  ENV_VAR_RETRY_GRACE_PERIOD_SECONDS = "FLAGD_RETRY_GRACE_PERIOD"
47
49
  ENV_VAR_SELECTOR = "FLAGD_SOURCE_SELECTOR"
50
+ ENV_VAR_PROVIDER_ID = "FLAGD_SOURCE_PROVIDER_ID"
48
51
  ENV_VAR_STREAM_DEADLINE_MS = "FLAGD_STREAM_DEADLINE_MS"
49
52
  ENV_VAR_TLS = "FLAGD_TLS"
50
53
  ENV_VAR_TLS_CERT = "FLAGD_SERVER_CERT_PATH"
54
+ ENV_VAR_DEFAULT_AUTHORITY = "FLAGD_DEFAULT_AUTHORITY"
51
55
 
52
56
  T = typing.TypeVar("T")
53
57
 
@@ -81,6 +85,7 @@ class Config:
81
85
  port: typing.Optional[int] = None,
82
86
  tls: typing.Optional[bool] = None,
83
87
  selector: typing.Optional[str] = None,
88
+ provider_id: typing.Optional[str] = None,
84
89
  resolver: typing.Optional[ResolverType] = None,
85
90
  offline_flag_source_path: typing.Optional[str] = None,
86
91
  offline_poll_interval_ms: typing.Optional[int] = None,
@@ -93,6 +98,9 @@ class Config:
93
98
  cache: typing.Optional[CacheType] = None,
94
99
  max_cache_size: typing.Optional[int] = None,
95
100
  cert_path: typing.Optional[str] = None,
101
+ default_authority: typing.Optional[str] = None,
102
+ channel_credentials: typing.Optional[grpc.ChannelCredentials] = None,
103
+ sync_metadata_disabled: typing.Optional[bool] = None,
96
104
  ):
97
105
  self.host = env_or_default(ENV_VAR_HOST, DEFAULT_HOST) if host is None else host
98
106
 
@@ -227,3 +235,25 @@ class Config:
227
235
  self.selector = (
228
236
  env_or_default(ENV_VAR_SELECTOR, None) if selector is None else selector
229
237
  )
238
+
239
+ self.provider_id = (
240
+ env_or_default(ENV_VAR_PROVIDER_ID, None)
241
+ if provider_id is None
242
+ else provider_id
243
+ )
244
+
245
+ self.default_authority = (
246
+ env_or_default(ENV_VAR_DEFAULT_AUTHORITY, None)
247
+ if default_authority is None
248
+ else default_authority
249
+ )
250
+
251
+ self.channel_credentials = channel_credentials
252
+
253
+ # TODO: remove the metadata call entirely after https://github.com/open-feature/flagd/issues/1584
254
+ # This is a temporary stop-gap solutions to support servers that don't implement sync.GetMetadata
255
+ # (see: https://buf.build/open-feature/flagd/docs/main:flagd.sync.v1#flagd.sync.v1.FlagSyncService.GetMetadata).
256
+ # Using this option disables call to sync.GetMetadata
257
+ # Disabling will prevent static context from flagd being used in evaluations.
258
+ # GetMetadata and this option will be removed.
259
+ self.sync_metadata_disabled = sync_metadata_disabled
@@ -24,13 +24,18 @@
24
24
  import typing
25
25
  import warnings
26
26
 
27
+ import grpc
28
+
27
29
  from openfeature.evaluation_context import EvaluationContext
30
+ from openfeature.event import ProviderEventDetails
28
31
  from openfeature.flag_evaluation import FlagResolutionDetails
32
+ from openfeature.hook import Hook
29
33
  from openfeature.provider import AbstractProvider
30
34
  from openfeature.provider.metadata import Metadata
31
35
 
32
36
  from .config import CacheType, Config, ResolverType
33
37
  from .resolvers import AbstractResolver, GrpcResolver, InProcessResolver
38
+ from .sync_metadata_hook import SyncMetadataHook
34
39
 
35
40
  T = typing.TypeVar("T")
36
41
 
@@ -47,6 +52,7 @@ class FlagdProvider(AbstractProvider):
47
52
  timeout: typing.Optional[int] = None,
48
53
  retry_backoff_ms: typing.Optional[int] = None,
49
54
  selector: typing.Optional[str] = None,
55
+ provider_id: typing.Optional[str] = None,
50
56
  resolver_type: typing.Optional[ResolverType] = None,
51
57
  offline_flag_source_path: typing.Optional[str] = None,
52
58
  stream_deadline_ms: typing.Optional[int] = None,
@@ -56,6 +62,9 @@ class FlagdProvider(AbstractProvider):
56
62
  retry_backoff_max_ms: typing.Optional[int] = None,
57
63
  retry_grace_period: typing.Optional[int] = None,
58
64
  cert_path: typing.Optional[str] = None,
65
+ default_authority: typing.Optional[str] = None,
66
+ channel_credentials: typing.Optional[grpc.ChannelCredentials] = None,
67
+ sync_metadata_disabled: typing.Optional[bool] = None,
59
68
  ):
60
69
  """
61
70
  Create an instance of the FlagdProvider
@@ -88,6 +97,7 @@ class FlagdProvider(AbstractProvider):
88
97
  retry_backoff_max_ms=retry_backoff_max_ms,
89
98
  retry_grace_period=retry_grace_period,
90
99
  selector=selector,
100
+ provider_id=provider_id,
91
101
  resolver=resolver_type,
92
102
  offline_flag_source_path=offline_flag_source_path,
93
103
  stream_deadline_ms=stream_deadline_ms,
@@ -95,9 +105,20 @@ class FlagdProvider(AbstractProvider):
95
105
  cache=cache,
96
106
  max_cache_size=max_cache_size,
97
107
  cert_path=cert_path,
108
+ default_authority=default_authority,
109
+ channel_credentials=channel_credentials,
110
+ sync_metadata_disabled=sync_metadata_disabled,
98
111
  )
112
+ self.enriched_context: dict = {}
99
113
 
100
114
  self.resolver = self.setup_resolver()
115
+ self.hooks: list[Hook] = [SyncMetadataHook(self.get_enriched_context)]
116
+
117
+ def get_enriched_context(self) -> EvaluationContext:
118
+ return EvaluationContext(attributes=self.enriched_context)
119
+
120
+ def get_provider_hooks(self) -> list[Hook]:
121
+ return self.hooks
101
122
 
102
123
  def setup_resolver(self) -> AbstractResolver:
103
124
  if self.config.resolver == ResolverType.RPC:
@@ -114,7 +135,7 @@ class FlagdProvider(AbstractProvider):
114
135
  ):
115
136
  return InProcessResolver(
116
137
  self.config,
117
- self.emit_provider_ready,
138
+ self.emit_provider_ready_with_context,
118
139
  self.emit_provider_error,
119
140
  self.emit_provider_stale,
120
141
  self.emit_provider_configuration_changed,
@@ -184,3 +205,10 @@ class FlagdProvider(AbstractProvider):
184
205
  return self.resolver.resolve_object_details(
185
206
  key, default_value, evaluation_context
186
207
  )
208
+
209
+ def emit_provider_ready_with_context(
210
+ self, details: ProviderEventDetails, context: dict
211
+ ) -> None:
212
+ self.enriched_context = context
213
+ self.emit_provider_ready(details)
214
+ pass
@@ -137,7 +137,10 @@ class GrpcResolver:
137
137
 
138
138
  def _state_change_callback(self, new_state: ChannelConnectivity) -> None:
139
139
  logger.debug(f"gRPC state change: {new_state}")
140
- if new_state == ChannelConnectivity.READY:
140
+ if (
141
+ new_state == grpc.ChannelConnectivity.READY
142
+ or new_state == grpc.ChannelConnectivity.IDLE
143
+ ):
141
144
  if not self.thread or not self.thread.is_alive():
142
145
  self.thread = threading.Thread(
143
146
  target=self.listen,
@@ -276,7 +279,7 @@ class GrpcResolver:
276
279
  return cached_flag
277
280
 
278
281
  context = self._convert_context(evaluation_context)
279
- call_args = {"timeout": self.deadline}
282
+ call_args = {"timeout": self.deadline, "wait_for_ready": True}
280
283
  try:
281
284
  request: Message
282
285
  if flag_type == FlagType.BOOLEAN:
@@ -21,7 +21,7 @@ class InProcessResolver:
21
21
  def __init__(
22
22
  self,
23
23
  config: Config,
24
- emit_provider_ready: typing.Callable[[ProviderEventDetails], None],
24
+ emit_provider_ready: typing.Callable[[ProviderEventDetails, dict], None],
25
25
  emit_provider_error: typing.Callable[[ProviderEventDetails], None],
26
26
  emit_provider_stale: typing.Callable[[ProviderEventDetails], None],
27
27
  emit_provider_configuration_changed: typing.Callable[
@@ -121,7 +121,7 @@ class InProcessResolver:
121
121
  )
122
122
 
123
123
  variant, value = flag.get_variant(variant)
124
- if not value:
124
+ if value is None:
125
125
  raise ParseError(f"Resolved variant {variant} not in variants config.")
126
126
 
127
127
  return FlagResolutionDetails(
@@ -24,7 +24,7 @@ class FileWatcher(FlagStateConnector):
24
24
  self,
25
25
  config: Config,
26
26
  flag_store: FlagStore,
27
- emit_provider_ready: typing.Callable[[ProviderEventDetails], None],
27
+ emit_provider_ready: typing.Callable[[ProviderEventDetails, dict], None],
28
28
  emit_provider_error: typing.Callable[[ProviderEventDetails], None],
29
29
  ):
30
30
  if config.offline_flag_source_path is None:
@@ -94,7 +94,8 @@ class FileWatcher(FlagStateConnector):
94
94
  self.emit_provider_ready(
95
95
  ProviderEventDetails(
96
96
  message="Reloading file contents recovered from error state"
97
- )
97
+ ),
98
+ {},
98
99
  )
99
100
  self.should_emit_ready_on_success = False
100
101
 
@@ -5,6 +5,8 @@ import time
5
5
  import typing
6
6
 
7
7
  import grpc
8
+ from google.protobuf.json_format import MessageToDict
9
+ from google.protobuf.struct_pb2 import Struct
8
10
 
9
11
  from openfeature.evaluation_context import EvaluationContext
10
12
  from openfeature.event import ProviderEventDetails
@@ -26,7 +28,7 @@ class GrpcWatcher(FlagStateConnector):
26
28
  self,
27
29
  config: Config,
28
30
  flag_store: FlagStore,
29
- emit_provider_ready: typing.Callable[[ProviderEventDetails], None],
31
+ emit_provider_ready: typing.Callable[[ProviderEventDetails, dict], None],
30
32
  emit_provider_error: typing.Callable[[ProviderEventDetails], None],
31
33
  emit_provider_stale: typing.Callable[[ProviderEventDetails], None],
32
34
  ):
@@ -41,6 +43,7 @@ class GrpcWatcher(FlagStateConnector):
41
43
  self.streamline_deadline_seconds = config.stream_deadline_ms * 0.001
42
44
  self.deadline = config.deadline_ms * 0.001
43
45
  self.selector = config.selector
46
+ self.provider_id = config.provider_id
44
47
  self.emit_provider_ready = emit_provider_ready
45
48
  self.emit_provider_error = emit_provider_error
46
49
  self.emit_provider_stale = emit_provider_stale
@@ -54,13 +57,23 @@ class GrpcWatcher(FlagStateConnector):
54
57
  def _generate_channel(self, config: Config) -> grpc.Channel:
55
58
  target = f"{config.host}:{config.port}"
56
59
  # Create the channel with the service config
57
- options = [
60
+ options: list[tuple[str, typing.Any]] = [
58
61
  ("grpc.keepalive_time_ms", config.keep_alive_time),
59
62
  ("grpc.initial_reconnect_backoff_ms", config.retry_backoff_ms),
60
63
  ("grpc.max_reconnect_backoff_ms", config.retry_backoff_max_ms),
61
64
  ("grpc.min_reconnect_backoff_ms", config.stream_deadline_ms),
62
65
  ]
63
- if config.tls:
66
+ if config.default_authority is not None:
67
+ options.append(("grpc.default_authority", config.default_authority))
68
+
69
+ if config.channel_credentials is not None:
70
+ channel_args = {
71
+ "options": options,
72
+ "credentials": config.channel_credentials,
73
+ }
74
+ channel = grpc.secure_channel(target, **channel_args)
75
+
76
+ elif config.tls:
64
77
  channel_args = {
65
78
  "options": options,
66
79
  "credentials": grpc.ssl_channel_credentials(),
@@ -106,7 +119,10 @@ class GrpcWatcher(FlagStateConnector):
106
119
 
107
120
  def _state_change_callback(self, new_state: grpc.ChannelConnectivity) -> None:
108
121
  logger.debug(f"gRPC state change: {new_state}")
109
- if new_state == grpc.ChannelConnectivity.READY:
122
+ if (
123
+ new_state == grpc.ChannelConnectivity.READY
124
+ or new_state == grpc.ChannelConnectivity.IDLE
125
+ ):
110
126
  if not self.thread or not self.thread.is_alive():
111
127
  self.thread = threading.Thread(
112
128
  target=self.listen,
@@ -147,16 +163,38 @@ class GrpcWatcher(FlagStateConnector):
147
163
  self.active = False
148
164
  self.channel.close()
149
165
 
166
+ def _create_request_args(self) -> dict:
167
+ request_args = {}
168
+ if self.selector is not None:
169
+ request_args["selector"] = self.selector
170
+ if self.provider_id is not None:
171
+ request_args["provider_id"] = self.provider_id
172
+
173
+ return request_args
174
+
150
175
  def listen(self) -> None:
151
176
  call_args = (
152
177
  {"timeout": self.streamline_deadline_seconds}
153
178
  if self.streamline_deadline_seconds > 0
154
179
  else {}
155
180
  )
156
- request_args = {"selector": self.selector} if self.selector is not None else {}
181
+ request_args = self._create_request_args()
157
182
 
158
183
  while self.active:
159
184
  try:
185
+ context_values_response: sync_pb2.GetMetadataResponse
186
+ if self.config.sync_metadata_disabled:
187
+ context_values_response = sync_pb2.GetMetadataResponse(
188
+ metadata=Struct()
189
+ )
190
+ else:
191
+ context_values_request = sync_pb2.GetMetadataRequest()
192
+ context_values_response = self.stub.GetMetadata(
193
+ context_values_request, wait_for_ready=True
194
+ )
195
+
196
+ context_values = MessageToDict(context_values_response)
197
+
160
198
  request = sync_pb2.SyncFlagsRequest(**request_args)
161
199
 
162
200
  logger.debug("Setting up gRPC sync flags connection")
@@ -173,7 +211,8 @@ class GrpcWatcher(FlagStateConnector):
173
211
  self.emit_provider_ready(
174
212
  ProviderEventDetails(
175
213
  message="gRPC sync connection established"
176
- )
214
+ ),
215
+ context_values["metadata"],
177
216
  )
178
217
  self.connected = True
179
218
 
@@ -72,10 +72,8 @@ class Flag:
72
72
  data["default_variant"] = data["defaultVariant"]
73
73
  del data["defaultVariant"]
74
74
 
75
- if "source" in data:
76
- del data["source"]
77
- if "selector" in data:
78
- del data["selector"]
75
+ data.pop("source", None)
76
+ data.pop("selector", None)
79
77
  try:
80
78
  flag = cls(key=key, **data)
81
79
  return flag
@@ -0,0 +1,14 @@
1
+ import typing
2
+
3
+ from openfeature.evaluation_context import EvaluationContext
4
+ from openfeature.hook import Hook, HookContext, HookHints
5
+
6
+
7
+ class SyncMetadataHook(Hook):
8
+ def __init__(self, context_supplier: typing.Callable[[], EvaluationContext]):
9
+ self.context_supplier = context_supplier
10
+
11
+ def before(
12
+ self, hook_context: HookContext, hints: HookHints
13
+ ) -> typing.Optional[EvaluationContext]:
14
+ return self.context_supplier()
@@ -5,7 +5,7 @@ import warnings
5
5
 
6
6
  from openfeature.schemas.protobuf.flagd.evaluation.v1 import evaluation_pb2 as openfeature_dot_schemas_dot_protobuf_dot_flagd_dot_evaluation_dot_v1_dot_evaluation__pb2
7
7
 
8
- GRPC_GENERATED_VERSION = '1.70.0'
8
+ GRPC_GENERATED_VERSION = '1.71.0'
9
9
  GRPC_VERSION = grpc.__version__
10
10
  _version_not_supported = False
11
11
 
@@ -5,7 +5,7 @@ import warnings
5
5
 
6
6
  from openfeature.schemas.protobuf.flagd.sync.v1 import sync_pb2 as openfeature_dot_schemas_dot_protobuf_dot_flagd_dot_sync_dot_v1_dot_sync__pb2
7
7
 
8
- GRPC_GENERATED_VERSION = '1.70.0'
8
+ GRPC_GENERATED_VERSION = '1.71.0'
9
9
  GRPC_VERSION = grpc.__version__
10
10
  _version_not_supported = False
11
11
 
@@ -5,7 +5,7 @@ import warnings
5
5
 
6
6
  from openfeature.schemas.protobuf.schema.v1 import schema_pb2 as openfeature_dot_schemas_dot_protobuf_dot_schema_dot_v1_dot_schema__pb2
7
7
 
8
- GRPC_GENERATED_VERSION = '1.70.0'
8
+ GRPC_GENERATED_VERSION = '1.71.0'
9
9
  GRPC_VERSION = grpc.__version__
10
10
  _version_not_supported = False
11
11
 
@@ -5,7 +5,7 @@ import warnings
5
5
 
6
6
  from openfeature.schemas.protobuf.sync.v1 import sync_service_pb2 as openfeature_dot_schemas_dot_protobuf_dot_sync_dot_v1_dot_sync__service__pb2
7
7
 
8
- GRPC_GENERATED_VERSION = '1.70.0'
8
+ GRPC_GENERATED_VERSION = '1.71.0'
9
9
  GRPC_VERSION = grpc.__version__
10
10
  _version_not_supported = False
11
11
 
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: openfeature-provider-flagd
3
- Version: 0.2.0
3
+ Version: 0.2.2
4
4
  Summary: OpenFeature provider for the flagd flag evaluation engine
5
5
  Project-URL: Homepage, https://github.com/open-feature/python-sdk-contrib
6
6
  Author-email: OpenFeature <openfeature-core@groups.io>
@@ -1,35 +1,36 @@
1
1
  openfeature/.gitignore,sha256=1ZPH4VtVL4bMQmBJXQJxeAyUa74LCtqV1a_yjeDIzSQ,8
2
2
  openfeature/contrib/provider/flagd/__init__.py,sha256=WlrcPaCH31dEG1IvrvpeuhAaQ8Ni8LEzDpNM_x-qKOA,65
3
- openfeature/contrib/provider/flagd/config.py,sha256=ObWUUMVaRf199AWjfg4xrTOR7qcx5venyTNPUJkSTmA,7024
3
+ openfeature/contrib/provider/flagd/config.py,sha256=GAt8skBFMzF4skf8l8yr7jl7KBUM509ogZDpvF3NByk,8380
4
4
  openfeature/contrib/provider/flagd/flag_type.py,sha256=rZYfmqQEmtqVVTb8e-d8Wt8ZCnHtf7xPSmYxyU8w0R0,158
5
- openfeature/contrib/provider/flagd/provider.py,sha256=n0sRslXEOhFkXZnFXQguC5HkC18ekTFLKEmj7WSTJ50,6795
5
+ openfeature/contrib/provider/flagd/provider.py,sha256=wJGCFdfZlk6Y-ABU3PNSiKvrlQlNMZavsdrR2QlTqJY,7931
6
+ openfeature/contrib/provider/flagd/sync_metadata_hook.py,sha256=fd3uRtwDhVlCW9vZhS35p9qENkl6wDHwlFH-L48aPDM,456
6
7
  openfeature/contrib/provider/flagd/resolvers/__init__.py,sha256=CzsnsfxJCaD_S1gBf15kkJBVD-gVLKIwDi4W1nE-dXw,181
7
- openfeature/contrib/provider/flagd/resolvers/grpc.py,sha256=lVwznNS8WU41xIBjazxhhpJKfCLTEgnon4CHDh0DY-g,13470
8
- openfeature/contrib/provider/flagd/resolvers/in_process.py,sha256=fUE6Q1dzwE4V7KYYpdp2_qKj7vrkL47fG906kyxubr8,4618
8
+ openfeature/contrib/provider/flagd/resolvers/grpc.py,sha256=KJTrwZM-XF2-Xw-O4YvrN-Xvn_KqsO9T29lBAjiDS-4,13581
9
+ openfeature/contrib/provider/flagd/resolvers/in_process.py,sha256=Ce42cagr521rW6zW_LjT1yeVYeVqBJrNM6ASlYbdn2c,4628
9
10
  openfeature/contrib/provider/flagd/resolvers/protocol.py,sha256=Wc7oUH6ytPuKg8Lj_3y-_LLkQkuvsow9IiSfmlM8I2o,1402
10
11
  openfeature/contrib/provider/flagd/resolvers/process/custom_ops.py,sha256=bZX1At0dRxhghhWz4g-8N-YPobcVfYSv9PTsYh-lZDc,4549
11
- openfeature/contrib/provider/flagd/resolvers/process/flags.py,sha256=nDXkl80DiJBlSrz0l7xT-DAr6zzZSS0mwOSjvUAV2gY,3255
12
+ openfeature/contrib/provider/flagd/resolvers/process/flags.py,sha256=jEU-8xjGm-NXPfB6O4no4Zn8XjGTFPGRkdXx-t8uQp0,3199
12
13
  openfeature/contrib/provider/flagd/resolvers/process/targeting.py,sha256=qjYgfoJK9B1ghh19cUq7f-VEql7Hh1J7sh0BhnNc_h0,905
13
14
  openfeature/contrib/provider/flagd/resolvers/process/connector/__init__.py,sha256=hyYYxRYEnSho5F28M2hbhhtkG4DQTwJjD36ddI0Xs7M,289
14
- openfeature/contrib/provider/flagd/resolvers/process/connector/file_watcher.py,sha256=m_hscNJ0tHQgY5xuOriK20SiYKjz5FpLPoDfNJDDXKs,3889
15
- openfeature/contrib/provider/flagd/resolvers/process/connector/grpc_watcher.py,sha256=ZTHLpMZSIMl7doK75-rZF64XfxGCNmocr4Vf8UhLHTY,7276
15
+ openfeature/contrib/provider/flagd/resolvers/process/connector/file_watcher.py,sha256=f4jg9tHj94R_PJ4sCZ2f6B1r7Fgm1NqEZr68qH9pYYk,3920
16
+ openfeature/contrib/provider/flagd/resolvers/process/connector/grpc_watcher.py,sha256=XwG71pZqLLEBn7FAT0BhrO1LcCTuEwNZAXvEr37gioY,8835
16
17
  openfeature/schemas/protobuf/flagd/evaluation/v1/evaluation_pb2.py,sha256=YPBimdlVBCfnVqxq9uek3H4nyaIR0MhggCAaR0XtIt4,7527
17
18
  openfeature/schemas/protobuf/flagd/evaluation/v1/evaluation_pb2.pyi,sha256=afh-fdsmyfS62C-iq7EQsqr_5yZP_QxJXp81Q0Lcllk,18333
18
- openfeature/schemas/protobuf/flagd/evaluation/v1/evaluation_pb2_grpc.py,sha256=AFwLCOCunFSWGnlt8Asj3pxORfcx168r4jVmt9CLB0I,17189
19
+ openfeature/schemas/protobuf/flagd/evaluation/v1/evaluation_pb2_grpc.py,sha256=owH7bBrwEdzXxvabwv1ebzWVczshLFE3TZcDoAj8Owk,17189
19
20
  openfeature/schemas/protobuf/flagd/evaluation/v1/evaluation_pb2_grpc.pyi,sha256=tQm-QOlsMdhdN_7i5LsrtLtBPu4qpL0qzTE7DbSvXFs,7731
20
21
  openfeature/schemas/protobuf/flagd/sync/v1/sync_pb2.py,sha256=OIhVrNkhXtwuCdvbahXiOx-mQ3XVVPx5gh0UHzKN0_M,3528
21
22
  openfeature/schemas/protobuf/flagd/sync/v1/sync_pb2.pyi,sha256=Qsqe0vz5AJRAAjARXhjiaPKyIFbLttr4gTWae2zyJfM,5967
22
- openfeature/schemas/protobuf/flagd/sync/v1/sync_pb2_grpc.py,sha256=7sU7da4Sd8_Jg_TXuGLvNKmgzV98dBcMFKLb_EWQ0jU,8335
23
+ openfeature/schemas/protobuf/flagd/sync/v1/sync_pb2_grpc.py,sha256=gwoXi82VHekdHGuMGvFC0VbNpqVW8tnX2EkkxVDpvzU,8335
23
24
  openfeature/schemas/protobuf/flagd/sync/v1/sync_pb2_grpc.pyi,sha256=APjDd-Yo6i6FE_kOusWhTqk-8B_0wIEMXcImCuGpQt8,3782
24
25
  openfeature/schemas/protobuf/schema/v1/schema_pb2.py,sha256=dfGpiAYJ2VuRBrctH7s21KkdrIr9rzIaWibUZ3So3g4,7255
25
26
  openfeature/schemas/protobuf/schema/v1/schema_pb2.pyi,sha256=14Z6i0kRpCAQO8aCpc4Qwre1OJ7_d_ZIPq8V1vTuYzU,18369
26
- openfeature/schemas/protobuf/schema/v1/schema_pb2_grpc.py,sha256=4I0h1IXFzqAFX15J-h7yC7PQAu1CEcWEVwK4bhKLsog,16227
27
+ openfeature/schemas/protobuf/schema/v1/schema_pb2_grpc.py,sha256=af2955mIP2jB9LXZqnfN9cpQbPhl0tTg86MsQzsiy60,16227
27
28
  openfeature/schemas/protobuf/schema/v1/schema_pb2_grpc.pyi,sha256=EJ55jvso2Uzx50XryC7fsKGEOBi1bMOGCXaShl1TcJ4,7067
28
29
  openfeature/schemas/protobuf/sync/v1/sync_service_pb2.py,sha256=Sw0kLvO0wIRXsaFic8qmTuScMdf1-1WOHGWP4FOIB_s,3327
29
30
  openfeature/schemas/protobuf/sync/v1/sync_service_pb2.pyi,sha256=o3YpW_5iQJWjU7IFEsyspeqLyBDC4C_psBNErwziumE,7986
30
- openfeature/schemas/protobuf/sync/v1/sync_service_pb2_grpc.py,sha256=cAbP4qfaS2Dpze8BH3w-PAJfL7LCA7xZYJfFro5b_2I,6185
31
+ openfeature/schemas/protobuf/sync/v1/sync_service_pb2_grpc.py,sha256=Fnkq2H0sfLwc029RXfFxp5__f4UskT6Hbn_ZBT0LpJU,6185
31
32
  openfeature/schemas/protobuf/sync/v1/sync_service_pb2_grpc.pyi,sha256=4NoAkGicVNxx4yVK5_JvvPn1OGlRdPhqM2wCpkwCjEI,3024
32
- openfeature_provider_flagd-0.2.0.dist-info/METADATA,sha256=aWH2j09cJBu_IzuRgwufwnMUs9DA0jBhncib01UIOHg,21635
33
- openfeature_provider_flagd-0.2.0.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
34
- openfeature_provider_flagd-0.2.0.dist-info/licenses/LICENSE,sha256=xx0jnfkXJvxRnG63LTGOxlggYnIysveWIZ6H3PNdCrQ,11357
35
- openfeature_provider_flagd-0.2.0.dist-info/RECORD,,
33
+ openfeature_provider_flagd-0.2.2.dist-info/METADATA,sha256=-rYNDdWa7IAGVC23b4mx0BbpQdujHKRcHgm7iOi90BA,21635
34
+ openfeature_provider_flagd-0.2.2.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
35
+ openfeature_provider_flagd-0.2.2.dist-info/licenses/LICENSE,sha256=xx0jnfkXJvxRnG63LTGOxlggYnIysveWIZ6H3PNdCrQ,11357
36
+ openfeature_provider_flagd-0.2.2.dist-info/RECORD,,