2024-07-09 10:46:55 -07:00
|
|
|
import json
|
2024-07-09 13:51:05 -07:00
|
|
|
import logging
|
2024-07-17 09:51:19 -07:00
|
|
|
from dataclasses import asdict
|
2024-07-09 13:51:05 -07:00
|
|
|
from datetime import datetime
|
2024-07-11 10:52:29 -07:00
|
|
|
from typing import Any, Dict, List, Literal
|
2024-07-09 10:46:55 -07:00
|
|
|
|
2024-12-03 17:00:44 -08:00
|
|
|
from autogen_core import Image
|
2024-12-09 13:00:08 -05:00
|
|
|
from autogen_core.logging import LLMCallEvent
|
2024-07-09 10:46:55 -07:00
|
|
|
|
2024-07-17 09:51:19 -07:00
|
|
|
from .messages import (
|
2024-07-26 09:42:12 -07:00
|
|
|
AgentEvent,
|
2024-07-17 09:51:19 -07:00
|
|
|
AssistantContent,
|
|
|
|
FunctionExecutionContent,
|
|
|
|
OrchestrationEvent,
|
|
|
|
SystemContent,
|
|
|
|
UserContent,
|
|
|
|
WebSurferEvent,
|
|
|
|
)
|
2024-07-09 13:51:05 -07:00
|
|
|
|
|
|
|
|
2024-07-10 00:01:13 -07:00
|
|
|
# Convert UserContent to a string
|
|
|
|
def message_content_to_str(
|
|
|
|
message_content: UserContent | AssistantContent | SystemContent | FunctionExecutionContent,
|
|
|
|
) -> str:
|
|
|
|
if isinstance(message_content, str):
|
|
|
|
return message_content
|
|
|
|
elif isinstance(message_content, List):
|
|
|
|
converted: List[str] = list()
|
|
|
|
for item in message_content:
|
|
|
|
if isinstance(item, str):
|
|
|
|
converted.append(item.rstrip())
|
2024-07-17 09:51:19 -07:00
|
|
|
elif isinstance(item, Image):
|
|
|
|
converted.append("<Image>")
|
2024-07-10 00:01:13 -07:00
|
|
|
else:
|
|
|
|
converted.append(str(item).rstrip())
|
|
|
|
return "\n".join(converted)
|
|
|
|
else:
|
|
|
|
raise AssertionError("Unexpected response type.")
|
|
|
|
|
|
|
|
|
2024-10-01 15:59:03 -07:00
|
|
|
# MagenticOne log event handler
|
2024-07-12 15:21:45 -07:00
|
|
|
class LogHandler(logging.FileHandler):
|
|
|
|
def __init__(self, filename: str = "log.jsonl") -> None:
|
|
|
|
super().__init__(filename)
|
2024-11-04 17:18:46 -08:00
|
|
|
self.logs_list: List[Dict[str, Any]] = []
|
2024-07-09 13:51:05 -07:00
|
|
|
|
|
|
|
def emit(self, record: logging.LogRecord) -> None:
|
2024-07-29 13:09:31 -07:00
|
|
|
try:
|
|
|
|
ts = datetime.fromtimestamp(record.created).isoformat()
|
|
|
|
if isinstance(record.msg, OrchestrationEvent):
|
|
|
|
console_message = (
|
|
|
|
f"\n{'-'*75} \n" f"\033[91m[{ts}], {record.msg.source}:\033[0m\n" f"\n{record.msg.message}"
|
|
|
|
)
|
|
|
|
print(console_message, flush=True)
|
|
|
|
record.msg = json.dumps(
|
|
|
|
{
|
|
|
|
"timestamp": ts,
|
|
|
|
"source": record.msg.source,
|
|
|
|
"message": record.msg.message,
|
|
|
|
"type": "OrchestrationEvent",
|
|
|
|
}
|
|
|
|
)
|
2024-11-04 17:18:46 -08:00
|
|
|
self.logs_list.append(json.loads(record.msg))
|
2024-07-29 13:09:31 -07:00
|
|
|
super().emit(record)
|
|
|
|
elif isinstance(record.msg, AgentEvent):
|
|
|
|
console_message = (
|
|
|
|
f"\n{'-'*75} \n" f"\033[91m[{ts}], {record.msg.source}:\033[0m\n" f"\n{record.msg.message}"
|
|
|
|
)
|
|
|
|
print(console_message, flush=True)
|
|
|
|
record.msg = json.dumps(
|
|
|
|
{
|
|
|
|
"timestamp": ts,
|
|
|
|
"source": record.msg.source,
|
|
|
|
"message": record.msg.message,
|
|
|
|
"type": "AgentEvent",
|
|
|
|
}
|
|
|
|
)
|
2024-11-04 17:18:46 -08:00
|
|
|
self.logs_list.append(json.loads(record.msg))
|
2024-07-29 13:09:31 -07:00
|
|
|
super().emit(record)
|
|
|
|
elif isinstance(record.msg, WebSurferEvent):
|
|
|
|
console_message = f"\033[96m[{ts}], {record.msg.source}: {record.msg.message}\033[0m"
|
|
|
|
print(console_message, flush=True)
|
|
|
|
payload: Dict[str, Any] = {
|
2024-07-26 09:42:12 -07:00
|
|
|
"timestamp": ts,
|
2024-07-29 13:09:31 -07:00
|
|
|
"type": "WebSurferEvent",
|
2024-07-26 09:42:12 -07:00
|
|
|
}
|
2024-07-29 13:09:31 -07:00
|
|
|
payload.update(asdict(record.msg))
|
|
|
|
record.msg = json.dumps(payload)
|
2024-11-04 17:18:46 -08:00
|
|
|
self.logs_list.append(json.loads(record.msg))
|
2024-07-29 13:09:31 -07:00
|
|
|
super().emit(record)
|
|
|
|
elif isinstance(record.msg, LLMCallEvent):
|
|
|
|
record.msg = json.dumps(
|
|
|
|
{
|
|
|
|
"timestamp": ts,
|
|
|
|
"prompt_tokens": record.msg.prompt_tokens,
|
|
|
|
"completion_tokens": record.msg.completion_tokens,
|
|
|
|
"type": "LLMCallEvent",
|
|
|
|
}
|
|
|
|
)
|
2024-11-04 17:18:46 -08:00
|
|
|
self.logs_list.append(json.loads(record.msg))
|
2024-07-29 13:09:31 -07:00
|
|
|
super().emit(record)
|
|
|
|
except Exception:
|
|
|
|
self.handleError(record)
|
2024-07-11 10:52:29 -07:00
|
|
|
|
|
|
|
|
|
|
|
class SentinelMeta(type):
|
|
|
|
"""
|
|
|
|
A baseclass for sentinels that plays well with type hints.
|
|
|
|
Define new sentinels like this:
|
|
|
|
|
|
|
|
```
|
|
|
|
class MY_DEFAULT(metaclass=SentinelMeta):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
foo: list[str] | None | type[MY_DEFAULT] = MY_DEFAULT
|
|
|
|
```
|
|
|
|
|
|
|
|
Reference: https://stackoverflow.com/questions/69239403/type-hinting-parameters-with-a-sentinel-value-as-the-default
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __repr__(cls) -> str:
|
|
|
|
return f"<{cls.__name__}>"
|
|
|
|
|
|
|
|
def __bool__(cls) -> Literal[False]:
|
|
|
|
return False
|