Resources
Resources are a core primitive in the Model Context Protocol (MCP) that allow servers to expose data and content that can be read by clients and used as context for LLM interactions.
MCP resources is a concept of MCP servers. Server should list resources when requested with method resources/list
and retrieve when requested with method resources/read
.
Servers also can provide dynamic resources using templates by listing them via resources/templates/list
, however this feature is not yet supported in Foxy Contexts.
In Foxy Contexts there are two ways to include resources in your server:
- using
fxctx.NewResource
to define static resources - using
fxctx.NewResourceProvider
to define resource providers
Approach with resource provider is more flexible and allows to provide resources dynamically, however all such resources would be still included in response for resources/list
, in contrasts to concept of templates (which are not yet supported).
NewResource
In order to create new static resource you shall use fxctx.NewResource
function.
func NewGreatResource() fxctx.Resource {
return fxctx.NewResource(
mcp.Resource{
Name: "hello-world",
Uri: "hello-world://hello-world",
MimeType: Ptr("application/json"),
Description: Ptr("Hello World Resource"),
Annotations: &mcp.ResourceAnnotations{
Audience: []mcp.Role{
mcp.RoleAssistant, mcp.RoleUser,
},
},
},
func(_ context.Context, uri string) (*mcp.ReadResourceResult, error) {
return &mcp.ReadResourceResult{
Contents: []interface{}{
mcp.TextResourceContents{
MimeType: Ptr("application/json"),
Text: `{"hello": "world"}`,
Uri: uri,
},
},
}, nil
},
)
}
Register resources and start server
func main() {
err := app.
NewBuilder().
// adding the resource to the app
WithResource(NewGreatResource).
// setting up server
WithName("my-mcp-server").
WithVersion("0.0.1").
WithTransport(stdio.NewTransport()).
// Configuring fx logging to only show errors
WithFxOptions(
fx.Provide(func() *zap.Logger {
cfg := zap.NewDevelopmentConfig()
cfg.Level.SetLevel(zap.ErrorLevel)
logger, _ := cfg.Build()
return logger
}),
fx.Option(fx.WithLogger(
func(logger *zap.Logger) fxevent.Logger {
return &fxevent.ZapLogger{Logger: logger}
},
)),
).Run()
if err != nil {
log.Fatal(err)
}
}
NewResourceProvider
In order to create new resource provider that would be returning resources dynamically, you shall use fxctx.NewResourceProvider
function. It would then take two functions - one in order to list resources and another to read them.
func NewGreatResourceProvider() fxctx.ResourceProvider {
return fxctx.NewResourceProvider(
// This is the callback that would be executed when the resources/list is requested:
func(_ context.Context) ([]mcp.Resource, error) {
return []mcp.Resource{
{
Name: "my-great-resource-one",
Description: Ptr("Does something great"),
Uri: "/resources/great/one",
},
}, nil
},
// This function reads the resource for a given uri to run when resources/read is requested:
func(_ context.Context, uri string) (*mcp.ReadResourceResult, error) {
// you would probably be doing something more complicated here
// like reading from a database or calling an external service
// based on what you have parsed from the uri
if uri == "/resources/great/one" {
return &mcp.ReadResourceResult{
Contents: []interface{}{
mcp.TextResourceContents{
MimeType: Ptr("application/json"),
Text: string(`{"great": "resource"}`),
Uri: uri,
},
},
}, nil
}
// this error would be wrapped in JSON-RPC error response
return nil, fmt.Errorf("resource not found")
},
)
}
Examples
Check out complete examples of MCP Servers with resources:
- k8s_contexts_resources - provides k8s contexts as resources
- hello_world_resource - provides one static very simple resource
- git_repository_resource - provides one static resource with information about git repository