Source code for steamship.plugin.capabilities

"""Plugin capabilities.

Capabilities are a concept communicated back and forth between plugins and client code via blocks, and they are meant to
indicate client code's need for certain levels of support of a range of features.

Clients can request NATIVE, BEST_EFFORT, or OPTIONAL support for features that a plugin may or may not support.  Plugins
are expected to parse this and fail-fast if the user has requested support for a feature that the plugin does not
support, so that users are not e.g. billed for usage they can't incorporate.

Capability requests can include other information on the request itself, but oftentimes indicate that certain blocks
will be tagged in Steamship-native ways as part of the rest of the payload.  For example, ConversationSupport is a
capability that indicates the CHAT TagKind will be included in blocks that are part of the input, and the plugin is
expected to incorporate these with a model that supports them.

In the case that a Plugin does not support behavior indicated by the Capability request, it will throw, listing the
models that it could not support at the levels requested.  Otherwise, when Plugins respond, they'll include another
block indicating at which level they served the requested capabilities.
"""
import logging
from enum import Enum, Flag, auto
from typing import Dict, Iterable, Iterator, List, Mapping, Optional, Tuple, Type, TypeVar, Union

from pydantic import BaseModel, Extra, Field
from pydantic.dataclasses import ClassVar

from steamship import Block, MimeTypes, SteamshipError, Tag
from steamship.agents.schema import Tool
from steamship.base.client import Client
from steamship.data.block import is_block_id

CapabilityType = TypeVar("CapabilityType", bound="Capability")


