IronMCP Framework Starter
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@IronMCP Framework Startercreate a new tool for subscriber data balance"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
IronMCP Framework Starter
Starter empresarial para crear servidores MCP en Node.js + TypeScript con:
Transporte
stdiopara clientes locales.Transporte
httpcon Streamable HTTP stateless.Configuracion por YAML + variables de entorno.
Logs estructurados con Pino.
REST client centralizado con autenticacion.
Validadores de codigo y seguridad.
Contrato obligatorio para tools: schema, auth, timeout, auditoria e idempotencia.
Requisitos
Node.js
>=22npm
>=10
Inicio rapido
npm install
cp .env.example .env
npm run validatePara probar HTTP:
npm run dev:httpEn otra terminal:
bash scripts/smoke-http.shHealth check HTTP normal:
curl http://127.0.0.1:3000/healthzPara probar STDIO con MCP Inspector:
npm run build
npm run mcp:inspect:stdioProbar la tool REST autenticada
El proyecto trae una tool llamada call_secure_api. Por defecto llama a https://httpbin.org/bearer, que espera un header Authorization: Bearer <token>.
Edita .env:
API_AUTH_TYPE=bearer-env
API_TOKEN=demo-token
API_BASE_URL=https://httpbin.orgLuego ejecuta:
npm run dev:http
bash scripts/smoke-http.sh examples/http/call-secure-api.jsonScripts principales
npm run dev:stdio # arranca MCP por STDIO
npm run dev:http # arranca MCP por HTTP en /mcp
npm run build # compila TypeScript
npm run typecheck # valida tipos
npm run lint # ESLint
npm run format:check # Prettier check
npm run test # Vitest
npm run secrets:check # Secretlint
npm run knip # dependencias/codigo muerto
npm run mcp:validate # validador propio del estandar MCP
npm run validate # quality gate completoEstructura
src/
framework/
auth/ # proveedores de autenticacion
rest/ # REST client estandar
transports/ # STDIO y HTTP
config.ts # unico lugar permitido para process.env
create-mcp-server.ts # registro estandarizado de tools
errors.ts # errores normalizados
logger.ts # Pino; stderr cuando transport=stdio
types.ts # contrato de tools
tools/
health.tool.ts
call-secure-api.tool.tsReglas del estandar
El validador propio tools/mcp-validate.mjs falla el build si detecta:
console.*ensrc/.process.envfuera desrc/framework/config.ts.fetch(fuera desrc/framework/rest/rest-client.ts.import directo de
axios.tools sin
inputSchema,timeoutMs,auth,audit,idempotentohandler.posibles secretos hardcodeados en
config/default.yamlo.env.example.
Crear una nueva tool
Copia src/tools/health.tool.ts y cambia:
export const myTool = defineTool({
name: "my_tool",
title: "My Tool",
description: "Descripcion clara para el modelo.",
inputSchema: {
id: z.string(),
},
timeoutMs: 5000,
idempotent: true,
auth: {
required: true,
scopes: ["my-domain:read"],
},
audit: {
category: "read",
pii: false,
},
async handler(input, context) {
return context.restClient.get(`/items/${input.id}`);
},
});Luego registra la tool en src/tools/index.ts.
Notas importantes sobre STDIO
Cuando transport=stdio, el servidor MCP usa stdin/stdout para JSON-RPC. Por eso este starter manda logs a stderr y prohibe console.log en src/.
Siguiente paso sugerido
Cuando este starter ya funcione para tu equipo, el siguiente paso es convertir src/framework en paquete interno, por ejemplo @company/mcp-framework, y crear un CLI create-mcp que genere nuevos MCPs con esta misma estructura.
Crear una nueva tool: consultar balance de datos móviles
Esta guía muestra cómo crear una tool MCP que consume este endpoint REST:
GET /api/v2.0/mobile/upselling/subscribers/{subscriberId}/balances/data?_format=json&load_roaming=false&onlinetetheringquota=falseLa tool se llamará:
get_subscriber_data_balanceImportante: nunca colocar el Bearer token directamente en el código. El token debe ir en
.env.
1. Configurar variables de entorno
Editar el archivo .env:
API_BASE_URL=https://www.juliomorales.dev/apirest
API_AUTH_TYPE=bearer-env
API_TOKEN=TU_TOKEN_AQUI2. Crear el archivo de la tool
Crear el archivo:
src/tools/get-subscriber-data-balance.tool.tsAgregar este código:
import * as z from "zod/v4";
import type { RestClient } from "../framework/rest/rest-client.js";
import { defineTool } from "../framework/types.js";
const getSubscriberDataBalanceInputSchema = z.object({
subscriberId: z.string().min(8).describe("Número de suscriptor. Ejemplo: 50230448966"),
});
export function createGetSubscriberDataBalanceTool(restClient: RestClient) {
return defineTool({
name: "get_subscriber_data_balance",
title: "Get Subscriber Data Balance",
description:
"Consulta el balance de datos móviles disponibles, usados y vencimiento de un suscriptor Guatemala.",
inputSchema: {
subscriberId: z.string().min(8).describe("Número de suscriptor. Ejemplo: 50230448966"),
},
outputSchema: {
description: z.string().optional(),
remainingValue: z.number().optional(),
remainingFormatted: z.string().optional(),
usedValue: z.number().optional(),
usedFormatted: z.string().optional(),
reservedValue: z.number().optional(),
reservedFormatted: z.string().optional(),
expiresAt: z.string().optional(),
expiresIn: z.string().optional(),
includedApps: z.array(z.string()).optional(),
onlineTetheringFormatted: z.string().optional(),
raw: z.unknown(),
},
timeoutMs: 10000,
idempotent: true,
auth: {
required: true,
scopes: ["upselling:read", "balances:read"],
},
audit: {
category: "external-api",
pii: true,
},
async handler(input) {
const parsedInput = getSubscriberDataBalanceInputSchema.parse(input);
const response = await restClient.get<{
data?: {
local?: {
description?: {
value?: string;
formattedValue?: string;
};
summaryRemainingValue?: {
value?: number;
formattedValue?: string;
};
summaryReservedAmount?: {
value?: number;
formattedValue?: string;
};
summaryUsedValue?: {
value?: number;
formattedValue?: string;
};
summaryDateValue?: {
value?: string;
formattedValue?: string;
};
includedApps?: {
tags?: {
value?: string[];
};
};
onlineTethering?: {
label?: string;
value?: number;
};
};
};
}>(
`/api/v2.0/mobile/upselling/subscribers/${encodeURIComponent(
parsedInput.subscriberId,
)}/balances/data?_format=json&load_roaming=false&onlinetetheringquota=false`,
{
headers: {
Referer: "https://www.juliomorales.dev/",
Accept: "application/json, text/plain, */*",
},
},
);
const local = response.data?.local;
return {
description: local?.description?.formattedValue ?? local?.description?.value,
remainingValue: local?.summaryRemainingValue?.value,
remainingFormatted: local?.summaryRemainingValue?.formattedValue,
usedValue: local?.summaryUsedValue?.value,
usedFormatted: local?.summaryUsedValue?.formattedValue,
reservedValue: local?.summaryReservedAmount?.value,
reservedFormatted: local?.summaryReservedAmount?.formattedValue,
expiresAt: local?.summaryDateValue?.value,
expiresIn: local?.summaryDateValue?.formattedValue,
includedApps: local?.includedApps?.tags?.value,
onlineTetheringFormatted: local?.onlineTethering?.label,
raw: response,
};
},
});
}3. Registrar la tool
Editar:
src/tools/index.tsDebe quedar similar a esto:
import type { RestClient } from "../framework/rest/rest-client.js";
import { callSecureApiTool } from "./call-secure-api.tool.js";
import { createGetSubscriberBalanceTool } from "./get-subscriber-balance.tool.js";
import { createGetSubscriberDataBalanceTool } from "./get-subscriber-data-balance.tool.js";
import { healthTool } from "./health.tool.js";
export function createTools(restClient: RestClient) {
return [
healthTool,
callSecureApiTool,
createGetSubscriberBalanceTool(restClient),
createGetSubscriberDataBalanceTool(restClient),
];
}4. Crear ejemplo HTTP para probar
Crear el archivo:
examples/http/call-subscriber-data-balance.jsonCon este contenido:
{
"jsonrpc": "2.0",
"id": 2,
"method": "tools/call",
"params": {
"name": "get_subscriber_data_balance",
"arguments": {
"subscriberId": "50230448966"
}
}
}5. Validar el código
Ejecutar:
npm run validateSi todo está correcto, debe pasar:
mcp:validate
typecheck
lint
format:check
test
secrets:check6. Levantar el servidor HTTP
npm run dev:http7. Probar la tool
En otra terminal:
bash scripts/smoke-http.sh examples/http/call-subscriber-data-balance.jsonLa respuesta debe incluir información como:
{
"description": "30GB + REDES SOCIALES + MUSICA + SPORTS",
"remainingFormatted": "15.48 GB",
"usedFormatted": "14.52 GB",
"reservedFormatted": "30 GB",
"expiresIn": "11 días, 10 horas",
"includedApps": ["WHATSAPP Incluido", "Redes Sociales", "Entretenimiento"]
}Reglas importantes para developers
No usar
fetchdirecto.No usar
axiosdirecto.No colocar tokens en código.
Siempre usar
RestClient.Siempre validar input con Zod dentro del
handler.Toda tool debe tener:
namedescriptioninputSchemaoutputSchematimeoutMsauthauditidempotent
Siempre ejecutar:
npm run validateConventional Commits
IronMCP utiliza Conventional Commits.
Ejemplos:
git commit -m "feat: add MCP authentication middleware"
git commit -m "fix: resolve HTTP transport issue"
git commit -m "docs: update README"
git commit -m "refactor: simplify tool registration"
❤️ Support IronMCP
If IronMCP helps you build better MCP servers, consider supporting the project.
Your contribution helps maintain the framework, add new features, improve documentation, and keep the project open source.
Donate via PayPal
Or send a donation directly to:
info@nworldt.netEvery contribution, no matter the size, helps keep IronMCP growing.
Thank you for your support! 🚀
This server cannot be installed
Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/julioalberto64/iron-mcp-framework'
If you have feedback or need assistance with the MCP directory API, please join our Discord server