Skip to content

Commit

Permalink
Add health check support to gRPC app
Browse files Browse the repository at this point in the history
Signed-off-by: Yevgen Polyak <yevgen.polyak@gmail.com>
  • Loading branch information
marcduiker authored and evhen14 committed Jun 26, 2024
1 parent 15fb0a8 commit bc6cc6f
Show file tree
Hide file tree
Showing 15 changed files with 137 additions and 40 deletions.
2 changes: 1 addition & 1 deletion .github/holopin.yml
Original file line number Diff line number Diff line change
Expand Up @@ -2,4 +2,4 @@ organization: dapr
defaultSticker: clrqfdv4x24910fl5n4iwu5oa
stickers:
- id: clrqfdv4x24910fl5n4iwu5oa
alias: { id: clrqfdv4x24910fl5n4iwu5oa, alias: sdk-badge }
alias: sdk-badge
5 changes: 2 additions & 3 deletions .github/scripts/automerge.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,9 +25,8 @@

def fetch_pulls(mergeable_state, labels={'automerge'}):
return [pr for pr in repo.get_pulls(state='open', sort='created')
# noqa: E502
if (not pr.draft) and (pr.mergeable_state == mergeable_state) and \
(not labels or len(labels.intersection({label.name for label in pr.labels})) > 0)]
if (not pr.draft and pr.mergeable_state == mergeable_state
and (not labels or len(labels.intersection({label.name for label in pr.labels})) > 0))]


def is_approved(pr):
Expand Down
4 changes: 4 additions & 0 deletions CONTRIBUTING.md
Original file line number Diff line number Diff line change
Expand Up @@ -120,6 +120,10 @@ A non-exclusive list of code that must be places in `vendor/`:

**Thank You!** - Your contributions to open source, large or small, make projects like this possible. Thank you for taking the time to contribute.

## Github Dapr Bot Commands