[docs] class RequestLevel(Enum): """Specifies how the Plugin should handle the request for support of a specific feature.""" NATIVE = "native" """ The plugin user is requesting that the plugin supports this feature as a first-class feature. """ BEST_EFFORT = "best-effort" """ The plugin may choose to make a best effort to support the requested feature. For example, the plugin may craft a prompt to a model in a way that attempts to support the feature and parse the output appropriately. If the plugin supports the feature at a native level, but the request_level from the user was BEST_EFFORT, the plugin will still report NATIVE. It is not a requirement that plugins do this, and so many plugins may opt to only support their NATIVE levels. """ OPTIONAL = "optional" """ The plugin may ignore this feature when serving content. """ DISABLE = "disable" """ Actively disable this capability. In the case that the plugin does not support this capability, no behavior change occurs. In the case that the plugin *can't* disable this capability (it's a core part of the experience), the plugin should fast-fail. In the case that the plugin supports but can opt not to use this capability, it won't be used. """
[docs] class SupportLevel(Flag): """Flags for how plugins support capabilities. NATIVE and BEST_EFFORT *can* be OR-ed together here, but don't need to be: NATIVE implies at least BEST_EFFORT. """ NATIVE = auto() """The plugin supports this capability natively.""" BEST_EFFORT = auto() """The plugin claims to support this capability, in a way that is not native to any model.""" CAN_DISABLE = auto() """The plugin offers the ability to disable this behavior."""
[docs] class Capability(BaseModel): """Base class for all capabilities."""
[docs] class Config: extra = Extra.allow
NAME: ClassVar[str] """Name of the capability. Each capability provides its own name. When capabilities are deserialized, they become this base Capability class, with extra fields appended to them. This is not included in the init path for a class by default, with the intention that classes define their names at a class-definition-level. Make an effort to namespace these by organization, since this is technically extensible. """ name: str = Field(init_var=False, default=None) """Name of the capability as an instance field. The base class of capabilities doesn't have a NAME, but can represent the unresolved Capabilities that clients are requesting. See CapabilityImpl for the setting of this field. """ request_level: RequestLevel = RequestLevel.NATIVE
[docs] def is_plugin_support_valid( self, support_level: Optional[SupportLevel] ) -> Tuple[bool, Optional["Capability.Response"]]: """Checks if the plugin fulfills the capability request level for this specific capability. Returns a basic Response if the plugin supports the capability and the support level is at least what the consumer asked for. :param support_level: The level at which the Plugin asserts that it can support the requested capability :return: a bool that indicates whether the Plugin support level is valid, plus an Optional Response for this capability's support, if applicable. """ if ( self.request_level in (RequestLevel.OPTIONAL, RequestLevel.DISABLE) and support_level is None ): # We don't support it, but it's optional / actively disabled. return True, None if support_level is None: # Not optional / disabled, but we don't support it. return False, None if self.request_level == RequestLevel.NATIVE and SupportLevel.NATIVE not in support_level: # They want native support, but we don't support it at a native level. return False, None if ( self.request_level == RequestLevel.DISABLE and SupportLevel.CAN_DISABLE not in support_level ): return False, None # They want NATIVE support, and we support that, OR # They want BEST_EFFORT support, and we support that, OR # They want BEST_EFFORT support, and we support native return True, Capability.Response(fulfilled_at=support_level)
[docs] class Response(BaseModel): """Response regarding a specific capability served by the plugin. Responses indicate the level at which they served the request-level. """
[docs] class Config: extra = Extra.allow
name: str = Field(init_var=False, default=None) fulfilled_at: SupportLevel
[docs] class CapabilityImpl(Capability):
[docs] class Config: extra = Extra.forbid
def __init__(self, **kwargs): super().__init__(**kwargs) self.name = self.NAME
[docs] class UnsupportedCapabilityError(Exception): """ Raised when support for a given feature is requested from a plugin in a non-optional way, but the plugin doesn't support it """ def __init__(self, unsupported_capabilities: List[Capability]): names = ", ".join([cap.name for cap in unsupported_capabilities]) super().__init__( f"The following features were requested but are not supported by this plugin: {names}" ) self.unsupported_capabilities = unsupported_capabilities
[docs] class CapabilityPluginRequest(BaseModel): """Model representing the text in a STEAMSHIP_PLUGIN_CAPABILITIES block when it is used as a request.""" requested_capabilities: List[Capability]
[docs] @classmethod def from_block(cls, block: Block) -> "CapabilityPluginRequest": assert block.mime_type == MimeTypes.STEAMSHIP_PLUGIN_CAPABILITIES_REQUEST assert block.text return cls.parse_raw(block.text)
[docs] def to_block(self) -> Block: return Block(text=self.json(), mime_type=MimeTypes.STEAMSHIP_PLUGIN_CAPABILITIES_REQUEST)
[docs] def create_block(self, client: Client, file_id: str) -> Block: return Block.create( text=self.json(), mime_type=MimeTypes.STEAMSHIP_PLUGIN_CAPABILITIES_REQUEST, file_id=file_id, client=client, )
[docs] class CapabilityPluginResponse(BaseModel): """Model representing the text in a STEAMSHIP_PLUGIN_CAPABILITIES block when it is used as a response.""" capability_responses: List[Capability.Response]
[docs] @classmethod def from_block(cls, block: Block) -> "CapabilityPluginResponse": assert block.mime_type == MimeTypes.STEAMSHIP_PLUGIN_CAPABILITIES_RESPONSE assert block.text return cls.parse_raw(block.text)
[docs] def to_block(self) -> Block: return Block(text=self.json(), mime_type=MimeTypes.STEAMSHIP_PLUGIN_CAPABILITIES_RESPONSE)
[docs] def create_block(self, client: Client, file_id: str) -> Block: return Block.create( text=self.json(), mime_type=MimeTypes.STEAMSHIP_PLUGIN_CAPABILITIES_RESPONSE, file_id=file_id, client=client, )
[docs] class RequestedCapabilities: _requested: Optional[Dict[str, Capability]] _supported_levels: Dict[str, SupportLevel] _names_to_types: Dict[str, Type[CapabilityType]] def __init__(self, supported_levels: Mapping[Type[CapabilityType], SupportLevel]): self._requested = None self._supported_levels = { cap_type.NAME: request_level for cap_type, request_level in supported_levels.items() } self._names_to_types = {cap_type.NAME: cap_type for cap_type in supported_levels.keys()} def __contains__(self, typ: Type[CapabilityType]): return self.get(typ) is not None def __getitem__(self, typ: Type[CapabilityType]): v = self.get(typ) if v is None: raise KeyError(typ) return v
[docs] def load_requests(self, request: CapabilityPluginRequest) -> CapabilityPluginResponse: """ Load requests from a CapabilityPluginRequest into this mapping and verify that we can support the requested capabilities. :param request: The CapabilityPluginRequest provided to the Plugin :return: a CapabilityPluginResponse which indicates the level at which we are serving each request """ unsupported = [] responses = [] self._requested = {} for requested_capability in request.requested_capabilities: support_level = self._supported_levels.get(requested_capability.name) is_valid, response = requested_capability.is_plugin_support_valid(support_level) if not is_valid: unsupported.append(requested_capability) if response: responses.append(response) self._requested[requested_capability.name] = requested_capability if unsupported: raise UnsupportedCapabilityError(unsupported) return CapabilityPluginResponse(capability_responses=responses)
[docs] def extract_from_blocks(self, blocks: Iterable[Block]) -> Optional[CapabilityPluginResponse]: """Find the block in a list that defines capability requests, and initialize this data structure with it. It may be the case that there is no block indicating capability requests; Older clients may be passing blocks to the plugin without programmatic knowledge of capabilities. :param blocks: A list of blocks that was passed as input to a plugin :return: a CapabilityPluginResponse which indicates the level at which we are serving each request """ capabilities_block = None for block in blocks: if block.mime_type == MimeTypes.STEAMSHIP_PLUGIN_CAPABILITIES_REQUEST: if capabilities_block is not None: logging.error( f"Found more than one block with MIME_TYPE {MimeTypes.STEAMSHIP_PLUGIN_CAPABILITIES_REQUEST} in request blocks. Using first one found." ) break capabilities_block = block if not capabilities_block: return None return self.load_requests(CapabilityPluginRequest.parse_raw(capabilities_block.text))
[docs] def get( self, typ: Type[CapabilityType], default: Optional[CapabilityType] = None ) -> Optional[CapabilityType]: if self._requested is None: raise SteamshipError( "RequestedCapabilities has not been loaded with a set of requests yet. Load a request with .load_requests()" ) capability = self._requested.get(typ.NAME) if capability: return typ.parse_obj(capability) return default
[docs] class SystemPromptSupport(CapabilityImpl): """This plugin supports system prompts separate from the per-request model prompt. The system prompt will come across in other blocks on the request. """ NAME = "steamship.system_prompt_support" request_level = RequestLevel.BEST_EFFORT """ If NATIVE, asserts that the model being used supports System Prompt in a first-class capacity. If BEST_EFFORT, the system prompt will be incorporated into the request in another way, usually via concatenation with the prompt. """
[docs] class ConversationSupport(CapabilityImpl): """This plugin supports conversations. The content of the conversation will come across in other blocks on the request, using the CHAT TagKind. """ NAME = "steamship.conversation_support"
[docs] class FunctionCallingSupport(CapabilityImpl): """This plugin supports function calling. Function definitions come across as a list of Tool objects. If the plugin determines a function should be called, it will return a FunctionCallInvocation block, and then will expect a FunctionCallResult block as part of the following request. """ NAME = "steamship.function_calling_support" functions: List[Tool] """A list of Tools which the LLM can choose from to execute."""
[docs] class FunctionCallInvocation(BaseModel): """Describes a request from a plugin to invoke a function tool_name specifies the name of a Tool that was provided in FunctionCallingSupport, and args to it will be mapped to their values in the args member. A FunctionCallResult block will be expected as part of the next request. """ MIME_TYPE: ClassVar[MimeTypes] = MimeTypes.STEAMSHIP_PLUGIN_FUNCTION_CALL_INVOCATION tool_name: str """The name of the tool the plugin is requesting to call""" args: Dict[str, Union[int, str, float]] """The names of arguments that the plugin is providing for the function call, mapped to their values"""
[docs] @classmethod def from_block(cls, block: Block) -> "FunctionCallingSupport.FunctionCallInvocation": assert block.mime_type == cls.MIME_TYPE assert block.text return cls.parse_raw(block.text)
[docs] def to_block(self) -> Block: return Block(text=self.json(), mime_type=self.MIME_TYPE)
[docs] def create_block( self, client: Client, file_id: str, tags: Optional[List[Tag]] = None ) -> Block: return Block.create( text=self.json(), mime_type=self.MIME_TYPE, file_id=file_id, client=client, tags=tags, )
[docs] class FunctionCallResult(BaseModel): """Describes a result of a function call. A block with this content will be expected after the Plugin requests a call with FunctionCallInvocation. """ MIME_TYPE: ClassVar[MimeTypes] = MimeTypes.STEAMSHIP_PLUGIN_FUNCTION_CALL_RESULT tool_name: str """The name of the tool for which the result of a function call is being provided""" result: List[str] """The result of the tool invocation. Each item in the list can be text, or a Block identifier."""
[docs] @classmethod def from_block(cls, block: Block) -> "FunctionCallingSupport.FunctionCallResult": assert block.mime_type == cls.MIME_TYPE assert block.text return cls.parse_raw(block.text)
[docs] def to_block(self) -> Block: return Block(text=self.json(), mime_type=self.MIME_TYPE)
[docs] def create_block( self, client: Client, file_id: str, tags: Optional[List[Tag]] = None ) -> Block: return Block.create( text=self.json(), mime_type=self.MIME_TYPE, file_id=file_id, client=client, tags=tags, )
[docs] def result_items(self, client: Client) -> Iterator[Union[str, Block]]: for result_item in self.result: if is_block_id(result_item): yield Block.get(client, _id=result_item) else: yield result_item