v0.7.0
Frequenz Client Base Library Release Notes
Summary
This release improves the BaseApiClient
interface and introduces HTTP2 keep-alive, among other changes.
Upgrading
-
GrpcStreamBroadcaster
now takes aAsyncIterable
instead of aAsyncIterator
as thestream_method
. This is to match the type of streaming methods generated bygrpc
, so no conversion to anAsyncIterator
is needed. -
GrpcStreamBroadcaster
no longer tries to reconnect when a server closes a connection. This behaviour can be overridden by passingretry_on_exhausted_stream=True
when constructingGrpcStreamBroadcaster
instances. -
gRPC URLs don't have a default port anymore, unless a default is set via
ChannelOptions
. If you want to set a default port for URLs, please pass customChannelOptions
asdefaults
toparse_grpc_uri
or aschannel_defaults
toBaseApiClient
.
- The
ExponentialBackoff
andLinearBackoff
classes now require keyword arguments for their constructor. This change was made to make the classes easier to use and to avoid confusion with the order of the arguments.
- HTTP2 keep-alive is now enabled by default, with an interval of 60 seconds between pings, and a 20 second timeout for responses from the service. These values are configurable and may be updated based on specific requirements.
-
The
BaseApiClient
class is not generic anymore, and doesn't take a function to create the stub. Instead, subclasses should create their own stub right after calling the parent constructor. This enables subclasses to cast the stub to the generatedXxxAsyncStub
class, which have properasync
type hints. To convert you client:# Old from my_service_pb2_grpc import MyServiceStub class MyApiClient(BaseApiClient[MyServiceStub]): def __init__(self, server_url: str, *, ...) -> None: super().__init__(server_url, MyServiceStub, ...) ... # New from __future__ import annotations import my_service_pb2_grpc class MyApiClient(BaseApiClient): def __init__(self, server_url: str, *, ...) -> None: super().__init__(server_url, connect=connect) stub = my_service_pb2_grpc.MyServiceStub(self.channel) # We need the type: ignore here because the generated async stub only lives in # the .pyi file (the interpreter doesn't know anything about it) so we can't use # a proper `cast()`, we can only use the async stub as a type hint. self._stub: my_service_pb2_grpc.MyServiceAsyncStub = stub # type: ignore ... @property def stub(self) -> my_service_pb2_grpc.MyServiceAsyncStub: if self.channel is None: raise ClientNotConnected(server_url=self.server_url, operation="stub") return self._stub
You probably also need to ignore the
no-member
check frompylint
, aspylint
doesn't read*.pyi
files, so it won't find theasync
stub and complain.[tool.pylint.messages_control] disable = [ # [..] # Checked by mypy "no-member", # [..] ]
After this, you should be able to remove a lot of
cast
s ortype: ignore
from the code when calling the stubasync
methods.
New Features
- Added support for HTTP2 keep-alive.
What's Changed
- Clear release notes by @shsms in #83
- Replace the
stream_method
return toAsyncIterable
by @llucax in #87 - Don't allow a default port in URLs by default by @llucax in #85
- Bump the required group with 9 updates by @dependabot in #89
- Don't retry by default when the stream is exhausted by @shsms in #91
- Support http2 keep-alive by @shsms in #90
- Remove generic type from
BaseApiClient
by @llucax in #92 - Prepare for v0.7.0 by @shsms in #93
Full Changelog: v0.6.1...v0.7.0