feat(web): full SEO stack — metadata, JSON-LD, sitemap, robots, OG image
Some checks failed
Deploy to Production / deploy (push) Failing after 46s
Some checks failed
Deploy to Production / deploy (push) Failing after 46s
Ported and adapted from the BuildMyDiscord SEO setup: - lib/seo.ts — single source for site constants, the FAQ data (shared by the rendered FAQ and the FAQPage schema so they never drift) and JSON-LD builders. - Rich root metadata: title template, keywords, Open Graph, Twitter card, robots directives, canonical. - JSON-LD: Organization + WebSite + SoftwareApplication sitewide, FAQPage on the landing page. No AggregateRating — there are no real reviews yet. - app/robots.ts — allow all, explicit allow-list for AI answer-engine crawlers (GPTBot, ClaudeBot, PerplexityBot, …), disallow private routes. - app/sitemap.ts — every public marketing + docs route. - app/opengraph-image.tsx — monochrome on-brand 1200x630 share card. - app/manifest.ts + public/llms.txt. - Per-page metadata for pricing, changelog, security, privacy, terms, docs, templates and status. - opengraph-image + apple-icon pinned to the edge runtime — next/og crashes during a Node-runtime prerender. Verified: next build passes; /robots.txt, /sitemap.xml, /manifest.webmanifest and /opengraph-image all generate. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
parent
617886352c
commit
b843394d0f
@ -1,6 +1,12 @@
|
|||||||
import { CodeBlock } from '@/components/code-block';
|
import { CodeBlock } from '@/components/code-block';
|
||||||
|
import { pageMetadata } from '@/lib/seo';
|
||||||
|
|
||||||
export const metadata = { title: 'Changelog — BuildMyMCPServer' };
|
export const metadata = pageMetadata({
|
||||||
|
title: 'Changelog',
|
||||||
|
description:
|
||||||
|
'Product updates and release notes for BuildMyMCPServer — new features, fixes and improvements to the MCP server platform.',
|
||||||
|
path: '/changelog',
|
||||||
|
});
|
||||||
|
|
||||||
interface Release {
|
interface Release {
|
||||||
version: string;
|
version: string;
|
||||||
|
|||||||
@ -1,5 +1,7 @@
|
|||||||
import Link from 'next/link';
|
|
||||||
import { CodeBlock } from '@/components/code-block';
|
import { CodeBlock } from '@/components/code-block';
|
||||||
|
import { JsonLd } from '@/components/json-ld';
|
||||||
|
import { FAQ, faqJsonLd } from '@/lib/seo';
|
||||||
|
import Link from 'next/link';
|
||||||
|
|
||||||
const PROMPT_EXAMPLE = `Create an MCP server that searches our Notion workspace.
|
const PROMPT_EXAMPLE = `Create an MCP server that searches our Notion workspace.
|
||||||
Tools: search_pages, get_page_content.
|
Tools: search_pages, get_page_content.
|
||||||
@ -45,62 +47,37 @@ const MARKETPLACE_POINTS: { t: string; d: string }[] = [
|
|||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const FAQ: { q: string; a: string }[] = [
|
|
||||||
{
|
|
||||||
q: 'What is MCP?',
|
|
||||||
a: 'Model Context Protocol — an open standard from Anthropic for connecting AI assistants to external tools, data and APIs over a transport like Streamable HTTP.',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
q: 'Do I need to write code?',
|
|
||||||
a: 'No. You describe the tool in natural language. We generate the TypeScript server, run static checks, build a Docker image and deploy it to a public OAuth-protected URL.',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
q: 'Which clients work?',
|
|
||||||
a: 'Claude Desktop, Cursor, ChatGPT Custom Connectors, VS Code Copilot, Continue.dev — anything that speaks the MCP spec.',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
q: 'How is auth handled?',
|
|
||||||
a: 'Every generated server is an OAuth 2.1 Resource Server. Our control plane is the Authorization Server (PKCE + Dynamic Client Registration + Resource Indicators per RFC 8707).',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
q: 'Can I self-host?',
|
|
||||||
a: 'Yes. The runner is a plain Docker container; the control plane is open to BYO Postgres + Redis. See the self-hosting guide in docs.',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
q: 'What about secrets?',
|
|
||||||
a: 'AES-256-GCM at rest in Postgres, injected as environment variables into the runtime container. Never logged, never echoed back.',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
q: 'Cold starts?',
|
|
||||||
a: 'No cold starts. Containers stay warm. Sub-50ms tool-call overhead on average for in-region requests.',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
q: 'Rate limits?',
|
|
||||||
a: 'Default 100 requests/min/IP per tool. Configurable per server. Quota enforced at the Traefik layer before hitting your container.',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
q: 'How fast is generation?',
|
|
||||||
a: 'Spec → image → live URL typically completes in 45-90 seconds.',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
q: 'Logs and metrics?',
|
|
||||||
a: 'Live log streaming to the dashboard, structured tool-call metrics (P50/P95/P99 latency, error rate, per-tool throughput) — all retained for 30 days.',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
q: 'What if I cancel?',
|
|
||||||
a: 'You can export the full TypeScript source of every server you built. No vendor lock-in.',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
q: 'Custom domain?',
|
|
||||||
a: 'Pro plan and above. Add a CNAME, we provision Let’s Encrypt automatically.',
|
|
||||||
},
|
|
||||||
];
|
|
||||||
|
|
||||||
const TIERS = [
|
const TIERS = [
|
||||||
{ name: 'Hobby', price: '€0', tag: 'Forever free', features: ['1 server', '100k calls/mo', 'BMM subdomain', 'Community support'] },
|
{
|
||||||
{ name: 'Pro', price: '€49', tag: '/ month', features: ['5 servers', '1M calls/mo', 'Custom domain', 'Priority build queue', 'Email support'] },
|
name: 'Hobby',
|
||||||
{ name: 'Team', price: '€149', tag: '/ month', features: ['25 servers', '10M calls/mo', 'RBAC + audit log', 'SLA 99.9%', 'Slack support'] },
|
price: '€0',
|
||||||
{ name: 'Enterprise', price: '€499+', tag: '/ month', features: ['Unlimited', 'BYOC', 'SSO / SAML', 'Dedicated cluster', 'Customer success'] },
|
tag: 'Forever free',
|
||||||
|
features: ['1 server', '100k calls/mo', 'BMM subdomain', 'Community support'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Pro',
|
||||||
|
price: '€49',
|
||||||
|
tag: '/ month',
|
||||||
|
features: [
|
||||||
|
'5 servers',
|
||||||
|
'1M calls/mo',
|
||||||
|
'Custom domain',
|
||||||
|
'Priority build queue',
|
||||||
|
'Email support',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Team',
|
||||||
|
price: '€149',
|
||||||
|
tag: '/ month',
|
||||||
|
features: ['25 servers', '10M calls/mo', 'RBAC + audit log', 'SLA 99.9%', 'Slack support'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Enterprise',
|
||||||
|
price: '€499+',
|
||||||
|
tag: '/ month',
|
||||||
|
features: ['Unlimited', 'BYOC', 'SSO / SAML', 'Dedicated cluster', 'Customer success'],
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
export default function Landing() {
|
export default function Landing() {
|
||||||
@ -176,12 +153,26 @@ export default function Landing() {
|
|||||||
</div>
|
</div>
|
||||||
<div className="grid gap-6 md:grid-cols-3">
|
<div className="grid gap-6 md:grid-cols-3">
|
||||||
{[
|
{[
|
||||||
{ n: '01', t: 'Describe your tool', d: 'A sentence is enough. List your secrets and which APIs to call.' },
|
{
|
||||||
{ n: '02', t: 'We generate, check, deploy', d: 'Claude writes the spec. We render TypeScript, run static checks, build a container, deploy to your subdomain.' },
|
n: '01',
|
||||||
{ n: '03', t: 'Install in your client', d: 'Copy the snippet into Claude Desktop, Cursor or ChatGPT. OAuth flow on first use.' },
|
t: 'Describe your tool',
|
||||||
|
d: 'A sentence is enough. List your secrets and which APIs to call.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
n: '02',
|
||||||
|
t: 'We generate, check, deploy',
|
||||||
|
d: 'Claude writes the spec. We render TypeScript, run static checks, build a container, deploy to your subdomain.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
n: '03',
|
||||||
|
t: 'Install in your client',
|
||||||
|
d: 'Copy the snippet into Claude Desktop, Cursor or ChatGPT. OAuth flow on first use.',
|
||||||
|
},
|
||||||
].map((s) => (
|
].map((s) => (
|
||||||
<div key={s.n} className="panel p-5">
|
<div key={s.n} className="panel p-5">
|
||||||
<div className="mono text-[11px] tracking-widest text-[--color-fg-subtle]">{s.n}</div>
|
<div className="mono text-[11px] tracking-widest text-[--color-fg-subtle]">
|
||||||
|
{s.n}
|
||||||
|
</div>
|
||||||
<h3 className="mt-4 text-[15px] font-semibold tracking-tight">{s.t}</h3>
|
<h3 className="mt-4 text-[15px] font-semibold tracking-tight">{s.t}</h3>
|
||||||
<p className="mt-2 text-[13px] leading-relaxed text-[--color-fg-muted]">{s.d}</p>
|
<p className="mt-2 text-[13px] leading-relaxed text-[--color-fg-muted]">{s.d}</p>
|
||||||
</div>
|
</div>
|
||||||
@ -211,16 +202,23 @@ export default function Landing() {
|
|||||||
<section className="border-b border-[--color-border] py-20">
|
<section className="border-b border-[--color-border] py-20">
|
||||||
<div className="mx-auto max-w-6xl px-6">
|
<div className="mx-auto max-w-6xl px-6">
|
||||||
<div className="mb-10 max-w-2xl">
|
<div className="mb-10 max-w-2xl">
|
||||||
<h2 className="text-[28px] font-semibold tracking-tight">Built for the work you actually have</h2>
|
<h2 className="text-[28px] font-semibold tracking-tight">
|
||||||
|
Built for the work you actually have
|
||||||
|
</h2>
|
||||||
<p className="mt-2 text-[14px] text-[--color-fg-muted]">
|
<p className="mt-2 text-[14px] text-[--color-fg-muted]">
|
||||||
Anything with an HTTP API or a database, in minutes.
|
Anything with an HTTP API or a database, in minutes.
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
<div className="grid gap-3 md:grid-cols-3">
|
<div className="grid gap-3 md:grid-cols-3">
|
||||||
{EXAMPLES.map((e) => (
|
{EXAMPLES.map((e) => (
|
||||||
<div key={e.title} className="panel p-4 transition-colors hover:border-[--color-border-strong]">
|
<div
|
||||||
|
key={e.title}
|
||||||
|
className="panel p-4 transition-colors hover:border-[--color-border-strong]"
|
||||||
|
>
|
||||||
<div className="text-[13px] font-semibold tracking-tight">{e.title}</div>
|
<div className="text-[13px] font-semibold tracking-tight">{e.title}</div>
|
||||||
<p className="mt-1 text-[12.5px] leading-relaxed text-[--color-fg-muted]">{e.desc}</p>
|
<p className="mt-1 text-[12.5px] leading-relaxed text-[--color-fg-muted]">
|
||||||
|
{e.desc}
|
||||||
|
</p>
|
||||||
</div>
|
</div>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
@ -273,7 +271,9 @@ export default function Landing() {
|
|||||||
key={t.name}
|
key={t.name}
|
||||||
className={`panel p-5 ${i === 1 ? 'border-[--color-accent]/40' : ''}`}
|
className={`panel p-5 ${i === 1 ? 'border-[--color-accent]/40' : ''}`}
|
||||||
>
|
>
|
||||||
<div className="text-[12px] uppercase tracking-wider text-[--color-fg-subtle]">{t.name}</div>
|
<div className="text-[12px] uppercase tracking-wider text-[--color-fg-subtle]">
|
||||||
|
{t.name}
|
||||||
|
</div>
|
||||||
<div className="mt-2 flex items-baseline gap-1">
|
<div className="mt-2 flex items-baseline gap-1">
|
||||||
<span className="text-[26px] font-semibold tracking-tight">{t.price}</span>
|
<span className="text-[26px] font-semibold tracking-tight">{t.price}</span>
|
||||||
<span className="text-[12px] text-[--color-fg-subtle]">{t.tag}</span>
|
<span className="text-[12px] text-[--color-fg-subtle]">{t.tag}</span>
|
||||||
@ -291,6 +291,7 @@ export default function Landing() {
|
|||||||
|
|
||||||
{/* FAQ */}
|
{/* FAQ */}
|
||||||
<section className="py-20">
|
<section className="py-20">
|
||||||
|
<JsonLd data={faqJsonLd()} />
|
||||||
<div className="mx-auto max-w-6xl px-6">
|
<div className="mx-auto max-w-6xl px-6">
|
||||||
<h2 className="text-[28px] font-semibold tracking-tight">FAQ</h2>
|
<h2 className="text-[28px] font-semibold tracking-tight">FAQ</h2>
|
||||||
<div className="mt-8 grid gap-x-12 gap-y-6 md:grid-cols-2">
|
<div className="mt-8 grid gap-x-12 gap-y-6 md:grid-cols-2">
|
||||||
|
|||||||
@ -1,6 +1,12 @@
|
|||||||
|
import { pageMetadata } from '@/lib/seo';
|
||||||
import Link from 'next/link';
|
import Link from 'next/link';
|
||||||
|
|
||||||
export const metadata = { title: 'Pricing — BuildMyMCPServer' };
|
export const metadata = pageMetadata({
|
||||||
|
title: 'Pricing',
|
||||||
|
description:
|
||||||
|
'BuildMyMCPServer pricing — start free with one hosted MCP server, scale to Pro, Team and Enterprise. Pay for tool calls, not boilerplate.',
|
||||||
|
path: '/pricing',
|
||||||
|
});
|
||||||
|
|
||||||
const TIERS = [
|
const TIERS = [
|
||||||
{
|
{
|
||||||
@ -113,7 +119,9 @@ export default function Pricing() {
|
|||||||
<span className="text-[28px] font-semibold tracking-tight">{t.price}</span>
|
<span className="text-[28px] font-semibold tracking-tight">{t.price}</span>
|
||||||
<span className="text-[12px] text-[--color-fg-subtle]">{t.tag}</span>
|
<span className="text-[12px] text-[--color-fg-subtle]">{t.tag}</span>
|
||||||
</div>
|
</div>
|
||||||
<p className="mt-2 text-[12px] leading-relaxed text-[--color-fg-muted]">{t.description}</p>
|
<p className="mt-2 text-[12px] leading-relaxed text-[--color-fg-muted]">
|
||||||
|
{t.description}
|
||||||
|
</p>
|
||||||
<ul className="mt-4 space-y-1.5 text-[12.5px] text-[--color-fg-muted]">
|
<ul className="mt-4 space-y-1.5 text-[12.5px] text-[--color-fg-muted]">
|
||||||
{t.features.map((f) => (
|
{t.features.map((f) => (
|
||||||
<li key={f}>— {f}</li>
|
<li key={f}>— {f}</li>
|
||||||
|
|||||||
@ -1,4 +1,11 @@
|
|||||||
export const metadata = { title: 'Privacy — BuildMyMCPServer' };
|
import { pageMetadata } from '@/lib/seo';
|
||||||
|
|
||||||
|
export const metadata = pageMetadata({
|
||||||
|
title: 'Privacy',
|
||||||
|
description:
|
||||||
|
'BuildMyMCPServer privacy policy — what data we collect, how it is used, and the rights you have over it.',
|
||||||
|
path: '/privacy',
|
||||||
|
});
|
||||||
|
|
||||||
const SECTIONS = [
|
const SECTIONS = [
|
||||||
{
|
{
|
||||||
@ -29,7 +36,7 @@ const SECTIONS = [
|
|||||||
{
|
{
|
||||||
h: 'Subprocessors',
|
h: 'Subprocessors',
|
||||||
p: [
|
p: [
|
||||||
'Anthropic (generation) — only the prompt text you send. Anthropic\'s data-retention policy applies.',
|
"Anthropic (generation) — only the prompt text you send. Anthropic's data-retention policy applies.",
|
||||||
'Hetzner (compute).',
|
'Hetzner (compute).',
|
||||||
'Backblaze (encrypted backups).',
|
'Backblaze (encrypted backups).',
|
||||||
'Stripe (billing).',
|
'Stripe (billing).',
|
||||||
@ -91,7 +98,10 @@ export default function Privacy() {
|
|||||||
<h2 className="text-[16px] font-semibold tracking-tight">Contact</h2>
|
<h2 className="text-[16px] font-semibold tracking-tight">Contact</h2>
|
||||||
<p className="mt-3 text-[13.5px] leading-relaxed text-[--color-fg-muted]">
|
<p className="mt-3 text-[13.5px] leading-relaxed text-[--color-fg-muted]">
|
||||||
Data controller: BuildMyMCPServer. Email{' '}
|
Data controller: BuildMyMCPServer. Email{' '}
|
||||||
<a className="text-[--color-accent] underline" href="mailto:privacy@buildmymcpserver.com">
|
<a
|
||||||
|
className="text-[--color-accent] underline"
|
||||||
|
href="mailto:privacy@buildmymcpserver.com"
|
||||||
|
>
|
||||||
privacy@buildmymcpserver.com
|
privacy@buildmymcpserver.com
|
||||||
</a>{' '}
|
</a>{' '}
|
||||||
for any of the above.
|
for any of the above.
|
||||||
|
|||||||
@ -1,7 +1,13 @@
|
|||||||
import Link from 'next/link';
|
|
||||||
import { CodeBlock } from '@/components/code-block';
|
import { CodeBlock } from '@/components/code-block';
|
||||||
|
import { pageMetadata } from '@/lib/seo';
|
||||||
|
import Link from 'next/link';
|
||||||
|
|
||||||
export const metadata = { title: 'Security — BuildMyMCPServer' };
|
export const metadata = pageMetadata({
|
||||||
|
title: 'Security',
|
||||||
|
description:
|
||||||
|
'How BuildMyMCPServer secures your MCP servers — per-server Docker isolation, AES-256-GCM encrypted secrets, OAuth 2.1 and a hardened control plane.',
|
||||||
|
path: '/security',
|
||||||
|
});
|
||||||
|
|
||||||
const PILLARS = [
|
const PILLARS = [
|
||||||
{
|
{
|
||||||
@ -18,7 +24,7 @@ const PILLARS = [
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: 'No token passthrough',
|
title: 'No token passthrough',
|
||||||
body: 'When a tool calls a downstream API, it uses its own server-side credentials — not the user\'s OAuth token. Tokens never leak across trust boundaries. This is mandated by the MCP authorization spec.',
|
body: "When a tool calls a downstream API, it uses its own server-side credentials — not the user's OAuth token. Tokens never leak across trust boundaries. This is mandated by the MCP authorization spec.",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: 'Static security checks',
|
title: 'Static security checks',
|
||||||
@ -49,8 +55,8 @@ export default function Security() {
|
|||||||
Built like infrastructure.
|
Built like infrastructure.
|
||||||
</h1>
|
</h1>
|
||||||
<p className="mt-3 text-[14px] leading-relaxed text-[--color-fg-muted]">
|
<p className="mt-3 text-[14px] leading-relaxed text-[--color-fg-muted]">
|
||||||
We host code generated by an LLM, on behalf of customers, that exposes their internal
|
We host code generated by an LLM, on behalf of customers, that exposes their internal APIs
|
||||||
APIs to AI clients. The threat model is real. Here is what we do about it.
|
to AI clients. The threat model is real. Here is what we do about it.
|
||||||
</p>
|
</p>
|
||||||
</header>
|
</header>
|
||||||
|
|
||||||
@ -67,7 +73,10 @@ export default function Security() {
|
|||||||
<h2 className="text-[18px] font-semibold tracking-tight">Disclosure</h2>
|
<h2 className="text-[18px] font-semibold tracking-tight">Disclosure</h2>
|
||||||
<p className="mt-2 text-[13.5px] leading-relaxed text-[--color-fg-muted]">
|
<p className="mt-2 text-[13.5px] leading-relaxed text-[--color-fg-muted]">
|
||||||
Found a vulnerability? Email{' '}
|
Found a vulnerability? Email{' '}
|
||||||
<a className="text-[--color-accent] underline" href="mailto:security@buildmymcpserver.com">
|
<a
|
||||||
|
className="text-[--color-accent] underline"
|
||||||
|
href="mailto:security@buildmymcpserver.com"
|
||||||
|
>
|
||||||
security@buildmymcpserver.com
|
security@buildmymcpserver.com
|
||||||
</a>{' '}
|
</a>{' '}
|
||||||
with a clear reproduction. We respond within 48h. We do not run a paid bounty yet, but we
|
with a clear reproduction. We respond within 48h. We do not run a paid bounty yet, but we
|
||||||
|
|||||||
14
apps/web/app/(marketing)/status/layout.tsx
Normal file
14
apps/web/app/(marketing)/status/layout.tsx
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
import { pageMetadata } from '@/lib/seo';
|
||||||
|
|
||||||
|
// status/page.tsx is a client component and cannot export metadata itself —
|
||||||
|
// this layout carries it.
|
||||||
|
export const metadata = pageMetadata({
|
||||||
|
title: 'Status',
|
||||||
|
description:
|
||||||
|
'Live operational status of BuildMyMCPServer — control plane, build pipeline and hosted MCP servers.',
|
||||||
|
path: '/status',
|
||||||
|
});
|
||||||
|
|
||||||
|
export default function StatusLayout({ children }: { children: React.ReactNode }) {
|
||||||
|
return children;
|
||||||
|
}
|
||||||
@ -1,4 +1,11 @@
|
|||||||
export const metadata = { title: 'Terms — BuildMyMCPServer' };
|
import { pageMetadata } from '@/lib/seo';
|
||||||
|
|
||||||
|
export const metadata = pageMetadata({
|
||||||
|
title: 'Terms',
|
||||||
|
description:
|
||||||
|
'BuildMyMCPServer terms of service — the agreement that governs use of the platform.',
|
||||||
|
path: '/terms',
|
||||||
|
});
|
||||||
|
|
||||||
const SECTIONS = [
|
const SECTIONS = [
|
||||||
{
|
{
|
||||||
|
|||||||
@ -1,40 +1,37 @@
|
|||||||
import { ImageResponse } from 'next/og';
|
import { ImageResponse } from 'next/og';
|
||||||
|
|
||||||
|
// Edge runtime — see opengraph-image.tsx: avoids the next/og fileURLToPath
|
||||||
|
// crash during a Node-runtime prerender (notably on Windows builds).
|
||||||
|
export const runtime = 'edge';
|
||||||
|
|
||||||
export const size = { width: 180, height: 180 };
|
export const size = { width: 180, height: 180 };
|
||||||
export const contentType = 'image/png';
|
export const contentType = 'image/png';
|
||||||
|
|
||||||
export default function AppleIcon() {
|
export default function AppleIcon() {
|
||||||
return new ImageResponse(
|
return new ImageResponse(
|
||||||
(
|
<div
|
||||||
<div
|
style={{
|
||||||
style={{
|
width: '100%',
|
||||||
width: '100%',
|
height: '100%',
|
||||||
height: '100%',
|
background: '#6366F1',
|
||||||
background: '#6366F1',
|
borderRadius: 38,
|
||||||
borderRadius: 38,
|
display: 'flex',
|
||||||
display: 'flex',
|
alignItems: 'center',
|
||||||
alignItems: 'center',
|
justifyContent: 'center',
|
||||||
justifyContent: 'center',
|
}}
|
||||||
}}
|
>
|
||||||
>
|
<svg width="120" height="120" viewBox="0 0 32 32" xmlns="http://www.w3.org/2000/svg">
|
||||||
<svg
|
<title>BuildMyMCPServer</title>
|
||||||
width="120"
|
<path
|
||||||
height="120"
|
d="M8.5 22.5V9.5L16 16L23.5 9.5V22.5"
|
||||||
viewBox="0 0 32 32"
|
stroke="#FFFFFF"
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
strokeWidth={2.4}
|
||||||
>
|
strokeLinecap="round"
|
||||||
<title>BuildMyMCPServer</title>
|
strokeLinejoin="round"
|
||||||
<path
|
fill="none"
|
||||||
d="M8.5 22.5V9.5L16 16L23.5 9.5V22.5"
|
/>
|
||||||
stroke="#FFFFFF"
|
</svg>
|
||||||
strokeWidth={2.4}
|
</div>,
|
||||||
strokeLinecap="round"
|
|
||||||
strokeLinejoin="round"
|
|
||||||
fill="none"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
),
|
|
||||||
{ ...size },
|
{ ...size },
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,5 +1,13 @@
|
|||||||
import Link from 'next/link';
|
|
||||||
import { Logo } from '@/components/logo';
|
import { Logo } from '@/components/logo';
|
||||||
|
import { pageMetadata } from '@/lib/seo';
|
||||||
|
import Link from 'next/link';
|
||||||
|
|
||||||
|
export const metadata = pageMetadata({
|
||||||
|
title: 'Docs',
|
||||||
|
description:
|
||||||
|
'BuildMyMCPServer documentation — quickstart, MCP concepts, the OAuth 2.1 flow, authoring tools, self-hosting and the API reference.',
|
||||||
|
path: '/docs',
|
||||||
|
});
|
||||||
|
|
||||||
const SECTIONS: { heading: string; items: { href: string; label: string }[] }[] = [
|
const SECTIONS: { heading: string; items: { href: string; label: string }[] }[] = [
|
||||||
{
|
{
|
||||||
|
|||||||
@ -1,14 +1,57 @@
|
|||||||
|
import { JsonLd } from '@/components/json-ld';
|
||||||
import { SiteBanner } from '@/components/site-banner';
|
import { SiteBanner } from '@/components/site-banner';
|
||||||
|
import {
|
||||||
|
SEO_KEYWORDS,
|
||||||
|
SITE_DESCRIPTION,
|
||||||
|
SITE_NAME,
|
||||||
|
SITE_TAGLINE,
|
||||||
|
SITE_URL,
|
||||||
|
siteJsonLd,
|
||||||
|
} from '@/lib/seo';
|
||||||
import { GeistMono } from 'geist/font/mono';
|
import { GeistMono } from 'geist/font/mono';
|
||||||
import { GeistSans } from 'geist/font/sans';
|
import { GeistSans } from 'geist/font/sans';
|
||||||
import type { Metadata } from 'next';
|
import type { Metadata } from 'next';
|
||||||
import './globals.css';
|
import './globals.css';
|
||||||
|
|
||||||
|
const TITLE = `${SITE_NAME} — ${SITE_TAGLINE}`;
|
||||||
|
|
||||||
export const metadata: Metadata = {
|
export const metadata: Metadata = {
|
||||||
title: 'BuildMyMCPServer — Describe your tool. We host the server.',
|
metadataBase: new URL(SITE_URL),
|
||||||
description:
|
title: {
|
||||||
'From prompt to production MCP server in 60 seconds. OAuth 2.1, Streamable HTTP, ready for Claude, Cursor & ChatGPT.',
|
default: TITLE,
|
||||||
metadataBase: new URL(process.env.NEXT_PUBLIC_APP_URL ?? 'http://localhost:3001'),
|
template: `%s | ${SITE_NAME}`,
|
||||||
|
},
|
||||||
|
description: SITE_DESCRIPTION,
|
||||||
|
applicationName: SITE_NAME,
|
||||||
|
keywords: SEO_KEYWORDS,
|
||||||
|
authors: [{ name: SITE_NAME }],
|
||||||
|
creator: SITE_NAME,
|
||||||
|
publisher: SITE_NAME,
|
||||||
|
alternates: { canonical: '/' },
|
||||||
|
openGraph: {
|
||||||
|
type: 'website',
|
||||||
|
locale: 'en_US',
|
||||||
|
url: SITE_URL,
|
||||||
|
siteName: SITE_NAME,
|
||||||
|
title: TITLE,
|
||||||
|
description: SITE_DESCRIPTION,
|
||||||
|
},
|
||||||
|
twitter: {
|
||||||
|
card: 'summary_large_image',
|
||||||
|
title: TITLE,
|
||||||
|
description: SITE_DESCRIPTION,
|
||||||
|
},
|
||||||
|
robots: {
|
||||||
|
index: true,
|
||||||
|
follow: true,
|
||||||
|
googleBot: {
|
||||||
|
index: true,
|
||||||
|
follow: true,
|
||||||
|
'max-image-preview': 'large',
|
||||||
|
'max-snippet': -1,
|
||||||
|
'max-video-preview': -1,
|
||||||
|
},
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function RootLayout({ children }: { children: React.ReactNode }) {
|
export default function RootLayout({ children }: { children: React.ReactNode }) {
|
||||||
@ -19,6 +62,7 @@ export default function RootLayout({ children }: { children: React.ReactNode })
|
|||||||
suppressHydrationWarning
|
suppressHydrationWarning
|
||||||
>
|
>
|
||||||
<body>
|
<body>
|
||||||
|
<JsonLd data={siteJsonLd()} />
|
||||||
<SiteBanner />
|
<SiteBanner />
|
||||||
{children}
|
{children}
|
||||||
</body>
|
</body>
|
||||||
|
|||||||
15
apps/web/app/manifest.ts
Normal file
15
apps/web/app/manifest.ts
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
import { SITE_DESCRIPTION, SITE_NAME } from '@/lib/seo';
|
||||||
|
import type { MetadataRoute } from 'next';
|
||||||
|
|
||||||
|
export default function manifest(): MetadataRoute.Manifest {
|
||||||
|
return {
|
||||||
|
name: SITE_NAME,
|
||||||
|
short_name: 'BuildMyMCP',
|
||||||
|
description: SITE_DESCRIPTION,
|
||||||
|
start_url: '/',
|
||||||
|
display: 'standalone',
|
||||||
|
background_color: '#0a0a0b',
|
||||||
|
theme_color: '#0a0a0b',
|
||||||
|
icons: [{ src: '/icon.svg', sizes: 'any', type: 'image/svg+xml' }],
|
||||||
|
};
|
||||||
|
}
|
||||||
107
apps/web/app/opengraph-image.tsx
Normal file
107
apps/web/app/opengraph-image.tsx
Normal file
@ -0,0 +1,107 @@
|
|||||||
|
import { ImageResponse } from 'next/og';
|
||||||
|
|
||||||
|
// Edge runtime: next/og loads its assets via fileURLToPath, which throws an
|
||||||
|
// "Invalid URL" during a Node-runtime prerender. The edge runtime sidesteps
|
||||||
|
// that and is how the OG image route builds reliably.
|
||||||
|
export const runtime = 'edge';
|
||||||
|
|
||||||
|
// Sitewide Open Graph / social-share card. Monochrome to match the brand —
|
||||||
|
// flat colours only, a single indigo accent, no gradients.
|
||||||
|
export const alt = 'BuildMyMCPServer — Describe your tool. We host the MCP server.';
|
||||||
|
export const size = { width: 1200, height: 630 };
|
||||||
|
export const contentType = 'image/png';
|
||||||
|
|
||||||
|
export default function Image() {
|
||||||
|
return new ImageResponse(
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
width: '100%',
|
||||||
|
height: '100%',
|
||||||
|
display: 'flex',
|
||||||
|
flexDirection: 'column',
|
||||||
|
justifyContent: 'space-between',
|
||||||
|
backgroundColor: '#0a0a0b',
|
||||||
|
padding: '72px',
|
||||||
|
fontFamily: 'sans-serif',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div style={{ display: 'flex', alignItems: 'center', gap: '16px' }}>
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
width: '46px',
|
||||||
|
height: '46px',
|
||||||
|
display: 'flex',
|
||||||
|
alignItems: 'center',
|
||||||
|
justifyContent: 'center',
|
||||||
|
border: '2px solid #fafafa',
|
||||||
|
borderRadius: '9px',
|
||||||
|
color: '#fafafa',
|
||||||
|
fontSize: '26px',
|
||||||
|
fontWeight: 700,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
M
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
style={{ color: '#fafafa', fontSize: '28px', fontWeight: 600, letterSpacing: '-0.02em' }}
|
||||||
|
>
|
||||||
|
BuildMyMCPServer
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div style={{ display: 'flex', flexDirection: 'column' }}>
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
color: '#fafafa',
|
||||||
|
fontSize: '74px',
|
||||||
|
fontWeight: 700,
|
||||||
|
lineHeight: 1.05,
|
||||||
|
letterSpacing: '-0.03em',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Describe your tool.
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
color: '#fafafa',
|
||||||
|
fontSize: '74px',
|
||||||
|
fontWeight: 700,
|
||||||
|
lineHeight: 1.05,
|
||||||
|
letterSpacing: '-0.03em',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
We host the MCP server.
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
color: '#a1a1aa',
|
||||||
|
fontSize: '27px',
|
||||||
|
marginTop: '26px',
|
||||||
|
letterSpacing: '-0.01em',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Prompt to a hosted, OAuth 2.1-protected MCP server in 60 seconds.
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div style={{ display: 'flex', alignItems: 'center', justifyContent: 'space-between' }}>
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
display: 'flex',
|
||||||
|
alignItems: 'center',
|
||||||
|
gap: '11px',
|
||||||
|
color: '#71717a',
|
||||||
|
fontSize: '22px',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
style={{ width: '9px', height: '9px', borderRadius: '9px', backgroundColor: '#6366f1' }}
|
||||||
|
/>
|
||||||
|
OAuth 2.1 · Streamable HTTP · Docker-isolated
|
||||||
|
</div>
|
||||||
|
<div style={{ color: '#71717a', fontSize: '22px' }}>buildmymcpserver.com</div>
|
||||||
|
</div>
|
||||||
|
</div>,
|
||||||
|
{ ...size },
|
||||||
|
);
|
||||||
|
}
|
||||||
38
apps/web/app/robots.ts
Normal file
38
apps/web/app/robots.ts
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
import { SITE_URL } from '@/lib/seo';
|
||||||
|
import type { MetadataRoute } from 'next';
|
||||||
|
|
||||||
|
// AI search/answer crawlers — explicitly allowed so the product surfaces in
|
||||||
|
// ChatGPT, Claude, Perplexity, Google AI and similar answer engines.
|
||||||
|
const AI_CRAWLERS = [
|
||||||
|
'GPTBot',
|
||||||
|
'OAI-SearchBot',
|
||||||
|
'ChatGPT-User',
|
||||||
|
'ClaudeBot',
|
||||||
|
'anthropic-ai',
|
||||||
|
'Claude-Web',
|
||||||
|
'PerplexityBot',
|
||||||
|
'Perplexity-User',
|
||||||
|
'Google-Extended',
|
||||||
|
'CCBot',
|
||||||
|
'Applebot-Extended',
|
||||||
|
'Amazonbot',
|
||||||
|
'meta-externalagent',
|
||||||
|
'DuckAssistBot',
|
||||||
|
];
|
||||||
|
|
||||||
|
export default function robots(): MetadataRoute.Robots {
|
||||||
|
return {
|
||||||
|
rules: [
|
||||||
|
{
|
||||||
|
userAgent: '*',
|
||||||
|
allow: '/',
|
||||||
|
disallow: ['/api/', '/admin', '/dashboard', '/login'],
|
||||||
|
},
|
||||||
|
// Aggressive scraper with no search value.
|
||||||
|
{ userAgent: 'Bytespider', disallow: '/' },
|
||||||
|
{ userAgent: AI_CRAWLERS, allow: '/', disallow: ['/api/', '/admin', '/dashboard'] },
|
||||||
|
],
|
||||||
|
sitemap: `${SITE_URL}/sitemap.xml`,
|
||||||
|
host: SITE_URL,
|
||||||
|
};
|
||||||
|
}
|
||||||
36
apps/web/app/sitemap.ts
Normal file
36
apps/web/app/sitemap.ts
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
import { SITE_URL } from '@/lib/seo';
|
||||||
|
import type { MetadataRoute } from 'next';
|
||||||
|
|
||||||
|
type Entry = {
|
||||||
|
path: string;
|
||||||
|
priority: number;
|
||||||
|
changeFrequency: MetadataRoute.Sitemap[number]['changeFrequency'];
|
||||||
|
};
|
||||||
|
|
||||||
|
const ROUTES: Entry[] = [
|
||||||
|
{ path: '/', priority: 1.0, changeFrequency: 'weekly' },
|
||||||
|
{ path: '/pricing', priority: 0.9, changeFrequency: 'weekly' },
|
||||||
|
{ path: '/templates', priority: 0.9, changeFrequency: 'daily' },
|
||||||
|
{ path: '/docs', priority: 0.8, changeFrequency: 'weekly' },
|
||||||
|
{ path: '/docs/concepts', priority: 0.7, changeFrequency: 'monthly' },
|
||||||
|
{ path: '/docs/oauth', priority: 0.7, changeFrequency: 'monthly' },
|
||||||
|
{ path: '/docs/authoring', priority: 0.7, changeFrequency: 'monthly' },
|
||||||
|
{ path: '/docs/api-reference', priority: 0.7, changeFrequency: 'monthly' },
|
||||||
|
{ path: '/docs/self-hosting', priority: 0.7, changeFrequency: 'monthly' },
|
||||||
|
{ path: '/docs/faq', priority: 0.6, changeFrequency: 'monthly' },
|
||||||
|
{ path: '/changelog', priority: 0.6, changeFrequency: 'weekly' },
|
||||||
|
{ path: '/security', priority: 0.5, changeFrequency: 'monthly' },
|
||||||
|
{ path: '/status', priority: 0.4, changeFrequency: 'weekly' },
|
||||||
|
{ path: '/privacy', priority: 0.3, changeFrequency: 'yearly' },
|
||||||
|
{ path: '/terms', priority: 0.3, changeFrequency: 'yearly' },
|
||||||
|
];
|
||||||
|
|
||||||
|
export default function sitemap(): MetadataRoute.Sitemap {
|
||||||
|
const now = new Date();
|
||||||
|
return ROUTES.map((r) => ({
|
||||||
|
url: `${SITE_URL}${r.path}`,
|
||||||
|
lastModified: now,
|
||||||
|
changeFrequency: r.changeFrequency,
|
||||||
|
priority: r.priority,
|
||||||
|
}));
|
||||||
|
}
|
||||||
14
apps/web/app/templates/layout.tsx
Normal file
14
apps/web/app/templates/layout.tsx
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
import { pageMetadata } from '@/lib/seo';
|
||||||
|
|
||||||
|
// templates/page.tsx is a client component and cannot export metadata itself —
|
||||||
|
// this layout carries it.
|
||||||
|
export const metadata = pageMetadata({
|
||||||
|
title: 'Template marketplace',
|
||||||
|
description:
|
||||||
|
'Browse the BuildMyMCPServer template marketplace — fork a ready-made MCP server, add your own credentials and deploy in seconds.',
|
||||||
|
path: '/templates',
|
||||||
|
});
|
||||||
|
|
||||||
|
export default function TemplatesLayout({ children }: { children: React.ReactNode }) {
|
||||||
|
return children;
|
||||||
|
}
|
||||||
13
apps/web/components/json-ld.tsx
Normal file
13
apps/web/components/json-ld.tsx
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
// Renders a JSON-LD structured-data script. The `<` escape prevents a
|
||||||
|
// `</script>` sequence in the data from breaking out of the tag.
|
||||||
|
export function JsonLd({ data }: { data: object }) {
|
||||||
|
return (
|
||||||
|
<script
|
||||||
|
type="application/ld+json"
|
||||||
|
// biome-ignore lint/security/noDangerouslySetInnerHtml: JSON-LD must be injected as a raw script.
|
||||||
|
dangerouslySetInnerHTML={{
|
||||||
|
__html: JSON.stringify(data).replace(/</g, '\\u003c'),
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
210
apps/web/lib/seo.ts
Normal file
210
apps/web/lib/seo.ts
Normal file
@ -0,0 +1,210 @@
|
|||||||
|
// Central SEO source of truth — site constants, FAQ data, JSON-LD builders.
|
||||||
|
// The FAQ array here is rendered on the landing page AND emitted as FAQPage
|
||||||
|
// structured data, so the two never drift (Google requires them to match).
|
||||||
|
|
||||||
|
import type { Metadata } from 'next';
|
||||||
|
|
||||||
|
export const SITE_URL = process.env.NEXT_PUBLIC_APP_URL ?? 'https://buildmymcpserver.com';
|
||||||
|
export const SITE_NAME = 'BuildMyMCPServer';
|
||||||
|
export const SITE_TAGLINE = 'Describe your tool. We host the MCP server.';
|
||||||
|
export const SITE_DESCRIPTION =
|
||||||
|
'From a natural-language prompt to a hosted, OAuth 2.1-protected MCP server in 60 seconds. Streamable HTTP, ready for Claude, Cursor and ChatGPT.';
|
||||||
|
|
||||||
|
export const SEO_KEYWORDS = [
|
||||||
|
'MCP server',
|
||||||
|
'Model Context Protocol',
|
||||||
|
'hosted MCP server',
|
||||||
|
'MCP server hosting',
|
||||||
|
'MCP server builder',
|
||||||
|
'create MCP server',
|
||||||
|
'deploy MCP server',
|
||||||
|
'MCP server generator',
|
||||||
|
'OAuth MCP server',
|
||||||
|
'Streamable HTTP MCP',
|
||||||
|
'Claude MCP server',
|
||||||
|
'Cursor MCP',
|
||||||
|
'ChatGPT connector',
|
||||||
|
'MCP template marketplace',
|
||||||
|
];
|
||||||
|
|
||||||
|
export interface FaqItem {
|
||||||
|
q: string;
|
||||||
|
a: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Rendered on the landing page and emitted as FAQPage JSON-LD — keep in sync
|
||||||
|
// by virtue of being a single export.
|
||||||
|
export const FAQ: FaqItem[] = [
|
||||||
|
{
|
||||||
|
q: 'What is MCP?',
|
||||||
|
a: 'Model Context Protocol — an open standard from Anthropic for connecting AI assistants to external tools, data and APIs over a transport like Streamable HTTP.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
q: 'Do I need to write code?',
|
||||||
|
a: 'No. You describe the tool in natural language. We generate the TypeScript server, run static checks, build a Docker image and deploy it to a public OAuth-protected URL.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
q: 'Which clients work?',
|
||||||
|
a: 'Claude Desktop, Cursor, ChatGPT Custom Connectors, VS Code Copilot, Continue.dev — anything that speaks the MCP spec.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
q: 'How is auth handled?',
|
||||||
|
a: 'Every generated server is an OAuth 2.1 Resource Server. Our control plane is the Authorization Server (PKCE + Dynamic Client Registration + Resource Indicators per RFC 8707).',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
q: 'Can I self-host?',
|
||||||
|
a: 'Yes. The runner is a plain Docker container; the control plane is open to BYO Postgres + Redis. See the self-hosting guide in docs.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
q: 'What about secrets?',
|
||||||
|
a: 'AES-256-GCM at rest in Postgres, injected as environment variables into the runtime container. Never logged, never echoed back.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
q: 'Cold starts?',
|
||||||
|
a: 'No cold starts. Containers stay warm. Sub-50ms tool-call overhead on average for in-region requests.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
q: 'Rate limits?',
|
||||||
|
a: 'Default 100 requests/min/IP per tool. Configurable per server. Quota enforced at the proxy layer before hitting your container.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
q: 'How fast is generation?',
|
||||||
|
a: 'Spec to image to live URL typically completes in 45-90 seconds.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
q: 'Logs and metrics?',
|
||||||
|
a: 'Live log streaming to the dashboard, structured tool-call metrics (P50/P95/P99 latency, error rate, per-tool throughput) — all retained for 30 days.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
q: 'What if I cancel?',
|
||||||
|
a: 'You can export the full TypeScript source of every server you built. No vendor lock-in.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
q: 'Custom domain?',
|
||||||
|
a: 'Pro plan and above. Add a CNAME, we provision a TLS certificate automatically.',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const SOFTWARE_FEATURES = [
|
||||||
|
'Prompt to a deployed MCP server in under 60 seconds',
|
||||||
|
'OAuth 2.1 authorization server — PKCE, Dynamic Client Registration (RFC 7591), Resource Indicators (RFC 8707)',
|
||||||
|
'Streamable HTTP transport, compatible with Claude Desktop, Cursor, ChatGPT, VS Code Copilot and Continue.dev',
|
||||||
|
'Every generated server runs in an isolated Docker container',
|
||||||
|
'Customer secrets encrypted with AES-256-GCM, injected only at runtime',
|
||||||
|
'Live build-log streaming to the dashboard',
|
||||||
|
'Template marketplace — publish your server or fork someone else’s',
|
||||||
|
'Full TypeScript source export, no vendor lock-in',
|
||||||
|
'Self-hostable control plane with BYO Postgres and Redis',
|
||||||
|
];
|
||||||
|
|
||||||
|
const OFFERS = [
|
||||||
|
{
|
||||||
|
name: 'Hobby',
|
||||||
|
price: '0',
|
||||||
|
description: '1 server, 100k tool calls/month, BMM subdomain, community support.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Pro',
|
||||||
|
price: '49',
|
||||||
|
description:
|
||||||
|
'5 servers, 1M tool calls/month, custom domain, priority build queue, email support.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Team',
|
||||||
|
price: '149',
|
||||||
|
description: '25 servers, 10M tool calls/month, RBAC + audit log, 99.9% SLA, Slack support.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Enterprise',
|
||||||
|
price: '499',
|
||||||
|
description:
|
||||||
|
'Unlimited servers, bring-your-own-cloud, SSO/SAML, dedicated cluster, customer success.',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
/** Organization + WebSite + SoftwareApplication graph — emitted sitewide. */
|
||||||
|
export function siteJsonLd(): object {
|
||||||
|
return {
|
||||||
|
'@context': 'https://schema.org',
|
||||||
|
'@graph': [
|
||||||
|
{
|
||||||
|
'@type': 'Organization',
|
||||||
|
'@id': `${SITE_URL}/#organization`,
|
||||||
|
name: SITE_NAME,
|
||||||
|
url: SITE_URL,
|
||||||
|
logo: { '@type': 'ImageObject', url: `${SITE_URL}/icon.svg` },
|
||||||
|
foundingDate: '2026',
|
||||||
|
foundingLocation: {
|
||||||
|
'@type': 'Place',
|
||||||
|
address: { '@type': 'PostalAddress', addressCountry: 'CH' },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'@type': 'WebSite',
|
||||||
|
'@id': `${SITE_URL}/#website`,
|
||||||
|
url: SITE_URL,
|
||||||
|
name: SITE_NAME,
|
||||||
|
publisher: { '@id': `${SITE_URL}/#organization` },
|
||||||
|
description: SITE_DESCRIPTION,
|
||||||
|
inLanguage: 'en',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'@type': 'SoftwareApplication',
|
||||||
|
'@id': `${SITE_URL}/#software`,
|
||||||
|
name: SITE_NAME,
|
||||||
|
applicationCategory: 'DeveloperApplication',
|
||||||
|
operatingSystem: 'Web Browser',
|
||||||
|
url: SITE_URL,
|
||||||
|
inLanguage: 'en',
|
||||||
|
description:
|
||||||
|
'BuildMyMCPServer turns a natural-language prompt into a hosted, OAuth 2.1-protected Model Context Protocol (MCP) server. Describe the tools you need; the platform generates a TypeScript server, runs static checks, builds a Docker image and deploys it to a public Streamable HTTP endpoint that Claude, Cursor and ChatGPT can connect to. A template marketplace lets users publish and fork ready-made servers.',
|
||||||
|
featureList: SOFTWARE_FEATURES,
|
||||||
|
offers: OFFERS.map((o) => ({
|
||||||
|
'@type': 'Offer',
|
||||||
|
name: o.name,
|
||||||
|
price: o.price,
|
||||||
|
priceCurrency: 'EUR',
|
||||||
|
description: o.description,
|
||||||
|
})),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/** FAQPage structured data — emitted on the page that displays the FAQ. */
|
||||||
|
export function faqJsonLd(items: FaqItem[] = FAQ): object {
|
||||||
|
return {
|
||||||
|
'@context': 'https://schema.org',
|
||||||
|
'@type': 'FAQPage',
|
||||||
|
mainEntity: items.map((item) => ({
|
||||||
|
'@type': 'Question',
|
||||||
|
name: item.q,
|
||||||
|
acceptedAnswer: { '@type': 'Answer', text: item.a },
|
||||||
|
})),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Per-page metadata. `title` is a bare string so the root layout's
|
||||||
|
* "%s | BuildMyMCPServer" template appends the brand exactly once.
|
||||||
|
*/
|
||||||
|
export function pageMetadata(opts: { title: string; description: string; path: string }): Metadata {
|
||||||
|
const fullTitle = `${opts.title} | ${SITE_NAME}`;
|
||||||
|
return {
|
||||||
|
title: opts.title,
|
||||||
|
description: opts.description,
|
||||||
|
alternates: { canonical: opts.path },
|
||||||
|
openGraph: {
|
||||||
|
type: 'website',
|
||||||
|
siteName: SITE_NAME,
|
||||||
|
title: fullTitle,
|
||||||
|
description: opts.description,
|
||||||
|
url: `${SITE_URL}${opts.path}`,
|
||||||
|
},
|
||||||
|
twitter: {
|
||||||
|
card: 'summary_large_image',
|
||||||
|
title: fullTitle,
|
||||||
|
description: opts.description,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
40
apps/web/public/llms.txt
Normal file
40
apps/web/public/llms.txt
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
# BuildMyMCPServer
|
||||||
|
|
||||||
|
> Turn a natural-language prompt into a hosted, OAuth 2.1-protected Model Context Protocol (MCP) server in about 60 seconds. The platform generates a TypeScript server, runs static checks, builds a Docker image and deploys it to a public Streamable HTTP endpoint that Claude, Cursor and ChatGPT can connect to.
|
||||||
|
|
||||||
|
## What it is
|
||||||
|
|
||||||
|
BuildMyMCPServer is a platform for creating and hosting MCP servers without writing code. You describe the tools you want in plain language; the platform produces a working MCP server and runs it for you.
|
||||||
|
|
||||||
|
The workflow:
|
||||||
|
|
||||||
|
1. **Describe.** Write what the tool should do and which APIs or secrets it needs.
|
||||||
|
2. **Review.** The platform shows the generated spec — tool names, input schemas, required credentials — and lets you edit it before building.
|
||||||
|
3. **Build & deploy.** It renders the TypeScript server, runs static checks, builds a Docker image and deploys it as an isolated container behind a public OAuth-protected URL.
|
||||||
|
4. **Connect.** Drop the endpoint into any MCP client — Claude Desktop, Cursor, ChatGPT Custom Connectors, VS Code Copilot, Continue.dev.
|
||||||
|
|
||||||
|
## Key capabilities
|
||||||
|
|
||||||
|
- **Prompt-to-server generation** — no MCP boilerplate, no Docker management.
|
||||||
|
- **OAuth 2.1 authorization server** — PKCE, Dynamic Client Registration (RFC 7591), Resource Indicators (RFC 8707), RS256 JWKS.
|
||||||
|
- **Streamable HTTP transport** — the modern MCP transport, compatible with every major MCP client.
|
||||||
|
- **Per-server isolation** — each generated server runs in its own Docker container.
|
||||||
|
- **Encrypted secrets** — customer credentials are stored with AES-256-GCM envelope encryption and injected only at runtime; never logged.
|
||||||
|
- **Template marketplace** — publish a server you built as a template, or fork one someone else published and add your own credentials.
|
||||||
|
- **Source export** — export the full TypeScript source of any server. No vendor lock-in.
|
||||||
|
- **Self-hostable** — the runner is a plain Docker container; the control plane runs against your own Postgres and Redis.
|
||||||
|
|
||||||
|
## Pricing
|
||||||
|
|
||||||
|
- **Hobby** — free. 1 server, 100k tool calls/month.
|
||||||
|
- **Pro** — €49/month. 5 servers, 1M tool calls/month, custom domain, priority build queue.
|
||||||
|
- **Team** — €149/month. 25 servers, 10M tool calls/month, RBAC, audit log, 99.9% SLA.
|
||||||
|
- **Enterprise** — from €499/month. Unlimited servers, bring-your-own-cloud, SSO/SAML.
|
||||||
|
|
||||||
|
## Docs
|
||||||
|
|
||||||
|
- Concepts: https://buildmymcpserver.com/docs/concepts
|
||||||
|
- OAuth: https://buildmymcpserver.com/docs/oauth
|
||||||
|
- Authoring servers: https://buildmymcpserver.com/docs/authoring
|
||||||
|
- API reference: https://buildmymcpserver.com/docs/api-reference
|
||||||
|
- Self-hosting: https://buildmymcpserver.com/docs/self-hosting
|
||||||
Loading…
Reference in New Issue
Block a user