-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathendorsement_processor.py
213 lines (181 loc) · 8.27 KB
/
endorsement_processor.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
import asyncio
from typing import List, NoReturn
from aries_cloudcontroller import AcaPyClient
from nats.errors import BadSubscriptionError, Error, TimeoutError
from nats.js.client import JetStreamContext
from nats.js.errors import FetchTimeoutError
from endorser.util.endorsement import accept_endorsement, should_accept_endorsement
from shared.constants import (
ENDORSER_DURABLE_CONSUMER,
GOVERNANCE_AGENT_API_KEY,
GOVERNANCE_AGENT_URL,
GOVERNANCE_LABEL,
NATS_STREAM,
NATS_SUBJECT,
)
from shared.log_config import get_logger
from shared.models.endorsement import Endorsement
from shared.models.webhook_events.payloads import CloudApiWebhookEventGeneric
from shared.util.rich_parsing import parse_json_with_error_handling
logger = get_logger(__name__)
class EndorsementProcessor:
"""
Class to process endorsement webhook events that Benthos acapy-events-processor writes to `endorser_nats_subject`
"""
def __init__(self, jetstream: JetStreamContext) -> None:
self.jetstream: JetStreamContext = jetstream
self.endorser_nats_subject = f"{NATS_SUBJECT}.endorser.*"
self._tasks: List[asyncio.Task] = [] # To keep track of running tasks
def start(self) -> None:
"""
Starts the background tasks for processing endorsement events.
"""
self._tasks.append(
asyncio.create_task(
self._process_endorsement_requests(), name="Process endorsements"
)
)
logger.info("Endorsement processing started.")
async def stop(self) -> None:
"""
Stops all background tasks gracefully.
"""
for task in self._tasks:
task.cancel() # Request cancellation of the task
try:
await task # Wait for the task to be cancelled
except asyncio.CancelledError:
pass # Expected error upon cancellation, can be ignored
self._tasks.clear() # Clear the list of tasks
logger.info("Endorsement processing stopped.")
def are_tasks_running(self) -> bool:
"""
Checks if the background tasks are still running.
Returns:
True if all background tasks are running, False if any task has stopped.
"""
logger.debug("Checking if all tasks are running")
tasks_running = self._tasks and all(not task.done() for task in self._tasks)
if not tasks_running:
for task in self._tasks:
if task.done():
logger.error("Task `{}` is not running", task.get_name())
all_running = tasks_running
logger.debug("All tasks running: {}", all_running)
return all_running
async def _process_endorsement_requests(self) -> NoReturn:
subscription = await self._subscribe()
while True:
try:
messages = await subscription.fetch(batch=1, timeout=5, heartbeat=1)
for message in messages:
message_subject = message.subject
message_data = message.data.decode()
logger.trace(
"received message: {} with subject {}",
message_data,
message_subject,
)
try:
await self._process_endorsement_event(message_data)
except Exception as e: # pylint: disable=W0703
logger.error("Error processing endorsement event: {}", e)
await self._handle_unprocessable_endorse_event(
message_subject, message_data, e
)
finally:
await message.ack()
except FetchTimeoutError:
logger.trace("FetchTimeoutError continuing...")
await asyncio.sleep(0.1)
except TimeoutError as e:
logger.warning("Timeout error fetching messages re-subscribing: {}", e)
await subscription.unsubscribe()
subscription = await self._subscribe()
except Exception: # pylint: disable=W0718
logger.exception("Unexpected error in endorsement processing loop")
await asyncio.sleep(2)
async def _process_endorsement_event(self, event_json: str) -> None:
"""
Processes an individual endorsement event, evaluating if it should be accepted and then endorsing as governance
Args:
event_json: The JSON string representation of the endorsement event.
"""
event = parse_json_with_error_handling(
CloudApiWebhookEventGeneric, event_json, logger
)
logger.debug("Processing endorsement event for agent `{}`", event.origin)
# We're only interested in events from the governance agent
if event.wallet_id != GOVERNANCE_LABEL:
logger.warning("Endorsement request is not for governance agent.")
return
endorsement = Endorsement(**event.payload)
async with AcaPyClient(
base_url=GOVERNANCE_AGENT_URL, api_key=GOVERNANCE_AGENT_API_KEY
) as client:
# Check if endorsement request is indeed applicable
if not await should_accept_endorsement(client, endorsement):
logger.info( # check already logged the reason as warning
"Endorsement request with transaction id `{}` is not applicable for endorsement.",
endorsement.transaction_id,
)
return
logger.info(
"Endorsement request with transaction id `{}` is applicable for endorsement, accepting request.",
endorsement.transaction_id,
)
await accept_endorsement(client, endorsement)
async def _handle_unprocessable_endorse_event(
self, key: str, event_json: str, error: Exception
) -> None:
"""
Handles an event that could not be processed successfully. The unprocessable event is persisted
to a separate key for further investigation.
Args:
key: The Nats subject key where the problematic event was found.
error: The exception that occurred during event processing.
"""
bound_logger = logger.bind(body={"key": key})
bound_logger.warning("Handling problematic endorsement event")
unprocessable_key = f"unprocessable.{key}"
error_message = f"Could not process: {event_json}. Error: {error}"
bound_logger.info(
"Saving record of problematic event at key: {}. Error: `{}`",
unprocessable_key,
error_message,
)
await self.jetstream.publish(unprocessable_key, error_message.encode())
bound_logger.info("Successfully handled unprocessable event.")
async def _subscribe(self) -> JetStreamContext.PullSubscription:
"""
Subscribes to the NATS subject for endorsement events.
"""
logger.info("Subscribing to NATS subject: {}", self.endorser_nats_subject)
try:
subscribe_kwargs = {
"subject": self.endorser_nats_subject,
"durable": ENDORSER_DURABLE_CONSUMER,
"stream": NATS_STREAM,
}
subscription = await self.jetstream.pull_subscribe(**subscribe_kwargs)
except (BadSubscriptionError, Error) as e:
logger.error("Error subscribing to NATS subject: {}", e)
raise e
except Exception: # pylint: disable=W0703
logger.exception("Unknown error subscribing to NATS subject")
raise
logger.debug("Subscribed to NATS subject")
return subscription
async def check_jetstream(self):
try:
account_info = await self.jetstream.account_info()
is_working = account_info.streams > 0
logger.trace("JetStream check completed. Is working: {}", is_working)
return {
"is_working": is_working,
"streams_count": account_info.streams,
"consumers_count": account_info.consumers,
}
except Exception: # pylint: disable=W0718
logger.exception("Caught exception while checking jetstream status")
return {"is_working": False}