Hermes.Server.Component.Resource behaviour (hermes_mcp v0.9.1)

Defines the behaviour for MCP resources.

Resources represent data that can be read by the client, such as files, documents, or any other content. Each resource is identified by a URI and can provide content in various formats.

Example

defmodule MyServer.Resources.Documentation do
  @behaviour Hermes.Server.Behaviour.Resource

  alias Hermes.Server.Frame

  @impl true
  def uri, do: "file:///docs/readme.md"

  @impl true
  def name, do: "Project README"

  @impl true
  def description, do: "The main documentation for this project"

  @impl true
  def mime_type, do: "text/markdown"

  @impl true
  def read(_params, frame) do
    case File.read("README.md") do
      {:ok, content} -> 
        # Can track access in frame
        new_frame = Frame.assign(frame, :last_resource_access, DateTime.utc_now())
        {:ok, content, new_frame}

      {:error, reason} -> 
        {:error, "Failed to read README: #{inspect(reason)}"}
    end
  end
end

Example with dynamic content

defmodule MyServer.Resources.SystemStatus do
  @behaviour Hermes.Server.Behaviour.Resource

  @impl true
  def uri, do: "system://status"

  @impl true
  def name, do: "System Status"

  @impl true
  def description, do: "Current system status and metrics"

  @impl true
  def mime_type, do: "application/json"

  @impl true
  def read(_params, frame) do
    status = %{
      uptime: System.uptime(),
      memory: :erlang.memory(),
      user_id: frame.assigns[:user_id],
      timestamp: DateTime.utc_now()
    }

    {:ok, Jason.encode!(status), frame}
  end
end

Summary

Callbacks

Returns the MIME type of the resource content.

Reads the resource content.

Returns the URI that identifies this resource.

Functions

Validates that a module implements the Resource behaviour.

Converts a resource module into the MCP protocol format.

Types

content()

@type content() :: binary() | String.t()

params()

@type params() :: map()

Callbacks

mime_type()

@callback mime_type() :: String.t()

Returns the MIME type of the resource content.

Common MIME types:

  • text/plain for plain text
  • text/markdown for Markdown
  • application/json for JSON data
  • text/html for HTML
  • application/octet-stream for binary data

read(params, frame)

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

Reads the resource content.

Parameters

  • params - Optional parameters from the client (typically empty for resources)
  • frame - The server frame containing context and state

Return Values

  • {:ok, content} - Resource read successfully, frame unchanged
  • {:ok, content, new_frame} - Resource read successfully with frame updates
  • {:error, reason} - Failed to read resource

Content Types

The content should match the declared MIME type:

  • For text types, return a String
  • For binary types, return binary data
  • For JSON, return the JSON-encoded string

uri()

@callback uri() :: String.t()

Returns the URI that identifies this resource.

The URI should be unique within the server and follow standard URI conventions. Common schemes include:

  • file:// for file-based resources
  • http:// or https:// for web resources
  • Custom schemes for application-specific resources

Functions

implements?(module)

@spec implements?(module()) :: boolean()

Validates that a module implements the Resource behaviour.

to_protocol(resource_module)

@spec to_protocol(module()) :: map()

Converts a resource module into the MCP protocol format.