Source code for agentscope.models.response

# -*- coding: utf-8 -*-
"""Parser for model response."""
import json
from typing import Optional, Sequence, Any, Generator, Union, Tuple

from ..utils.common import _is_json_serializable


[docs] class ModelResponse: """Encapsulation of data returned by the model. The main purpose of this class is to align the return formats of different models and act as a bridge between models and agents. """ def __init__( self, text: str = None, embedding: Sequence = None, image_urls: Sequence[str] = None, raw: Any = None, parsed: Any = None, stream: Optional[Generator[str, None, None]] = None, ) -> None: """Initialize the model response. Args: text (`str`, optional): The text field. embedding (`Sequence`, optional): The embedding returned by the model. image_urls (`Sequence[str]`, optional): The image URLs returned by the model. raw (`Any`, optional): The raw data returned by the model. parsed (`Any`, optional): The parsed data returned by the model. stream (`Generator`, optional): The stream data returned by the model. """ self._text = text self.embedding = embedding self.image_urls = image_urls self.raw = raw self.parsed = parsed self._stream = stream self._is_stream_exhausted = False @property def text(self) -> str: """Return the text field. If the stream field is available, the text field will be updated accordingly.""" if self._text is None: if self.stream is not None: for _, chunk in self.stream: self._text = chunk return self._text @text.setter def text(self, value: str) -> None: """Set the text field.""" self._text = value @property def stream(self) -> Union[None, Generator[Tuple[bool, str], None, None]]: """Return the stream generator if it exists.""" if self._stream is None: return self._stream else: return self._stream_generator_wrapper() @property def is_stream_exhausted(self) -> bool: """Whether the stream has been processed already.""" return self._is_stream_exhausted def _stream_generator_wrapper( self, ) -> Generator[Tuple[bool, str], None, None]: """During processing the stream generator, the text field is updated accordingly.""" if self._is_stream_exhausted: raise RuntimeError( "The stream has been processed already. Try to obtain the " "result from the text field.", ) # These two lines are used to avoid mypy checking error if self._stream is None: return try: last_text = next(self._stream) for text in self._stream: self._text = last_text yield False, last_text last_text = text self._text = last_text yield True, last_text return except StopIteration: return def __str__(self) -> str: if _is_json_serializable(self.raw): raw = self.raw else: raw = str(self.raw) serialized_fields = { "text": self.text, "embedding": self.embedding, "image_urls": self.image_urls, "parsed": self.parsed, "raw": raw, } return json.dumps(serialized_fields, indent=4, ensure_ascii=False)