A clean Go library for building Model Context Protocol (MCP) servers on top of mcp-go.
- Three ways to register tools: inline struct, function helper, or JSON file on disk
- Built-in argument extraction helpers (
ArgString,ArgFloat,ArgBool, …) - Result builders (
ResultText,ResultJSON,ResultError, …) - Both SSE (HTTP) and Stdio transports — pick at runtime
go get github.com/raykavin/gomcppackage main
import (
"context"
mcpserver "github.com/raykavin/gomcp/server"
)
func main() {
srv := mcpserver.NewMCPServer("my-agent", "1.0.0")
// Register a tool
srv.AddToolFunc(
"echo",
"Returns the input message",
mcpserver.JSONSchema{
Type: "object",
Required: []string{"message"},
Properties: map[string]mcpserver.JSONSchemaProperty{
"message": {Type: "string", Description: "Message to echo"},
},
},
func(ctx context.Context, req mcpproto.CallToolRequest) (*mcpproto.CallToolResult, error) {
msg, errResult := mcpserver.ArgStringRequired(req, "message")
if errResult != nil {
return errResult, nil
}
return mcpserver.ResultText("echo: " + msg), nil
},
)
// Start SSE transport
srv.Start(ctx, ":8080")
}Best for tools defined entirely in Go code.
err := srv.AddToolFunc(
"search", // name
"Search the web", // description
mcpserver.JSONSchema{...}, // schema
myHandler,
)Useful when building definitions programmatically or receiving them from external sources.
def := mcpserver.ToolDefinition{
Name: "add",
Description: "Adds two numbers",
Schema: mcpserver.JSONSchema{...},
}
err := srv.AddTool(def, myHandler)Keeps tool metadata outside Go code. Ideal for config-driven agents or tools shared across projects.
err := srv.AddToolFromFile("tools/search.json", myHandler)JSON file format:
{
"name": "search",
"description": "Search the web and return relevant results",
"schema": {
"type": "object",
"properties": {
"query": { "type": "string", "description": "The search query" },
"max_results": { "type": "number", "description": "Max results" }
},
"required": ["query"]
}
}For tools fetched from a database, remote config, etc.
data := []byte(`{...}`)
err := srv.AddToolFromJSON(data, myHandler)Inside a handler, use the helpers to safely extract typed arguments:
// String (returns error result if missing)
msg, errResult := mcpserver.ArgStringRequired(req, "message")
if errResult != nil {
return errResult, nil
}
// Optional string
if q, ok := mcpserver.ArgString(req, "query"); ok { ... }
// Number
if n, ok := mcpserver.ArgFloat(req, "count"); ok { ... }
// Bool
if verbose, ok := mcpserver.ArgBool(req, "verbose"); ok { ... }mcpserver.ResultText("plain string response")
mcpserver.ResultJSON(map[string]any{"key": "value"})
mcpserver.ResultError("something broke", err)
mcpserver.ResultErrorMsg("missing argument: query")ctx, stop := signal.NotifyContext(context.Background(), os.Interrupt)
defer stop()
if err := srv.Start(ctx, ":8080"); err != nil {
log.Fatal(err)
}Clients connect via http://localhost:8080/sse.
if err := srv.StartStdio(ctx); err != nil {
log.Fatal(err)
}srv.Shutdown(ctx)github.com/raykavin/gomcp/
├── server.go # MCPServer, Start, StartStdio, Shutdown
├── tool.go # ToolDefinition, JSONSchema, ParseToolDefinition
├── helpers.go # ResultText/JSON/Error, ArgString/Float/Bool helpers
├── go.mod
└── example/
├── main.go
└── tools/
└── search.json
go test ./serverContributions to gomcp are welcome! Here are some ways you can help improve the project:
- Report bugs and suggest features by opening issues on GitHub
- Submit pull requests with bug fixes or new features
- Improve documentation to help other users and developers
- Share your custom strategies with the community
gomcp is distributed under the MIT License.
For complete license terms and conditions, see the LICENSE file in the repository.
For support, collaboration, or questions about gomcp:
Email: raykavin.meireles@gmail.com
GitHub: @raykavin