Checkout the [daprbot documentation](https://docs.dapr.io/contributing/daprbot/) for Github commands you can run in this repo for common tasks. For example, you can run the `/assign` (as a comment on an issue) to assign issues to a user or group of users.

## Code of Conduct

This project has adopted the [Contributor Covenant Code of Conduct](https://github.com/dapr/community/blob/master/CODE-OF-CONDUCT.md)
16 changes: 9 additions & 7 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,12 +1,14 @@
# Dapr SDK for Python

[![PyPI version](https://badge.fury.io/py/dapr.svg)](https://badge.fury.io/py/dapr)
[![PyPI version](https://badge.fury.io/py/dapr-dev.svg)](https://badge.fury.io/py/dapr-dev)
![dapr-python](https://github.com/dapr/python-sdk/workflows/dapr-python/badge.svg?branch=master)
[![codecov](https://codecov.io/gh/dapr/python-sdk/branch/master/graph/badge.svg)](https://codecov.io/gh/dapr/python-sdk)
[![Discord](https://img.shields.io/discord/778680217417809931)](https://discord.com/channels/778680217417809931/778680217417809934)
[![License: Apache](https://img.shields.io/badge/License-Apache-yellow.svg)](http://www.apache.org/licenses/LICENSE-2.0)
[![FOSSA Status](https://app.fossa.com/api/projects/custom%2B162%2Fgithub.com%2Fdapr%2Fpython-sdk.svg?type=shield)](https://app.fossa.com/projects/custom%2B162%2Fgithub.com%2Fdapr%2Fpython-sdk?ref=badge_shield)
[![PyPI - Version](https://img.shields.io/pypi/v/dapr?style=flat&logo=pypi&logoColor=white&label=Latest%20version)](https://pypi.org/project/dapr/)
[![PyPI - Downloads](https://img.shields.io/pypi/dm/dapr?style=flat&logo=pypi&logoColor=white&label=Downloads)](https://pypi.org/project/dapr/)
[![GitHub Actions Workflow Status](https://img.shields.io/github/actions/workflow/status/dapr/python-sdk/.github%2Fworkflows%2Fbuild.yaml?branch=main&label=Build&logo=github)](https://github.com/dapr/python-sdk/actions/workflows/build.yaml)
[![codecov](https://codecov.io/gh/dapr/python-sdk/branch/main/graph/badge.svg)](https://codecov.io/gh/dapr/python-sdk)
[![GitHub License](https://img.shields.io/github/license/dapr/python-sdk?style=flat&label=License&logo=github)](https://github.com/dapr/python-sdk/blob/main/LICENSE)
[![GitHub issue custom search in repo](https://img.shields.io/github/issues-search/dapr/python-sdk?query=type%3Aissue%20is%3Aopen%20label%3A%22good%20first%20issue%22&label=Good%20first%20issues&style=flat&logo=github)](https://github.com/dapr/python-sdk/issues?q=is%3Aissue+is%3Aopen+label%3A%22good+first+issue%22)
[![Discord](https://img.shields.io/discord/778680217417809931?label=Discord&style=flat&logo=discord)](http://bit.ly/dapr-discord)
[![YouTube Channel Views](https://img.shields.io/youtube/channel/views/UCtpSQ9BLB_3EXdWAUQYwnRA?style=flat&label=YouTube%20views&logo=youtube)](https://youtube.com/@daprdev)
[![X (formerly Twitter) Follow](https://img.shields.io/twitter/follow/daprdev?logo=x&style=flat)](https://twitter.com/daprdev)

[Dapr](https://docs.dapr.io/concepts/overview/) is a portable, event-driven, serverless runtime for building distributed applications across cloud and edge.

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,4 +20,8 @@ The `examples` directory contains code samples for users to run to try out speci
The `daprdocs` directory contains the markdown files that are rendered into the [Dapr Docs](https://docs.dapr.io) website. When the documentation website is built this repo is cloned and configured so that its contents are rendered with the docs content. When writing docs keep in mind:

- All rules in the [docs guide]({{< ref contributing-docs.md >}}) should be followed in addition to these.
- All files and directories should be prefixed with `python-` to ensure all file/directory names are globally unique across all Dapr documentation.
- All files and directories should be prefixed with `python-` to ensure all file/directory names are globally unique across all Dapr documentation.

## Github Dapr Bot Commands

Checkout the [daprbot documentation](https://docs.dapr.io/contributing/daprbot/) for Github commands you can run in this repo for common tasks. For example, you can run the `/assign` (as a comment on an issue) to assign issues to a user or group of users.
2 changes: 1 addition & 1 deletion examples/pubsub-simple/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -66,7 +66,7 @@ sleep: 15

```bash
# 2. Start Publisher
dapr run --app-id python-publisher --app-protocol grpc --dapr-grpc-port=3500 python3 publisher.py
dapr run --app-id python-publisher --app-protocol grpc --dapr-grpc-port=3500 --enable-app-health-check python3 publisher.py
```

<!-- END_STEP -->
Expand Down
7 changes: 7 additions & 0 deletions examples/pubsub-simple/subscriber.py
Original file line number Diff line number Diff line change
Expand Up @@ -83,4 +83,11 @@ def mytopic_wildcard(event: v1.Event) -> TopicEventResponse:
return TopicEventResponse('success')


# Example of an unhealthy status
# def unhealthy():
# raise ValueError("Not healthy")
# app.register_health_check(unhealthy)

app.register_health_check(lambda: print('Healthy'))

app.run(50051)
3 changes: 2 additions & 1 deletion examples/workflow/requirements.txt
Original file line number Diff line number Diff line change
@@ -1 +1,2 @@
dapr-ext-workflow>=0.1.0
dapr-ext-workflow-dev>=0.4.1rc1.dev
dapr-dev>=1.13.0rc1.dev
32 changes: 32 additions & 0 deletions ext/dapr-ext-grpc/dapr/ext/grpc/_health_servicer.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
import grpc
from typing import Callable, Optional

from dapr.proto import appcallback_service_v1
from dapr.proto.runtime.v1.appcallback_pb2 import HealthCheckResponse

HealthCheckCallable = Optional[Callable[[], None]]


class _HealthCheckServicer(appcallback_service_v1.AppCallbackHealthCheckServicer):
"""The implementation of HealthCheck Server.
:class:`App` provides useful decorators to register method, topic, input bindings.
"""

def __init__(self):
self._health_check_cb: Optional[HealthCheckCallable] = None

def register_health_check(self, cb: HealthCheckCallable) -> None:
if not cb:
raise ValueError('health check callback must be defined')
self._health_check_cb = cb

def HealthCheck(self, request, context):
"""Health check."""

if not self._health_check_cb:
context.set_code(grpc.StatusCode.UNIMPLEMENTED) # type: ignore
context.set_details('Method not implemented!')
raise NotImplementedError('Method not implemented!')
self._health_check_cb()
return HealthCheckResponse()
16 changes: 15 additions & 1 deletion ext/dapr-ext-grpc/dapr/ext/grpc/app.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,8 @@
from typing import Dict, Optional

from dapr.conf import settings
from dapr.ext.grpc._servicier import _CallbackServicer, Rule # type: ignore
from dapr.ext.grpc._servicer import _CallbackServicer, Rule # type: ignore
from dapr.ext.grpc._health_servicer import _HealthCheckServicer # type: ignore
from dapr.proto import appcallback_service_v1


Expand All @@ -43,6 +44,7 @@ def __init__(self, max_grpc_message_length: Optional[int] = None, **kwargs):
kwargs: arguments to grpc.server()
"""
self._servicer = _CallbackServicer()
self._health_check_servicer = _HealthCheckServicer()
if not kwargs:
options = []
if max_grpc_message_length is not None:
Expand All @@ -56,6 +58,9 @@ def __init__(self, max_grpc_message_length: Optional[int] = None, **kwargs):
else:
self._server = grpc.server(**kwargs) # type: ignore
appcallback_service_v1.add_AppCallbackServicer_to_server(self._servicer, self._server)
appcallback_service_v1.add_AppCallbackHealthCheckServicer_to_server(
self._health_check_servicer, self._server
)

def __del__(self):
self.stop()
Expand All @@ -64,6 +69,15 @@ def add_external_service(self, servicer_callback, external_servicer):
"""Adds an external gRPC service to the same server"""
servicer_callback(external_servicer, self._server)

def register_health_check(self, health_check_callback):
"""Adds a health check callback
The below example adds a basic health check to check Dapr gRPC is running
@app.register_health_check(lambda: None)
"""
self._health_check_servicer.register_health_check(health_check_callback)

def run(self, app_port: Optional[int] = None, listen_address: Optional[str] = None) -> None:
"""Starts app gRPC server and waits until :class:`App`.stop() is called.
Expand Down
14 changes: 14 additions & 0 deletions ext/dapr-ext-grpc/tests/test_app.py
Original file line number Diff line number Diff line change
Expand Up @@ -74,3 +74,17 @@ def handle_dead_letter(event: v1.Event) -> None:
'AppTests.test_subscribe_decorator.<locals>.handle_dead_letter',
str(subscription_map['pubsub:topic2:']),
)

def test_register_health_check(self):
def health_check_cb():
pass

self._app.register_health_check(health_check_cb)
registered_cb = self._app._health_check_servicer._health_check_cb
self.assertIn(
'AppTests.test_register_health_check.<locals>.health_check_cb', str(registered_cb)
)

def test_no_health_check(self):
registered_cb = self._app._health_check_servicer._health_check_cb
self.assertIsNone(registered_cb)
20 changes: 20 additions & 0 deletions ext/dapr-ext-grpc/tests/test_health_servicer.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
import unittest
from unittest.mock import MagicMock

from dapr.ext.grpc._health_servicer import _HealthCheckServicer


class OnInvokeTests(unittest.TestCase):
def setUp(self):
self._health_servicer = _HealthCheckServicer()

def test_healthcheck_cb_called(self):
health_cb = MagicMock()
self._health_servicer.register_health_check(health_cb)
self._health_servicer.HealthCheck(None, MagicMock())
health_cb.assert_called_once()

def test_no_healthcheck_cb(self):
with self.assertRaises(NotImplementedError) as exception_context:
self._health_servicer.HealthCheck(None, MagicMock())
self.assertIn('Method not implemented!', exception_context.exception.args[0])
48 changes: 24 additions & 24 deletions ext/dapr-ext-grpc/tests/test_servicier.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,18 +19,18 @@

from dapr.clients.grpc._request import InvokeMethodRequest
from dapr.clients.grpc._response import InvokeMethodResponse, TopicEventResponse
from dapr.ext.grpc._servicier import _CallbackServicer
from dapr.ext.grpc._servicer import _CallbackServicer
from dapr.proto import common_v1, appcallback_v1

from google.protobuf.any_pb2 import Any as GrpcAny


class OnInvokeTests(unittest.TestCase):
def setUp(self):
self._servicier = _CallbackServicer()
self._servicer = _CallbackServicer()

def _on_invoke(self, method_name, method_cb):
self._servicier.register_method(method_name, method_cb)
self._servicer.register_method(method_name, method_cb)

# fake context
fake_context = MagicMock()
Expand All @@ -39,7 +39,7 @@ def _on_invoke(self, method_name, method_cb):
('key2', 'value1'),
)

return self._servicier.OnInvoke(
return self._servicer.OnInvoke(
common_v1.InvokeRequest(method=method_name, data=GrpcAny()),
fake_context,
)
Expand Down Expand Up @@ -93,18 +93,18 @@ def method_cb(request: InvokeMethodRequest):

class TopicSubscriptionTests(unittest.TestCase):
def setUp(self):
self._servicier = _CallbackServicer()
self._servicer = _CallbackServicer()
self._topic1_method = Mock()
self._topic2_method = Mock()
self._topic3_method = Mock()
self._topic3_method.return_value = TopicEventResponse('success')
self._topic4_method = Mock()

self._servicier.register_topic('pubsub1', 'topic1', self._topic1_method, {'session': 'key'})
self._servicier.register_topic('pubsub1', 'topic3', self._topic3_method, {'session': 'key'})
self._servicier.register_topic('pubsub2', 'topic2', self._topic2_method, {'session': 'key'})
self._servicier.register_topic('pubsub2', 'topic3', self._topic3_method, {'session': 'key'})
self._servicier.register_topic(
self._servicer.register_topic('pubsub1', 'topic1', self._topic1_method, {'session': 'key'})
self._servicer.register_topic('pubsub1', 'topic3', self._topic3_method, {'session': 'key'})
self._servicer.register_topic('pubsub2', 'topic2', self._topic2_method, {'session': 'key'})
self._servicer.register_topic('pubsub2', 'topic3', self._topic3_method, {'session': 'key'})
self._servicer.register_topic(
'pubsub3',
'topic4',
self._topic4_method,
Expand All @@ -121,12 +121,12 @@ def setUp(self):

def test_duplicated_topic(self):
with self.assertRaises(ValueError):
self._servicier.register_topic(
self._servicer.register_topic(
'pubsub1', 'topic1', self._topic1_method, {'session': 'key'}
)

def test_list_topic_subscription(self):
resp = self._servicier.ListTopicSubscriptions(None, None)
resp = self._servicer.ListTopicSubscriptions(None, None)
self.assertEqual('pubsub1', resp.subscriptions[0].pubsub_name)
self.assertEqual('topic1', resp.subscriptions[0].topic)
self.assertEqual({'session': 'key'}, resp.subscriptions[0].metadata)
Expand All @@ -143,23 +143,23 @@ def test_list_topic_subscription(self):
self.assertEqual({'session': 'key'}, resp.subscriptions[4].metadata)

def test_topic_event(self):
self._servicier.OnTopicEvent(
self._servicer.OnTopicEvent(
appcallback_v1.TopicEventRequest(pubsub_name='pubsub1', topic='topic1'),
self.fake_context,
)

self._topic1_method.assert_called_once()

def test_topic3_event_called_once(self):
self._servicier.OnTopicEvent(
self._servicer.OnTopicEvent(
appcallback_v1.TopicEventRequest(pubsub_name='pubsub1', topic='topic3'),
self.fake_context,
)

self._topic3_method.assert_called_once()

def test_topic3_event_response(self):
response = self._servicier.OnTopicEvent(
response = self._servicer.OnTopicEvent(
appcallback_v1.TopicEventRequest(pubsub_name='pubsub1', topic='topic3'),
self.fake_context,
)
Expand All @@ -169,7 +169,7 @@ def test_topic3_event_response(self):
)

def test_disable_topic_validation(self):
self._servicier.OnTopicEvent(
self._servicer.OnTopicEvent(
appcallback_v1.TopicEventRequest(pubsub_name='pubsub3', topic='should_be_ignored'),
self.fake_context,
)
Expand All @@ -178,20 +178,20 @@ def test_disable_topic_validation(self):

def test_non_registered_topic(self):
with self.assertRaises(NotImplementedError):
self._servicier.OnTopicEvent(
self._servicer.OnTopicEvent(
appcallback_v1.TopicEventRequest(pubsub_name='pubsub1', topic='topic_non_existed'),
self.fake_context,
)


class BindingTests(unittest.TestCase):
def setUp(self):
self._servicier = _CallbackServicer()
self._servicer = _CallbackServicer()
self._binding1_method = Mock()
self._binding2_method = Mock()

self._servicier.register_binding('binding1', self._binding1_method)
self._servicier.register_binding('binding2', self._binding2_method)
self._servicer.register_binding('binding1', self._binding1_method)
self._servicer.register_binding('binding2', self._binding2_method)

# fake context
self.fake_context = MagicMock()
Expand All @@ -202,15 +202,15 @@ def setUp(self):

def test_duplicated_binding(self):
with self.assertRaises(ValueError):
self._servicier.register_binding('binding1', self._binding1_method)
self._servicer.register_binding('binding1', self._binding1_method)

def test_list_bindings(self):
resp = self._servicier.ListInputBindings(None, None)
resp = self._servicer.ListInputBindings(None, None)
self.assertEqual('binding1', resp.bindings[0])
self.assertEqual('binding2', resp.bindings[1])

def test_binding_event(self):
self._servicier.OnBindingEvent(
self._servicer.OnBindingEvent(
appcallback_v1.BindingEventRequest(name='binding1'),
self.fake_context,
)
Expand All @@ -219,7 +219,7 @@ def test_binding_event(self):

def test_non_registered_binding(self):
with self.assertRaises(NotImplementedError):
self._servicier.OnBindingEvent(
self._servicer.OnBindingEvent(
appcallback_v1.BindingEventRequest(name='binding3'),
self.fake_context,
)
Expand Down
2 changes: 1 addition & 1 deletion ext/dapr-ext-workflow/dapr/ext/workflow/version.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,4 +13,4 @@
limitations under the License.
"""

__version__ = '0.4.0rc1.dev'
__version__ = '0.4.1rc1.dev'

0 comments on commit bc6cc6f

Please sign in to comment.