Tools
Through tools, LLMs can interact with external systems, perform computations, and take actions in the real world.
MCP tools is a concept of MCP servers. Server should list tools when requested with method tools/list
and call when requested with method tools/call
.
Foxy Contexts allows easy way to define a tool and register it within fx DI container.
NewTool
In order to create new tool you shall use fxctx.NewTool
function. It accepts tool name, description and function that would be called when tool is called.
func NewGreatTool() fxctx.Tool {
return fxctx.NewTool(
// This information about the tool would be used when it is listed:
&mcp.Tool{
Name: "my-great-tool",
Description: Ptr("The great tool"),
InputSchema: mcp.ToolInputSchema{ // here we tell client what we expect as input
Type: "object",
Properties: map[string]map[string]interface{}{},
Required: []string{},
},
},
// This is the callback that would be executed when the tool is called:
func(_ context.Context, args map[string]interface{}) *mcp.CallToolResult {
// here we can do anything we want
return &mcp.CallToolResult{
Content: []interface{}{
mcp.TextContent{
Type: "text",
Text: fmt.Sprintf("Sup"),
},
},
}
},
)
}
Register tool and start server
func main() {
app.
NewBuilder().
// adding the tool to the app
WithTool(NewGreatTool).
// setting up server
WithName("great-tool-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()
}
Using toolinput package
In order to define input schema for your tool, you can use toolinput
package. It allows you to define input schema and validate arriving input.
Here is an example of creating schema, giving it to the tool and validating input:
schema := toolinput.NewToolInputSchema(
toolinput.WithString("kubeconfig", "Path to kubeconfig file"),
)
// Here we define a tool that lists k8s contexts using client-go
return fxctx.NewTool(
// This information about the tool would be used when it is listed:
&mcp.Tool{
Name: "list-k8s-contexts",
Description: Ptr("List Kubernetes contexts from configuration files such as kubeconfig"),
InputSchema: schema.GetMcpToolInputSchema(),
},
// This is the callback that would be executed when the tool is called:
func(_ context.Context, args map[string]interface{}) *mcp.CallToolResult {
input, err := schema.Validate(args)
if err != nil {
log.Printf("failed to validate input: %v", err)
return &mcp.CallToolResult{
IsError: Ptr(true),
Content: []interface{}{
mcp.TextContent{
Type: "text",
Text: fmt.Sprintf("failed to validate input: %v", err),
},
},
}
}
path := input.StringOr("kubeconfig", "")
if path != "" {
os.Setenv("KUBECONFIG", path)
}
Examples
Check out complete examples of MCP Servers with tools: