Anubis.Server.Component.Tool behaviour (anubis_mcp v0.14.0)

Defines the behaviour for MCP tools.

Tools are functions that can be invoked by the client with specific parameters. Each tool must define its name, description, and parameter schema, as well as implement the execution logic.

Example

defmodule MyServer.Tools.Calculator do
  @behaviour Anubis.Server.Behaviour.Tool

  alias Anubis.Server.Frame

  @impl true
  def name, do: "calculator"

  @impl true
  def description, do: "Performs basic arithmetic operations"

  @impl true
  def input_schema do
    %{
      "type" => "object",
      "properties" => %{
        "operation" => %{
          "type" => "string",
          "enum" => ["add", "subtract", "multiply", "divide"]
        },
        "a" => %{"type" => "number"},
        "b" => %{"type" => "number"}
      },
      "required" => ["operation", "a", "b"]
    }
  end

  @impl true
  def execute(%{"operation" => "add", "a" => a, "b" => b}, frame) do
    result = a + b

    # Can access frame assigns
    user_id = frame.assigns[:user_id]

    # Can return updated frame if needed
    new_frame = Frame.assign(frame, :last_calculation, result)

    {:ok, result, new_frame}
  end

  @impl true
  def execute(%{"operation" => "divide", "a" => a, "b" => 0}, _frame) do
    {:error, "Cannot divide by zero"}
  end
end

Summary

Callbacks

Returns optional annotations for the tool.

Executes the tool with the given parameters.

Returns the JSON Schema for the tool's input parameters.

Returns the JSON Schema for the tool's output structure.

Returns the title that identifies this resource.

Types

annotations()

@type annotations() :: map() | nil

params()

@type params() :: map()

result()

@type result() :: term()

schema()

@type schema() :: map()

t()

@type t() :: %Anubis.Server.Component.Tool{
  annotations: map() | nil,
  description: String.t() | nil,
  handler: module() | nil,
  input_schema: map() | nil,
  name: String.t(),
  output_schema: map() | nil,
  title: String.t() | nil,
  validate_input: (map() -> {:ok, map()} | {:error, [Peri.Error.t()]}) | nil,
  validate_output: (map() -> {:ok, map()} | {:error, [Peri.Error.t()]}) | nil
}

Callbacks

annotations()

(optional)
@callback annotations() :: annotations()

Returns optional annotations for the tool.

Annotations provide additional metadata about the tool that may be used by clients for enhanced functionality. This is an optional callback.

Examples

def annotations do
  %{
    "confidence" => 0.95,
    "category" => "text-processing",
    "tags" => ["nlp", "text"]
  }
end

execute(params, frame)

@callback execute(params :: params(), frame :: Anubis.Server.Frame.t()) ::
  {:reply, response :: Anubis.Server.Response.t(),
   new_state :: Anubis.Server.Frame.t()}
  | {:noreply, new_state :: Anubis.Server.Frame.t()}
  | {:error, error :: Anubis.MCP.Error.t(),
     new_state :: Anubis.Server.Frame.t()}

Executes the tool with the given parameters.

Parameters

  • params - The validated input parameters from the client
  • frame - The server frame containing:
    • assigns - Custom data like session_id, client_info, user permissions
    • initialized - Whether the server has been initialized

Return Values

  • {:ok, result} - Tool executed successfully, frame unchanged
  • {:ok, result, new_frame} - Tool executed successfully with frame updates
  • {:error, reason} - Tool failed with the given reason

Frame Usage

The frame provides access to server state and context:

def execute(params, frame) do
  # Access assigns
  user_id = frame.assigns[:user_id]
  permissions = frame.assigns[:permissions]

  # Update frame if needed
  new_frame = Frame.assign(frame, :last_tool_call, DateTime.utc_now())

  {:ok, "Result", new_frame}
end

input_schema()

@callback input_schema() :: schema()

Returns the JSON Schema for the tool's input parameters.

This schema is used to validate client requests and generate documentation. The schema should follow the JSON Schema specification.

output_schema()

(optional)
@callback output_schema() :: schema()

Returns the JSON Schema for the tool's output structure.

This schema defines the expected structure of the tool's output in the structuredContent field. The schema should follow the JSON Schema specification. This is an optional callback.

title()

(optional)
@callback title() :: String.t()

Returns the title that identifies this resource.

Intended for UI and end-user contexts — optimized to be human-readable and easily understood, even by those unfamiliar with domain-specific terminology.

If not provided, the name should be used for display, except if annotations.title is defined, which takes precedence over name and title.