(Clicca sull'immagine sopra per vedere il video di questa lezione)
Questa lezione si concentra su come interagire con la comunità MCP, contribuire all'ecosistema MCP e seguire le migliori pratiche per lo sviluppo collaborativo. Comprendere come partecipare ai progetti open-source MCP è essenziale per chi vuole plasmare il futuro di questa tecnologia.
Al termine di questa lezione, sarai in grado di:
- Comprendere la struttura della comunità e dell'ecosistema MCP
- Partecipare efficacemente ai forum e alle discussioni della comunità MCP
- Contribuire ai repository open-source MCP
- Creare e condividere strumenti e server MCP personalizzati
- Seguire le migliori pratiche per lo sviluppo e la collaborazione MCP
- Scoprire risorse e framework comunitari per lo sviluppo MCP
L'ecosistema MCP è composto da vari componenti e partecipanti che lavorano insieme per far progredire il protocollo.
- Manutentori del Protocollo Core: L'organizzazione ufficiale Model Context Protocol GitHub mantiene le specifiche core MCP e le implementazioni di riferimento
- Sviluppatori di Strumenti: Individui e team che creano strumenti e server MCP
- Provider di Integrazione: Aziende che integrano MCP nei loro prodotti e servizi
- Utenti Finali: Sviluppatori e organizzazioni che usano MCP nelle loro applicazioni
- Contributori: Membri della comunità che contribuiscono con codice, documentazione o altre risorse
- Organizzazione GitHub MCP
- Documentazione MCP
- Specifiche MCP
- Discussioni GitHub
- Repository Esempi & Server MCP
- Client MCP - Elenco dei client che supportano integrazioni MCP
- Server MCP Comunitari - Elenco in crescita di server MCP sviluppati dalla comunità
- Awesome MCP Servers - Elenco curato di server MCP
- PulseMCP - Hub comunitario e newsletter per scoprire risorse MCP
- Server Discord - Connettiti con sviluppatori MCP
- Implementazioni SDK specifiche per linguaggio
- Articoli di blog e tutorial
L'ecosistema MCP accoglie diversi tipi di contributi:
-
Contributi di Codice:
- miglioramenti al protocollo core
- correzione di bug
- implementazioni di strumenti e server
- librerie client/server in diversi linguaggi
-
Documentazione:
- migliorare la documentazione esistente
- creare tutorial e guide
- tradurre la documentazione
- creare esempi e applicazioni di esempio
-
Supporto alla Comunità:
- rispondere a domande su forum e discussioni
- testare e segnalare problemi
- organizzare eventi comunitari
- fare mentoring a nuovi contributori
Per contribuire al protocollo core MCP o alle implementazioni ufficiali, segui questi principi dalle linee guida ufficiali per i contributi:
-
Semplicità e Minimalismo: La specifica MCP mantiene un alto standard per l'aggiunta di nuovi concetti. È più facile aggiungere cose a una specifica che rimuoverle.
-
Approccio Concreto: Le modifiche alle specifiche devono basarsi su sfide di implementazione specifiche, non su idee speculative.
-
Fasi di una Proposta:
- Definisci: esplora il problema, verifica che altri utenti MCP affrontino una sfida simile
- Prototipa: costruisci una soluzione esempio e dimostra la sua applicazione pratica
- Scrivi: sulla base del prototipo, scrivi una proposta di specifica
# Fai il fork del repository
git clone https://github.com/YOUR-USERNAME/modelcontextprotocol.git
cd modelcontextprotocol
# Installa le dipendenze
npm install
# Per modifiche allo schema, convalida e genera schema.json:
npm run check:schema:ts
npm run generate:schema
# Per modifiche alla documentazione
npm run check:docs
npm run format
# Visualizza in anteprima la documentazione localmente (opzionale):
npm run serve:docs// Codice originale con bug nel typescript-sdk
export function validateResource(resource: unknown): resource is MCPResource {
if (!resource || typeof resource !== 'object') {
return false;
}
// Bug: Mancata convalida delle proprietà
// Implementazione attuale:
const hasName = 'name' in resource;
const hasSchema = 'schema' in resource;
return hasName && hasSchema;
}
// Implementazione corretta in un contributo
export function validateResource(resource: unknown): resource is MCPResource {
if (!resource || typeof resource !== 'object') {
return false;
}
// Convalida migliorata
const hasName = 'name' in resource && typeof (resource as MCPResource).name === 'string';
const hasSchema = 'schema' in resource && typeof (resource as MCPResource).schema === 'object';
const hasDescription = !('description' in resource) || typeof (resource as MCPResource).description === 'string';
return hasName && hasSchema && hasDescription;
}# Esempio di contributo: uno strumento di elaborazione dati CSV per la libreria standard MCP
from mcp_tools import Tool, ToolRequest, ToolResponse, ToolExecutionException
import pandas as pd
import io
import json
from typing import Dict, Any, List, Optional
class CsvProcessingTool(Tool):
"""
Tool for processing and analyzing CSV data.
This tool allows models to extract information from CSV files,
run basic analysis, and convert data between formats.
"""
def get_name(self):
return "csvProcessor"
def get_description(self):
return "Processes and analyzes CSV data"
def get_schema(self):
return {
"type": "object",
"properties": {
"csvData": {
"type": "string",
"description": "CSV data as a string"
},
"csvUrl": {
"type": "string",
"description": "URL to a CSV file (alternative to csvData)"
},
"operation": {
"type": "string",
"enum": ["summary", "filter", "transform", "convert"],
"description": "Operation to perform on the CSV data"
},
"filterColumn": {
"type": "string",
"description": "Column to filter by (for filter operation)"
},
"filterValue": {
"type": "string",
"description": "Value to filter for (for filter operation)"
},
"outputFormat": {
"type": "string",
"enum": ["json", "csv", "markdown"],
"default": "json",
"description": "Output format for the processed data"
}
},
"oneOf": [
{"required": ["csvData", "operation"]},
{"required": ["csvUrl", "operation"]}
]
}
async def execute_async(self, request: ToolRequest) -> ToolResponse:
try:
# Estrai parametri
operation = request.parameters.get("operation")
output_format = request.parameters.get("outputFormat", "json")
# Ottieni dati CSV da dati diretti o URL
df = await self._get_dataframe(request)
# Elabora in base all'operazione richiesta
result = {}
if operation == "summary":
result = self._generate_summary(df)
elif operation == "filter":
column = request.parameters.get("filterColumn")
value = request.parameters.get("filterValue")
if not column:
raise ToolExecutionException("filterColumn is required for filter operation")
result = self._filter_data(df, column, value)
elif operation == "transform":
result = self._transform_data(df, request.parameters)
elif operation == "convert":
result = self._convert_format(df, output_format)
else:
raise ToolExecutionException(f"Unknown operation: {operation}")
return ToolResponse(result=result)
except Exception as e:
raise ToolExecutionException(f"CSV processing failed: {str(e)}")
async def _get_dataframe(self, request: ToolRequest) -> pd.DataFrame:
"""Gets a pandas DataFrame from either CSV data or URL"""
if "csvData" in request.parameters:
csv_data = request.parameters.get("csvData")
return pd.read_csv(io.StringIO(csv_data))
elif "csvUrl" in request.parameters:
csv_url = request.parameters.get("csvUrl")
return pd.read_csv(csv_url)
else:
raise ToolExecutionException("Either csvData or csvUrl must be provided")
def _generate_summary(self, df: pd.DataFrame) -> Dict[str, Any]:
"""Generates a summary of the CSV data"""
return {
"columns": df.columns.tolist(),
"rowCount": len(df),
"columnCount": len(df.columns),
"numericColumns": df.select_dtypes(include=['number']).columns.tolist(),
"categoricalColumns": df.select_dtypes(include=['object']).columns.tolist(),
"sampleRows": json.loads(df.head(5).to_json(orient="records")),
"statistics": json.loads(df.describe().to_json())
}
def _filter_data(self, df: pd.DataFrame, column: str, value: str) -> Dict[str, Any]:
"""Filters the DataFrame by a column value"""
if column not in df.columns:
raise ToolExecutionException(f"Column '{column}' not found")
filtered_df = df[df[column].astype(str).str.contains(value)]
return {
"originalRowCount": len(df),
"filteredRowCount": len(filtered_df),
"data": json.loads(filtered_df.to_json(orient="records"))
}
def _transform_data(self, df: pd.DataFrame, params: Dict[str, Any]) -> Dict[str, Any]:
"""Transforms the data based on parameters"""
# L'implementazione includerebbe varie trasformazioni
return {
"status": "success",
"message": "Transformation applied"
}
def _convert_format(self, df: pd.DataFrame, format: str) -> Dict[str, Any]:
"""Converts the DataFrame to different formats"""
if format == "json":
return {
"data": json.loads(df.to_json(orient="records")),
"format": "json"
}
elif format == "csv":
return {
"data": df.to_csv(index=False),
"format": "csv"
}
elif format == "markdown":
return {
"data": df.to_markdown(),
"format": "markdown"
}
else:
raise ToolExecutionException(f"Unsupported output format: {format}")Per fare un contributo di successo ai progetti MCP:
- Inizia in piccolo: Inizia con documentazione, correzioni di bug o piccoli miglioramenti
- Segui la Guida di Stile: Rispetta lo stile di codifica e le convenzioni del progetto
- Scrivi Test: Includi test unitari per i tuoi contributi di codice
- Documenta il tuo Lavoro: Aggiungi documentazione chiara per nuove funzionalità o modifiche
- Invia PR Mirate: Mantieni le pull request focalizzate su un singolo problema o funzionalità
- Interagisci con il Feedback: Rispondi tempestivamente al feedback sui tuoi contributi
# Clona il repository
git clone https://github.com/modelcontextprotocol/typescript-sdk.git
cd typescript-sdk
# Crea un nuovo branch per il tuo contributo
git checkout -b feature/my-contribution
# Apporta le tue modifiche
# ...
# Esegui i test per assicurarti che le tue modifiche non compromettano le funzionalità esistenti
npm test
# Commit delle tue modifiche con un messaggio descrittivo
git commit -am "Fix validation in resource handler"
# Pusha il tuo branch sul tuo fork
git push origin feature/my-contribution
# Crea una pull request dal tuo branch al repository principale
# Poi interagisci con il feedback e iterare sulla tua PR secondo necessitàUno dei modi più preziosi per contribuire all'ecosistema MCP è creare e condividere server MCP personalizzati. La comunità ha già sviluppato centinaia di server per vari servizi e use case.
Sono disponibili diversi framework per semplificare lo sviluppo di server MCP:
-
SDK Ufficiali (allineati con la Specifiche MCP 2025-11-25):
-
Framework Comunitari:
- MCP-Framework - Costruisci server MCP con eleganza e velocità in TypeScript
- MCP Declarative Java SDK - Server MCP basati su annotazioni con Java
- Quarkus MCP Server SDK - Framework Java per server MCP
- Next.js MCP Server Template - Progetto starter Next.js per server MCP
// Create a new .NET library project
// dotnet new classlib -n McpFinanceTools
using Microsoft.Mcp.Tools;
using System.Threading.Tasks;
using System.Net.Http;
using System.Text.Json;
namespace McpFinanceTools
{
// Stock quote tool
public class StockQuoteTool : IMcpTool
{
private readonly HttpClient _httpClient;
public StockQuoteTool(HttpClient httpClient = null)
{
_httpClient = httpClient ?? new HttpClient();
}
public string Name => "stockQuote";
public string Description => "Gets current stock quotes for specified symbols";
public object GetSchema()
{
return new {
type = "object",
properties = new {
symbol = new {
type = "string",
description = "Stock symbol (e.g., MSFT, AAPL)"
},
includeHistory = new {
type = "boolean",
description = "Whether to include historical data",
default = false
}
},
required = new[] { "symbol" }
};
}
public async Task<ToolResponse> ExecuteAsync(ToolRequest request)
{
// Extract parameters
string symbol = request.Parameters.GetProperty("symbol").GetString();
bool includeHistory = false;
if (request.Parameters.TryGetProperty("includeHistory", out var historyProp))
{
includeHistory = historyProp.GetBoolean();
}
// Call external API (example)
var quoteResult = await GetStockQuoteAsync(symbol);
// Add historical data if requested
if (includeHistory)
{
var historyData = await GetStockHistoryAsync(symbol);
quoteResult.Add("history", historyData);
}
// Return formatted result
return new ToolResponse {
Result = JsonSerializer.SerializeToElement(quoteResult)
};
}
private async Task<Dictionary<string, object>> GetStockQuoteAsync(string symbol)
{
// Implementation would call a real stock API
// This is a simplified example
return new Dictionary<string, object>
{
["symbol"] = symbol,
["price"] = 123.45,
["change"] = 2.5,
["percentChange"] = 1.2,
["lastUpdated"] = DateTime.UtcNow
};
}
private async Task<object> GetStockHistoryAsync(string symbol)
{
// Implementation would get historical data
// Simplified example
return new[]
{
new { date = DateTime.Now.AddDays(-7).Date, price = 120.25 },
new { date = DateTime.Now.AddDays(-6).Date, price = 122.50 },
new { date = DateTime.Now.AddDays(-5).Date, price = 121.75 }
// More historical data...
};
}
}
}
// Create package and publish to NuGet
// dotnet pack -c Release
// dotnet nuget push bin/Release/McpFinanceTools.1.0.0.nupkg -s https://api.nuget.org/v3/index.json -k YOUR_API_KEY// configurazione pom.xml per un pacchetto strumento MCP condivisibile
<!--
<project>
<groupId>com.example</groupId>
<artifactId>mcp-weather-tools</artifactId>
<version>1.0.0</version>
<dependencies>
<dependency>
<groupId>com.mcp</groupId>
<artifactId>mcp-server</artifactId>
<version>1.0.0</version>
</dependency>
</dependencies>
<distributionManagement>
<repository>
<id>github</id>
<name>GitHub Packages</name>
<url>https://maven.pkg.github.com/username/mcp-weather-tools</url>
</repository>
</distributionManagement>
</project>
-->
package com.example.mcp.weather;
import com.mcp.tools.Tool;
import com.mcp.tools.ToolRequest;
import com.mcp.tools.ToolResponse;
import com.mcp.tools.ToolExecutionException;
import java.net.http.HttpClient;
import java.net.http.HttpRequest;
import java.net.http.HttpResponse;
import java.net.URI;
import java.util.HashMap;
import java.util.Map;
public class WeatherForecastTool implements Tool {
private final HttpClient httpClient;
private final String apiKey;
public WeatherForecastTool(String apiKey) {
this.httpClient = HttpClient.newHttpClient();
this.apiKey = apiKey;
}
@Override
public String getName() {
return "weatherForecast";
}
@Override
public String getDescription() {
return "Gets weather forecast for a specified location";
}
@Override
public Object getSchema() {
Map<String, Object> schema = new HashMap<>();
// Definizione dello schema...
return schema;
}
@Override
public ToolResponse execute(ToolRequest request) {
try {
String location = request.getParameters().get("location").asText();
int days = request.getParameters().has("days") ?
request.getParameters().get("days").asInt() : 3;
// Chiamare l'API meteo
Map<String, Object> forecast = getForecast(location, days);
// Costruire la risposta
return new ToolResponse.Builder()
.setResult(forecast)
.build();
} catch (Exception ex) {
throw new ToolExecutionException("Weather forecast failed: " + ex.getMessage(), ex);
}
}
private Map<String, Object> getForecast(String location, int days) {
// L'implementazione chiamerebbe l'API meteo
// Esempio semplificato
Map<String, Object> result = new HashMap<>();
// Aggiungere dati di previsione...
return result;
}
}
// Costruire e pubblicare usando Maven
// mvn clean package
// mvn deploy# Struttura della directory per un pacchetto PyPI:
# mcp_nlp_tools/
# ├── LICENSE
# ├── README.md
# ├── setup.py
# ├── mcp_nlp_tools/
# │ ├── __init__.py
# │ ├── sentiment_tool.py
# │ └── translation_tool.py
# Esempio di setup.py
"""
from setuptools import setup, find_packages
setup(
name="mcp_nlp_tools",
version="0.1.0",
packages=find_packages(),
install_requires=[
"mcp_server>=1.0.0",
"transformers>=4.0.0",
"torch>=1.8.0"
],
author="Your Name",
author_email="your.email@example.com",
description="MCP tools for natural language processing tasks",
long_description=open("README.md").read(),
long_description_content_type="text/markdown",
url="https://github.com/username/mcp_nlp_tools",
classifiers=[
"Programming Language :: Python :: 3",
"License :: OSI Approved :: MIT License",
"Operating System :: OS Independent",
],
python_requires=">=3.8",
)
"""
# Esempio di implementazione di uno strumento NLP (sentiment_tool.py)
from mcp_tools import Tool, ToolRequest, ToolResponse, ToolExecutionException
from transformers import pipeline
import torch
class SentimentAnalysisTool(Tool):
"""MCP tool for sentiment analysis of text"""
def __init__(self, model_name="distilbert-base-uncased-finetuned-sst-2-english"):
# Carica il modello di analisi del sentiment
self.sentiment_analyzer = pipeline("sentiment-analysis", model=model_name)
def get_name(self):
return "sentimentAnalysis"
def get_description(self):
return "Analyzes the sentiment of text, classifying it as positive or negative"
def get_schema(self):
return {
"type": "object",
"properties": {
"text": {
"type": "string",
"description": "The text to analyze for sentiment"
},
"includeScore": {
"type": "boolean",
"description": "Whether to include confidence scores",
"default": True
}
},
"required": ["text"]
}
async def execute_async(self, request: ToolRequest) -> ToolResponse:
try:
# Estrai i parametri
text = request.parameters.get("text")
include_score = request.parameters.get("includeScore", True)
# Analizza il sentiment
sentiment_result = self.sentiment_analyzer(text)[0]
# Format del risultato
result = {
"sentiment": sentiment_result["label"],
"text": text
}
if include_score:
result["score"] = sentiment_result["score"]
# Restituisci il risultato
return ToolResponse(result=result)
except Exception as e:
raise ToolExecutionException(f"Sentiment analysis failed: {str(e)}")
# Per pubblicare:
# python setup.py sdist bdist_wheel
# python -m twine upload dist/*Quando condividi strumenti MCP con la comunità:
-
Documentazione Completa:
- Documenta scopo, utilizzo ed esempi
- Spiega parametri e valori di ritorno
- Documenta eventuali dipendenze esterne
-
Gestione degli Errori:
- Implementa una robusta gestione degli errori
- Fornisci messaggi di errore utili
- Gestisci con cura i casi limite
-
Considerazioni sulle Prestazioni:
- Ottimizza sia per velocità che per uso delle risorse
- Implementa caching quando appropriato
- Considera la scalabilità
-
Sicurezza:
- Usa chiavi API sicure e autenticazione
- Valida e sanitizza gli input
- Implementa limitazioni di velocità per chiamate API esterne
-
Testing:
- Includi una copertura di test completa
- Testa con diversi tipi di input e casi limite
- Documenta le procedure di test
La collaborazione efficace è la chiave per un ecosistema MCP prospero.
- Issue e Discussioni su GitHub
- Microsoft Tech Community
- Canali Discord e Slack
- Stack Overflow (tag:
model-context-protocolomcp)
Quando si revisionano contributi MCP:
- Chiarezza: Il codice è chiaro e ben documentato?
- Correttezza: Funziona come previsto?
- Coerenza: Segue le convenzioni del progetto?
- Completezza: Include test e documentazione?
- Sicurezza: Ci sono preoccupazioni di sicurezza?
Quando si sviluppa per MCP:
- Versionamento del Protocollo: Attieniti alla versione del protocollo MCP supportata dal tuo strumento
- Compatibilità Client: Considera la compatibilità retroattiva
- Compatibilità Server: Segui le linee guida di implementazione del server
- Modifiche Incompatibili: Documenta chiaramente eventuali modifiche incompatibili
Un importante contributo comunitario potrebbe essere lo sviluppo di un registro pubblico per gli strumenti MCP.
# Esempio di schema per un'API del registro degli strumenti comunitario
from fastapi import FastAPI, HTTPException, Depends
from pydantic import BaseModel, Field, HttpUrl
from typing import List, Optional
import datetime
import uuid
# Modelli per il registro degli strumenti
class ToolSchema(BaseModel):
"""JSON Schema for a tool"""
type: str
properties: dict
required: List[str] = []
class ToolRegistration(BaseModel):
"""Information for registering a tool"""
name: str = Field(..., description="Unique name for the tool")
description: str = Field(..., description="Description of what the tool does")
version: str = Field(..., description="Semantic version of the tool")
schema: ToolSchema = Field(..., description="JSON Schema for tool parameters")
author: str = Field(..., description="Author of the tool")
repository: Optional[HttpUrl] = Field(None, description="Repository URL")
documentation: Optional[HttpUrl] = Field(None, description="Documentation URL")
package: Optional[HttpUrl] = Field(None, description="Package URL")
tags: List[str] = Field(default_factory=list, description="Tags for categorization")
examples: List[dict] = Field(default_factory=list, description="Example usage")
class Tool(ToolRegistration):
"""Tool with registry metadata"""
id: uuid.UUID = Field(default_factory=uuid.uuid4)
created_at: datetime.datetime = Field(default_factory=datetime.datetime.now)
updated_at: datetime.datetime = Field(default_factory=datetime.datetime.now)
downloads: int = Field(default=0)
rating: float = Field(default=0.0)
ratings_count: int = Field(default=0)
# Applicazione FastAPI per il registro
app = FastAPI(title="MCP Tool Registry")
# Database in memoria per questo esempio
tools_db = {}
@app.post("/tools", response_model=Tool)
async def register_tool(tool: ToolRegistration):
"""Register a new tool in the registry"""
if tool.name in tools_db:
raise HTTPException(status_code=400, detail=f"Tool '{tool.name}' already exists")
new_tool = Tool(**tool.dict())
tools_db[tool.name] = new_tool
return new_tool
@app.get("/tools", response_model=List[Tool])
async def list_tools(tag: Optional[str] = None):
"""List all registered tools, optionally filtered by tag"""
if tag:
return [tool for tool in tools_db.values() if tag in tool.tags]
return list(tools_db.values())
@app.get("/tools/{tool_name}", response_model=Tool)
async def get_tool(tool_name: str):
"""Get information about a specific tool"""
if tool_name not in tools_db:
raise HTTPException(status_code=404, detail=f"Tool '{tool_name}' not found")
return tools_db[tool_name]
@app.delete("/tools/{tool_name}")
async def delete_tool(tool_name: str):
"""Delete a tool from the registry"""
if tool_name not in tools_db:
raise HTTPException(status_code=404, detail=f"Tool '{tool_name}' not found")
del tools_db[tool_name]
return {"message": f"Tool '{tool_name}' deleted"}- La comunità MCP è diversificata e accoglie vari tipi di contributi
- Contribuire a MCP può spaziare da miglioramenti al protocollo core a strumenti personalizzati
- Seguire le linee guida per i contributi migliora le possibilità di accettazione della tua PR
- Creare e condividere strumenti MCP è un modo prezioso per migliorare l'ecosistema
- La collaborazione comunitaria è essenziale per la crescita e il miglioramento di MCP
- Identifica un'area nell'ecosistema MCP dove potresti fare un contributo in base alle tue competenze e interessi
- Fai un fork del repository MCP e configura un ambiente di sviluppo locale
- Crea un piccolo miglioramento, una correzione di bug o uno strumento che possa beneficiare la comunità
- Documenta il tuo contributo con test e documentazione adeguati
- Invia una pull request al repository appropriato
Successivo: Lezioni dall'Adozione Precoce
Disclaimer:
Questo documento è stato tradotto utilizzando il servizio di traduzione automatica AI Co-op Translator. Pur impegnandoci per garantire l’accuratezza, si prega di considerare che le traduzioni automatiche possono contenere errori o imprecisioni. Il documento originale nella sua lingua nativa deve essere considerato la fonte autorevole. Per informazioni critiche, si raccomanda una traduzione professionale effettuata da un traduttore umano. Non siamo responsabili per eventuali fraintendimenti o interpretazioni errate derivanti dall’uso di questa traduzione.
