feat: add network mode with combined HTTP and SSE endpoints
- Add new 'network' transport mode that serves both HTTP and SSE endpoints simultaneously - HTTP endpoint available on specified port (/mcp) - SSE endpoint available on port+1 (/sse) - Maintain backward compatibility with stdio, http, sse modes - Update documentation and examples for new network mode - Default to network mode for improved user experience 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
27
README.md
27
README.md
@@ -133,20 +133,16 @@ To configure the MCP server for Gitea, add the following to your MCP configurati
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
- **sse mode**
|
- **network mode** (provides both HTTP and SSE endpoints)
|
||||||
|
|
||||||
```json
|
Start the server in network mode:
|
||||||
{
|
```bash
|
||||||
"mcpServers": {
|
gitea-mcp -t network --host https://gitea.com --token <your-token> --port 8080
|
||||||
"gitea": {
|
|
||||||
"url": "http://localhost:8080/sse"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
```
|
||||||
|
|
||||||
- **http mode**
|
Then configure your MCP client with either endpoint:
|
||||||
|
|
||||||
|
**HTTP endpoint:**
|
||||||
```json
|
```json
|
||||||
{
|
{
|
||||||
"mcpServers": {
|
"mcpServers": {
|
||||||
@@ -157,6 +153,17 @@ To configure the MCP server for Gitea, add the following to your MCP configurati
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
**SSE endpoint:**
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"mcpServers": {
|
||||||
|
"gitea": {
|
||||||
|
"url": "http://localhost:8081/sse"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
**Default log path**: `$HOME/.gitea-mcp/gitea-mcp.log`
|
**Default log path**: `$HOME/.gitea-mcp/gitea-mcp.log`
|
||||||
|
|
||||||
> [!NOTE]
|
> [!NOTE]
|
||||||
|
|||||||
10
cmd/cmd.go
10
cmd/cmd.go
@@ -20,14 +20,14 @@ func init() {
|
|||||||
flag.StringVar(
|
flag.StringVar(
|
||||||
&flagPkg.Mode,
|
&flagPkg.Mode,
|
||||||
"t",
|
"t",
|
||||||
"stdio",
|
"network",
|
||||||
"Transport type (stdio, sse or http)",
|
"Transport type (stdio, http, sse, or network). Network mode starts both HTTP and SSE servers",
|
||||||
)
|
)
|
||||||
flag.StringVar(
|
flag.StringVar(
|
||||||
&flagPkg.Mode,
|
&flagPkg.Mode,
|
||||||
"transport",
|
"transport",
|
||||||
"stdio",
|
"network",
|
||||||
"Transport type (stdio, sse or http)",
|
"Transport type (stdio, http, sse, or network). Network mode starts both HTTP and SSE servers",
|
||||||
)
|
)
|
||||||
flag.StringVar(
|
flag.StringVar(
|
||||||
&host,
|
&host,
|
||||||
@@ -39,7 +39,7 @@ func init() {
|
|||||||
&port,
|
&port,
|
||||||
"port",
|
"port",
|
||||||
8080,
|
8080,
|
||||||
"see or http port",
|
"Network server port (used for both HTTP and SSE endpoints)",
|
||||||
)
|
)
|
||||||
flag.StringVar(
|
flag.StringVar(
|
||||||
&token,
|
&token,
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"mcpServers": {
|
"mcpServers": {
|
||||||
"gitea": {
|
"gitea-stdio": {
|
||||||
"command": "gitea-mcp",
|
"command": "gitea-mcp",
|
||||||
"args": {
|
"args": {
|
||||||
"-t": "stdio",
|
"-t": "stdio",
|
||||||
@@ -11,6 +11,12 @@
|
|||||||
"GITEA_HOST": "https://gitea.com",
|
"GITEA_HOST": "https://gitea.com",
|
||||||
"GITEA_ACCESS_TOKEN": "<your personal access token>"
|
"GITEA_ACCESS_TOKEN": "<your personal access token>"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"gitea-network-http": {
|
||||||
|
"url": "http://localhost:8080/mcp"
|
||||||
|
},
|
||||||
|
"gitea-network-sse": {
|
||||||
|
"url": "http://localhost:8081/sse"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -42,25 +42,53 @@ func RegisterTool(s *server.MCPServer) {
|
|||||||
func Run() error {
|
func Run() error {
|
||||||
mcpServer = newMCPServer(flag.Version)
|
mcpServer = newMCPServer(flag.Version)
|
||||||
RegisterTool(mcpServer)
|
RegisterTool(mcpServer)
|
||||||
|
|
||||||
switch flag.Mode {
|
switch flag.Mode {
|
||||||
case "stdio":
|
case "stdio":
|
||||||
if err := server.ServeStdio(mcpServer); err != nil {
|
if err := server.ServeStdio(mcpServer); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
case "sse":
|
|
||||||
sseServer := server.NewSSEServer(mcpServer)
|
|
||||||
log.Infof("Gitea MCP SSE server listening on :%d", flag.Port)
|
|
||||||
if err := sseServer.Start(fmt.Sprintf(":%d", flag.Port)); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
case "http":
|
case "http":
|
||||||
httpServer := server.NewStreamableHTTPServer(mcpServer)
|
httpServer := server.NewStreamableHTTPServer(mcpServer)
|
||||||
log.Infof("Gitea MCP HTTP server listening on :%d", flag.Port)
|
log.Infof("Gitea MCP HTTP server listening on :%d", flag.Port)
|
||||||
if err := httpServer.Start(fmt.Sprintf(":%d", flag.Port)); err != nil {
|
if err := httpServer.Start(fmt.Sprintf(":%d", flag.Port)); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
case "sse":
|
||||||
|
sseServer := server.NewSSEServer(mcpServer)
|
||||||
|
log.Infof("Gitea MCP SSE server listening on :%d", flag.Port)
|
||||||
|
if err := sseServer.Start(fmt.Sprintf(":%d", flag.Port)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
case "network":
|
||||||
|
// Network mode: start both HTTP and SSE servers concurrently
|
||||||
|
log.Infof("Starting Gitea MCP server in network mode")
|
||||||
|
|
||||||
|
errChan := make(chan error, 2)
|
||||||
|
|
||||||
|
// Start HTTP server on port
|
||||||
|
go func() {
|
||||||
|
httpServer := server.NewStreamableHTTPServer(mcpServer)
|
||||||
|
log.Infof("Gitea MCP HTTP server listening on :%d", flag.Port)
|
||||||
|
if err := httpServer.Start(fmt.Sprintf(":%d", flag.Port)); err != nil {
|
||||||
|
errChan <- fmt.Errorf("HTTP server error: %w", err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
// Start SSE server on port+1
|
||||||
|
go func() {
|
||||||
|
sseServer := server.NewSSEServer(mcpServer)
|
||||||
|
ssePort := flag.Port + 1
|
||||||
|
log.Infof("Gitea MCP SSE server listening on :%d", ssePort)
|
||||||
|
if err := sseServer.Start(fmt.Sprintf(":%d", ssePort)); err != nil {
|
||||||
|
errChan <- fmt.Errorf("SSE server error: %w", err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
// Wait for first error
|
||||||
|
return <-errChan
|
||||||
default:
|
default:
|
||||||
return fmt.Errorf("invalid transport type: %s. Must be 'stdio', 'sse' or 'http'", flag.Mode)
|
return fmt.Errorf("invalid transport type: %s. Must be 'stdio', 'http', 'sse' or 'network'", flag.Mode)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user