Compare commits
2 Commits
network-mo
...
combined-s
| Author | SHA1 | Date | |
|---|---|---|---|
| 08045538bf | |||
|
|
e3f7405cd8 |
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
|
||||
{
|
||||
"mcpServers": {
|
||||
"gitea": {
|
||||
"url": "http://localhost:8080/sse"
|
||||
}
|
||||
}
|
||||
}
|
||||
Start the server in network mode:
|
||||
```bash
|
||||
gitea-mcp -t network --host https://gitea.com --token <your-token> --port 8080
|
||||
```
|
||||
|
||||
- **http mode**
|
||||
Then configure your MCP client with either endpoint:
|
||||
|
||||
**HTTP endpoint:**
|
||||
```json
|
||||
{
|
||||
"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`
|
||||
|
||||
> [!NOTE]
|
||||
|
||||
10
cmd/cmd.go
10
cmd/cmd.go
@@ -20,14 +20,14 @@ func init() {
|
||||
flag.StringVar(
|
||||
&flagPkg.Mode,
|
||||
"t",
|
||||
"stdio",
|
||||
"Transport type (stdio, sse, http, or network)",
|
||||
"network",
|
||||
"Transport type (stdio, http, sse, or network). Network mode starts both HTTP and SSE servers",
|
||||
)
|
||||
flag.StringVar(
|
||||
&flagPkg.Mode,
|
||||
"transport",
|
||||
"stdio",
|
||||
"Transport type (stdio, sse, http, or network)",
|
||||
"network",
|
||||
"Transport type (stdio, http, sse, or network). Network mode starts both HTTP and SSE servers",
|
||||
)
|
||||
flag.StringVar(
|
||||
&host,
|
||||
@@ -39,7 +39,7 @@ func init() {
|
||||
&port,
|
||||
"port",
|
||||
8080,
|
||||
"see or http port",
|
||||
"Network server port (used for both HTTP and SSE endpoints)",
|
||||
)
|
||||
flag.StringVar(
|
||||
&token,
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"mcpServers": {
|
||||
"gitea": {
|
||||
"gitea-stdio": {
|
||||
"command": "gitea-mcp",
|
||||
"args": {
|
||||
"-t": "stdio",
|
||||
@@ -11,6 +11,12 @@
|
||||
"GITEA_HOST": "https://gitea.com",
|
||||
"GITEA_ACCESS_TOKEN": "<your personal access token>"
|
||||
}
|
||||
},
|
||||
"gitea-network-http": {
|
||||
"url": "http://localhost:8080/mcp"
|
||||
},
|
||||
"gitea-network-sse": {
|
||||
"url": "http://localhost:8081/sse"
|
||||
}
|
||||
}
|
||||
}
|
||||
Binary file not shown.
BIN
gitea-mcp-v0.3.0
BIN
gitea-mcp-v0.3.0
Binary file not shown.
4
go.mod
4
go.mod
@@ -1,8 +1,6 @@
|
||||
module gitea.com/gitea/gitea-mcp
|
||||
|
||||
go 1.23.0
|
||||
|
||||
toolchain go1.23.11
|
||||
go 1.24.0
|
||||
|
||||
require (
|
||||
code.gitea.io/sdk/gitea v0.21.0
|
||||
|
||||
@@ -2,7 +2,6 @@ package operation
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"net/http"
|
||||
|
||||
"gitea.com/gitea/gitea-mcp/operation/issue"
|
||||
"gitea.com/gitea/gitea-mcp/operation/pull"
|
||||
@@ -43,56 +42,53 @@ func RegisterTool(s *server.MCPServer) {
|
||||
func Run() error {
|
||||
mcpServer = newMCPServer(flag.Version)
|
||||
RegisterTool(mcpServer)
|
||||
addr := fmt.Sprintf("127.0.0.1:%d", flag.Port)
|
||||
|
||||
switch flag.Mode {
|
||||
case "stdio":
|
||||
if err := server.ServeStdio(mcpServer); err != nil {
|
||||
return err
|
||||
}
|
||||
case "sse":
|
||||
sseServer := server.NewSSEServer(mcpServer)
|
||||
log.Infof("Gitea MCP Server running:")
|
||||
log.Infof(" sse: http://%s/", addr)
|
||||
if err := sseServer.Start(fmt.Sprintf(":%d", flag.Port)); err != nil {
|
||||
return err
|
||||
}
|
||||
case "http":
|
||||
httpServer := server.NewStreamableHTTPServer(mcpServer)
|
||||
log.Infof("Gitea MCP Server running:")
|
||||
log.Infof(" http: http://%s/", addr)
|
||||
log.Infof("Gitea MCP HTTP server listening on :%d", flag.Port)
|
||||
if err := httpServer.Start(fmt.Sprintf(":%d", flag.Port)); err != nil {
|
||||
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: serve both HTTP and SSE on same port with different URLs
|
||||
log.Infof("Network mode: Creating streamable HTTP server...")
|
||||
streamableServer := server.NewStreamableHTTPServer(mcpServer)
|
||||
log.Infof("Network mode: Created streamable HTTP server")
|
||||
// Network mode: start both HTTP and SSE servers concurrently
|
||||
log.Infof("Starting Gitea MCP server in network mode")
|
||||
|
||||
log.Infof("Network mode: Creating SSE server...")
|
||||
sseServer := server.NewSSEServer(mcpServer,
|
||||
server.WithSSEEndpoint("/sse"),
|
||||
server.WithMessageEndpoint("/message"),
|
||||
)
|
||||
log.Infof("Network mode: Created SSE server")
|
||||
errChan := make(chan error, 2)
|
||||
|
||||
// Create custom HTTP mux
|
||||
log.Infof("Network mode: Creating HTTP mux...")
|
||||
mux := http.NewServeMux()
|
||||
mux.Handle("/mcp", streamableServer)
|
||||
mux.Handle("/", sseServer)
|
||||
log.Infof("Network mode: Configured HTTP routes")
|
||||
// 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 single HTTP server
|
||||
log.Infof("Gitea MCP Server running in network mode:")
|
||||
log.Infof(" http: http://%s/mcp", addr)
|
||||
log.Infof(" sse: http://%s/sse", addr)
|
||||
err := http.ListenAndServe(fmt.Sprintf(":%d", flag.Port), mux)
|
||||
log.Errorf("Network mode: ListenAndServe returned with error: %v", err)
|
||||
return 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:
|
||||
return fmt.Errorf("invalid transport type: %s. Must be 'stdio', 'sse', 'http' or 'network'", flag.Mode)
|
||||
return fmt.Errorf("invalid transport type: %s. Must be 'stdio', 'http', 'sse' or 'network'", flag.Mode)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
@@ -45,7 +45,7 @@ func Default() *zap.Logger {
|
||||
MaxAge: 30,
|
||||
}))
|
||||
|
||||
if flag.Mode == "http" || flag.Mode == "sse" || flag.Mode == "network" {
|
||||
if flag.Mode == "http" || flag.Mode == "sse" {
|
||||
wss = append(wss, zapcore.AddSync(os.Stdout))
|
||||
}
|
||||
|
||||
|
||||
24
releases/README.md
Normal file
24
releases/README.md
Normal file
@@ -0,0 +1,24 @@
|
||||
# Releases
|
||||
|
||||
## v0.5.0-combined - Network Mode
|
||||
|
||||
This directory contains binary releases for the combined server implementation.
|
||||
|
||||
### Features
|
||||
- Network mode with both HTTP and SSE endpoints available simultaneously
|
||||
- HTTP endpoint on specified port (/mcp)
|
||||
- SSE endpoint on port+1 (/sse)
|
||||
- Backward compatibility with stdio, http, sse modes
|
||||
|
||||
### Usage
|
||||
```bash
|
||||
# Download and make executable
|
||||
chmod +x gitea-mcp-combined
|
||||
|
||||
# Start network mode
|
||||
./gitea-mcp-combined -t network --host https://gitea.ext.ben.io --token <token> --port 8080
|
||||
```
|
||||
|
||||
### Endpoints
|
||||
- HTTP: http://localhost:8080/mcp
|
||||
- SSE: http://localhost:8081/sse
|
||||
Reference in New Issue
Block a user