https://join39.org/api/mcp-servers
This API provides a simple interface to manage MCP servers in a public registry. It supports full CRUD operations without requiring authentication or API keys, making it easy for developers to integrate MCP servers into their applications.
- ✅ No Authentication Required - Public API for easy integration
- ✅ Full CRUD Operations - Create, Read, Update, Delete
- ✅ Input Validation - Ensures data quality and security
- ✅ Duplicate Prevention - Prevents duplicate endpoint submissions
- ✅ JSON API - Standard REST API with JSON responses
- ✅ Error Handling - Clear error messages and HTTP status codes
GET /api/mcp-servers
Retrieves all publicly available MCP servers.
Request:
curl https://join39.org/api/mcp-serversResponse:
{
"success": true,
"servers": [
{
"id": "687472d4476f071042eb5dfd",
"name": "Weather Agent",
"endpoint": "https://api.weather.com/mcp",
"description": "Provides real-time weather data",
"created_at": "2025-07-14T01:15:23.000Z",
"registry_provider": "nanda"
},
{
"id": "687472e8476f071042eb5dfe",
"name": "File Manager",
"endpoint": "https://files.example.com/mcp",
"description": "File operations and management",
"created_at": "2025-07-14T01:16:45.000Z",
"registry_provider": "nanda"
}
]
}POST /api/mcp-servers
Creates a new MCP server entry.
Request:
curl -X POST https://join39.org/api/mcp-servers \
-H "Content-Type: application/json" \
-d '{
"name": "My Awesome Agent",
"endpoint": "https://my-server.com/mcp",
"description": "Does amazing things with data"
}'Response:
{
"success": true,
"message": "MCP server submitted successfully",
"server_id": "687472d4476f071042eb5dfd",
"name": "My Awesome Agent",
"endpoint": "https://my-server.com/mcp"
}PUT /api/mcp-servers
Updates an existing MCP server.
Request:
curl -X PUT https://join39.org/api/mcp-servers \
-H "Content-Type: application/json" \
-d '{
"id": "687472d4476f071042eb5dfd",
"name": "My Updated Agent",
"endpoint": "https://my-server.com/v2/mcp",
"description": "Updated with new features"
}'Response:
{
"success": true,
"message": "MCP server updated successfully",
"server_id": "687472d4476f071042eb5dfd",
"name": "My Updated Agent",
"endpoint": "https://my-server.com/v2/mcp"
}DELETE /api/mcp-servers?id={server_id}
Removes an MCP server from the registry.
Request:
curl -X DELETE "https://join39.org/api/mcp-servers?id=687472d4476f071042eb5dfd"Response:
{
"success": true,
"message": "MCP server deleted successfully",
"server_id": "687472d4476f071042eb5dfd"
}| Field | Type | Required | Description |
|---|---|---|---|
id |
string | ✅ (for updates/deletes) | Unique server identifier |
name |
string | ✅ | Server display name (1-100 characters) |
endpoint |
string | ✅ | HTTP/HTTPS URL to the MCP server |
description |
string | ❌ | Optional description of the server |
created_at |
string | ❌ (auto-generated) | ISO timestamp of creation |
registry_provider |
string | ❌ (auto-set) | Registry provider (always "nanda") |
Create/Update Request:
{
"name": "string",
"endpoint": "string",
"description": "string (optional)"
}Update Request (additional field):
{
"id": "string",
"name": "string",
"endpoint": "string",
"description": "string (optional)"
}- Required: Yes
- Length: 1-100 characters
- Pattern: Letters, numbers, spaces, hyphens, underscores only
- Regex:
/^[a-zA-Z0-9\s\-_]+$/
- Required: Yes
- Format: Valid HTTP or HTTPS URL
- Examples:
- ✅
https://my-server.com/mcp - ✅
http://localhost:3000/mcp - ❌
my-server.com/mcp(missing protocol) - ❌
ftp://server.com/mcp(invalid protocol)
- ✅
- Required: No
- Type: String
- Length: No specific limit (reasonable length recommended)
| Code | Status | Description |
|---|---|---|
| 200 | OK | Request successful |
| 201 | Created | Server created successfully |
| 400 | Bad Request | Invalid input data |
| 404 | Not Found | Server not found |
| 409 | Conflict | Duplicate endpoint exists |
| 500 | Internal Server Error | Server error |
{
"error": "Error message",
"details": ["Detailed error 1", "Detailed error 2"]
}Validation Error (400):
{
"error": "Validation failed",
"details": [
"Name must be between 1 and 100 characters",
"Endpoint must be a valid URL"
]
}Duplicate Endpoint (409):
{
"error": "A server with this endpoint already exists"
}Server Not Found (404):
{
"error": "Server not found"
}Missing Required Fields (400):
{
"error": "Server ID is required"
}# 1. Create a new server
RESPONSE=$(curl -X POST https://join39.org/api/mcp-servers \
-H "Content-Type: application/json" \
-d '{
"name": "Test Agent",
"endpoint": "https://test.com/mcp",
"description": "Testing the API"
}')
echo $RESPONSE
# Output: {"success":true,"message":"MCP server submitted successfully","server_id":"687472d4476f071042eb5dfd","name":"Test Agent","endpoint":"https://test.com/mcp"}
# 2. Extract server ID (in a real script, you'd parse JSON)
SERVER_ID="687472d4476f071042eb5dfd"
# 3. Update the server
curl -X PUT https://join39.org/api/mcp-servers \
-H "Content-Type: application/json" \
-d '{
"id": "'$SERVER_ID'",
"name": "Updated Test Agent",
"endpoint": "https://test.com/v2/mcp",
"description": "Updated via API"
}'
# 4. List all servers to verify
curl https://join39.org/api/mcp-servers
# 5. Delete the server
curl -X DELETE "https://join39.org/api/mcp-servers?id=$SERVER_ID"# Try to create with invalid data
curl -X POST https://join39.org/api/mcp-servers \
-H "Content-Type: application/json" \
-d '{
"name": "",
"endpoint": "not-a-url"
}'
# Response:
# {
# "error": "Validation failed",
# "details": [
# "Name must be between 1 and 100 characters",
# "Endpoint must be a valid URL"
# ]
# }# First submission - succeeds
curl -X POST https://join39.org/api/mcp-servers \
-H "Content-Type: application/json" \
-d '{
"name": "First Agent",
"endpoint": "https://example.com/mcp"
}'
# Second submission with same endpoint - fails
curl -X POST https://join39.org/api/mcp-servers \
-H "Content-Type: application/json" \
-d '{
"name": "Second Agent",
"endpoint": "https://example.com/mcp"
}'
# Response: {"error":"A server with this endpoint already exists"}const API_BASE = 'https://join39.org/api/mcp-servers';
async function createServer(name, endpoint, description) {
const response = await fetch(API_BASE, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ name, endpoint, description })
});
return response.json();
}
async function listServers() {
const response = await fetch(API_BASE);
return response.json();
}import requests
API_BASE = 'https://join39.org/api/mcp-servers'
def create_server(name, endpoint, description=None):
data = {'name': name, 'endpoint': endpoint}
if description:
data['description'] = description
response = requests.post(API_BASE, json=data)
return response.json()
def list_servers():
response = requests.get(API_BASE)
return response.json()- Live API: https://join39.org/api/mcp-servers
- Web Interface: https://join39.org/servers
- Issues: Report API issues or request features
Last Updated: July 2025
API Version: 1.0
License: MIT