We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/last9/last9-mcp-server'
If you have feedback or need assistance with the MCP directory API, please join our Discord server
package traces
import (
"encoding/json"
"errors"
"fmt"
"io"
"last9-mcp/internal/models"
"last9-mcp/internal/utils"
"net/http"
"net/url"
"strconv"
"strings"
"github.com/acrmp/mcp"
)
// NewGetServiceGraphHandler creates a handler for getting service dependencies
func NewGetServiceGraphHandler(client *http.Client, cfg models.Config) func(mcp.CallToolRequestParams) (mcp.CallToolResult, error) {
return func(params mcp.CallToolRequestParams) (mcp.CallToolResult, error) {
spanName, ok := params.Arguments["span_name"].(string)
if !ok {
return mcp.CallToolResult{}, errors.New("span_name is required")
}
if spanName == "" {
return mcp.CallToolResult{}, errors.New("span_name cannot be empty")
}
lookbackMinutes := 60
if l, ok := params.Arguments["lookback_minutes"].(float64); ok {
lookbackMinutes = int(l)
}
// Get time range using the common utility
startTime, _, err := utils.GetTimeRange(params.Arguments, lookbackMinutes)
if err != nil {
return mcp.CallToolResult{}, err
}
// Build request URL
u, err := url.Parse(cfg.BaseURL + "/telemetry/api/v1/service_graph")
if err != nil {
return mcp.CallToolResult{}, fmt.Errorf("failed to parse URL: %w", err)
}
q := u.Query()
q.Set("start", strconv.FormatInt(startTime.Unix(), 10))
q.Set("span_name", spanName)
q.Set("lookback_minutes", strconv.Itoa(lookbackMinutes))
u.RawQuery = q.Encode()
// Create request
req, err := http.NewRequest("GET", u.String(), nil)
if err != nil {
return mcp.CallToolResult{}, fmt.Errorf("failed to create request: %w", err)
}
// Check if the auth token already has the "Basic" prefix
if !strings.HasPrefix(cfg.AuthToken, "Basic ") {
cfg.AuthToken = "Basic " + cfg.AuthToken
}
req.Header.Set("Authorization", cfg.AuthToken)
// Execute request
resp, err := client.Do(req)
if err != nil {
return mcp.CallToolResult{}, fmt.Errorf("request failed: %w", err)
}
defer resp.Body.Close()
if resp.StatusCode != http.StatusOK {
body, _ := io.ReadAll(resp.Body)
return mcp.CallToolResult{}, fmt.Errorf("service graph API request failed with status %d: %s", resp.StatusCode, string(body))
}
var result interface{}
if err := json.NewDecoder(resp.Body).Decode(&result); err != nil {
return mcp.CallToolResult{}, fmt.Errorf("failed to decode response: %w", err)
}
jsonData, err := json.Marshal(result)
if err != nil {
return mcp.CallToolResult{}, fmt.Errorf("failed to marshal response: %w", err)
}
return mcp.CallToolResult{
Content: []any{
mcp.TextContent{
Text: string(jsonData),
Type: "text",
},
},
}, nil
}
}