인증된 AgentReady.md 증명서
발급일 sig: 14ad8bf8747640e8 검증 →

분석된 URL

https://www.rivalr.app/

다른 URL 분석

AI-Ready 점수

71 / C

보통

/ 100

토큰 절감량

HTML 토큰 1936
Markdown 토큰 1
절감 100%

점수 상세

시맨틱 HTML 35/100
콘텐츠 효율성 93/100
AI 발견 가능성 61/100
구조화 데이터 85/100
접근성 85/100

신흥 프로토콜

3개 중 0개 감지

AI 에이전트가 찾는 well-known 엔드포인트. 감지되면 에이전트가 서비스를 자동으로 발견하고 연결할 수 있습니다.

  • OAuth Discovery RFC 8414
    /.well-known/oauth-authorization-server
  • MCP Server Card Anthropic
    /.well-known/mcp.json
  • A2A Agent Card Google
    /.well-known/agent.json

페이지에 <article> 또는 <main> 요소가 없습니다. 이 시맨틱 컨테이너는 AI 에이전트가 주요 콘텐츠 영역을 식별하고 내비게이션, 사이드바, 푸터를 무시하는 데 도움을 줍니다.

구현 방법

페이지의 주요 콘텐츠 주위에 <main> 요소를 추가하고, 블로그 게시물이나 제품 설명 같은 독립적인 콘텐츠 블록에는 <article>을 사용하세요.

제목 구조에 문제가 있습니다 (레벨 건너뛰기 또는 다중 h1 태그). 깔끔한 계층 구조는 AI 에이전트가 콘텐츠 구성을 이해하는 데 도움을 줍니다.

구현 방법

페이지당 정확히 하나의 <h1>을 유지하고, 제목이 순차적 순서를 따르도록 하세요: h1 > h2 > h3. 레벨을 건너뛰지 마세요 (예: h1에서 바로 h3).

사이트맵을 찾을 수 없습니다. 사이트맵은 AI 에이전트가 사이트의 모든 페이지를 발견하는 데 도움을 줍니다.

구현 방법

모든 공개 페이지를 나열하는 /sitemap.xml을 만드세요. 대부분의 CMS 플랫폼에서 자동 생성할 수 있습니다.

페이지가 <div> 요소에 크게 의존합니다. <section>, <nav>, <header>, <footer>, <aside> 같은 시맨틱 요소는 AI 에이전트에게 의미 있는 구조를 제공합니다.

구현 방법

범용 <div> 컨테이너를 적절한 시맨틱 요소로 교체하세요. 주제별 그룹에는 <section>, 내비게이션에는 <nav>, 페이지/섹션 헤더와 푸터에는 <header>/<footer>를 사용하세요.

사이트가 Markdown for Agents를 지원하지 않습니다. 이 Cloudflare 표준을 통해 AI 에이전트가 마크다운 형식으로 콘텐츠를 요청할 수 있으며, 토큰 사용량을 ~80% 줄일 수 있습니다.

구현 방법

다음 중 하나 이상을 구현하세요: (1) Accept: text/markdown에 마크다운 콘텐츠로 응답. (2) .md URL 제공 (예: /page.md). (3) <link rel="alternate" type="text/markdown"> 태그 추가. (4) 마크다운 발견을 위한 Link HTTP 헤더 추가.

{\n res.setHeader('Vary', 'Accept');\n res.setHeader('Link', '; rel=\"alternate\"; type=\"text/markdown\"');\n if ((req.headers.accept || '').includes('text/markdown')) {\n res.type('text/markdown; charset=utf-8');\n return res.send(renderMarkdown('page'));\n }\n res.render('page');\n});"},{"id":"fastify","label":"Fastify","language":"javascript","filename":"server.js","code":"// Mechanisms 1 + 4: content negotiation + Link header\nfastify.get('/page', async (req, reply) => {\n reply.header('Vary', 'Accept');\n reply.header('Link', '; rel=\"alternate\"; type=\"text/markdown\"');\n if ((req.headers.accept || '').includes('text/markdown')) {\n return reply.type('text/markdown; charset=utf-8').send(renderMarkdown('page'));\n }\n return reply.view('/page.ejs');\n});"},{"id":"nextjs","label":"Next.js","language":"typescript","filename":"app/page/route.ts","code":"// Next.js App Router — Route Handler returning Markdown\nimport { NextRequest } from 'next/server';\nimport { renderMarkdown } from '@/lib/md';\nexport async function GET(req: NextRequest) {\n const accept = req.headers.get('accept') || '';\n if (accept.includes('text/markdown')) {\n return new Response(await renderMarkdown('page'), {\n headers: {\n 'Content-Type': 'text/markdown; charset=utf-8',\n 'Vary': 'Accept',\n },\n });\n }\n // Fall through to the page component\n return new Response(null, { status: 404 });\n}"},{"id":"wordpress","label":"WordPress","language":"php","filename":"functions.php","code":"post_content));\n exit;\n});"},{"id":"static","label":"Hugo / Jekyll / Astro","language":"txt","filename":"static/page.md","code":"# Mechanism 2: serve .md alongside .html\n# Hugo: place page.md in /static/ — built unchanged\n# Jekyll: drop page.md in /assets/ — copied as-is\n# Astro: src/pages/page.md.ts that exports a GET returning markdown\n\n# Then advertise with mechanism 3 in :\n# "}] }'>

정규 URL을 찾을 수 없습니다. AI 에이전트가 페이지의 선호 버전을 식별하고 중복 콘텐츠를 방지하는 데 도움을 줍니다.

구현 방법

페이지의 정규 URL을 가리키는 <link rel="canonical" href="..."> 태그를 추가하세요.

많은 요소에 인라인 스타일 속성이 있습니다. 이는 콘텐츠를 추출하는 AI 에이전트에게 잡음이 됩니다.

구현 방법

모든 인라인 스타일을 스타일시트의 CSS 클래스로 이동하세요. 많은 고유 스타일이 필요하면 Tailwind 같은 유틸리티 CSS 프레임워크를 사용하세요.

Markdown 토큰: 1

  
Rivalr — Find Players & Join Activity Groups Near You

이 파일을 서버의 /index.md에 업로드하여 AI 에이전트가 페이지의 깔끔한 버전에 접근할 수 있게 하세요. Accept: text/markdown 콘텐츠 협상을 설정하여 자동으로 제공할 수도 있습니다.

권장 내용

llms.txt 다운로드
# rivalr.app

> Find players for football, basketball, padel and tennis. Join concerts, dinners and hikes. Post open spots or claim one — free, no group chat needed.

## Main
- [Rivalr — Find Players & Join Activity Groups Near You](https://www.rivalr.app/): Find players for football, basketball, padel and tennis. Join concerts, dinners and hikes. Post open spots or claim one…

전체 llms.txt는 도메인 전체 분석이 필요합니다 (곧 출시)

이 파일을 도메인 루트의 https://www.rivalr.app/llms.txt에 업로드하세요. ChatGPT, Claude, Perplexity 등의 AI 에이전트가 이 파일을 확인하여 사이트 구조를 파악합니다.

이 사이트에는 이미 llms.txt 파일이 있습니다.

유효하지 않은 형식 — # 제목으로 시작하고 의미 있는 콘텐츠가 있어야 합니다
<!doctype html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <link rel="icon" type="image/svg+xml" href="/favicon.svg" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <meta name="description" content="Find players for football, basketball, padel and tennis. Join concerts, dinners and hikes. Post open spots or claim one — free, no group chat needed." />

    <!-- Open Graph -->
    <meta property="og:type" content="website" />
    <meta property="og:url" content="https://www.rivalr.app/" />
    <meta property="og:title" content="Rivalr — Find Players &amp; Join Activity Groups" />
    <meta property="og:description" content="Connect with people for football, basketball, padel, concerts, dinners, hikes and more. Post open spots or join an existing group — no group chat needed." />
    <meta property="og:image" content="https://www.rivalr.app/og-image.png" />
    <meta property="og:image:width" content="1200" />
    <meta property="og:image:height" content="630" />
    <meta property="og:image:alt" content="Rivalr — Find Players &amp; Join Activity Groups" />
    <meta property="og:site_name" content="Rivalr" />

    <!-- Twitter Card -->
    <meta name="twitter:card" content="summary_large_image" />
    <meta name="twitter:title" content="Rivalr — Find Players &amp; Join Activity Groups" />
    <meta name="twitter:description" content="Connect with people for football, basketball, padel, concerts, dinners, hikes and more. Post open spots or join an existing group — no group chat needed." />
    <meta name="twitter:image" content="https://www.rivalr.app/og-image.png" />

    <title>Rivalr — Find Players &amp; Join Activity Groups Near You</title>

    <!-- Schema.org JSON-LD -->
    <script type="application/ld+json">
    {
      "@context": "https://schema.org",
      "@graph": [
        {
          "@type": "Organization",
          "@id": "https://www.rivalr.app/#organization",
          "name": "Rivalr",
          "url": "https://www.rivalr.app/",
          "logo": {
            "@type": "ImageObject",
            "url": "https://www.rivalr.app/og-image.png",
            "width": 1200,
            "height": 630
          },
          "description": "Rivalr connects people for shared activities — football, basketball, padel, concerts, dinners, hikes and more."
        },
        {
          "@type": "WebSite",
          "@id": "https://www.rivalr.app/#website",
          "url": "https://www.rivalr.app/",
          "name": "Rivalr",
          "publisher": { "@id": "https://www.rivalr.app/#organization" }
        },
        {
          "@type": "SoftwareApplication",
          "@id": "https://www.rivalr.app/#app",
          "name": "Rivalr",
          "operatingSystem": ["iOS", "Android"],
          "applicationCategory": "SocialNetworkingApplication",
          "description": "Find players for football, basketball, padel and tennis. Join concerts, dinners and hikes. Post open spots or claim one — free, no group chat needed.",
          "offers": {
            "@type": "Offer",
            "price": "0",
            "priceCurrency": "EUR"
          },
          "url": "https://www.rivalr.app/"
        }
      ]
    }
    </script>

    <link rel="preconnect" href="https://fonts.googleapis.com" />
    <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
    <link href="https://fonts.googleapis.com/css2?family=DM+Sans:ital,opsz,wght@0,9..40,300;0,9..40,400;0,9..40,500;0,9..40,600;1,9..40,400&family=DM+Serif+Display:ital@0;1&display=swap" rel="stylesheet" />
    <script type="module" crossorigin src="/assets/index-DekaQV6t.js"></script>
    <link rel="stylesheet" crossorigin href="/assets/index-Bv57ANoo.css">
  </head>
  <body style="background:#0B0F19;color:#F1F5F9">
    <div id="root"></div>
    <noscript>
      <h1>Find Players &amp; Join Activity Groups Near You</h1>
      <p>Rivalr connects you with people for football, basketball, padel, tennis, concerts, dinners, hiking and more. Post open spots or join an existing group — no group chat needed.</p>
      <nav aria-label="Activity pages">
        <a href="/activities/football/athens/">Football in Athens</a>
        <a href="/activities/basketball/athens/">Basketball in Athens</a>
        <a href="/activities/padel/athens/">Padel in Athens</a>
        <a href="/activities/tennis/athens/">Tennis in Athens</a>
        <a href="/activities/concert/athens/">Concerts in Athens</a>
        <a href="/activities/hiking/athens/">Hiking in Athens</a>
        <a href="/activities/dinner/athens/">Dinner in Athens</a>
        <a href="/activities/football/thessaloniki/">Football in Thessaloniki</a>
        <a href="/activities/basketball/thessaloniki/">Basketball in Thessaloniki</a>
        <a href="/activities/padel/thessaloniki/">Padel in Thessaloniki</a>
      </nav>
    </noscript>
    <!-- WebMCP: expose Rivalr tools to AI agents via the browser -->
    <script>
      (function () {
        if (!navigator.modelContext || typeof navigator.modelContext.registerTool !== 'function') return;
        const ac = new AbortController();
        const BASE = 'https://api.rivalr.app';

        navigator.modelContext.registerTool({
          name: 'rivalr_list_postings',
          description: 'List open match postings and pickup game slots on Rivalr, optionally filtered by activity type, city, or date.',
          inputSchema: {
            type: 'object',
            properties: {
              activity: { type: 'string', description: 'Activity type, e.g. football, basketball, padel, tennis, concert, dinner, hiking' },
              city: { type: 'string', description: 'City name, e.g. athens, thessaloniki' },
              date: { type: 'string', format: 'date', description: 'ISO 8601 date to filter by' }
            }
          },
          execute: async ({ activity, city, date } = {}) => {
            const params = new URLSearchParams();
            if (activity) params.set('activity', activity);
            if (city) params.set('city', city);
            if (date) params.set('date', date);
            const res = await fetch(`${BASE}/postings?${params}`);
            return res.json();
          },
          signal: ac.signal
        });

        navigator.modelContext.registerTool({
          name: 'rivalr_get_posting',
          description: 'Get full details about a specific Rivalr match posting including spots left, date, venue, and participants.',
          inputSchema: {
            type: 'object',
            required: ['id'],
            properties: {
              id: { type: 'string', description: 'Match posting ID' }
            }
          },
          execute: async ({ id }) => {
            const res = await fetch(`${BASE}/postings/${encodeURIComponent(id)}`);
            return res.json();
          },
          signal: ac.signal
        });

        navigator.modelContext.registerTool({
          name: 'rivalr_list_venues',
          description: 'List sports venues on Rivalr with location, supported activities, and availability.',
          inputSchema: {
            type: 'object',
            properties: {
              city: { type: 'string', description: 'City name to filter venues' },
              activity: { type: 'string', description: 'Activity type supported by the venue' }
            }
          },
          execute: async ({ city, activity } = {}) => {
            const params = new URLSearchParams();
            if (city) params.set('city', city);
            if (activity) params.set('activity', activity);
            const res = await fetch(`${BASE}/venues?${params}`);
            return res.json();
          },
          signal: ac.signal
        });

        window.addEventListener('unload', () => ac.abort(), { once: true });
      })();
    </script>
  </body>
</html>

시맨틱 HTML

article 또는 main 요소 사용 (0/100)

Missing <article> and <main> elements

올바른 제목 계층 구조 (0/100)

No headings found

시맨틱 HTML 요소 사용 (0/100)

0 semantic elements, 1 divs (ratio: 0%)

의미 있는 이미지 alt 속성 (100/100)

No images found

낮은 div 중첩 깊이 (100/100)

Avg div depth: 0.0, max: 0

콘텐츠 효율성

양호한 토큰 감소율 (100/100)

100% token reduction (HTML→Markdown)

양호한 콘텐츠 대 잡음 비율 (100/100)

Content ratio: 43.1% (3395 content chars / 7876 HTML bytes)

최소한의 인라인 스타일 (50/100)

1/30 elements with inline styles (3.3%)

적절한 페이지 무게 (100/100)

HTML size: 8KB

AI 발견 가능성

llms.txt 파일 있음 (50/100)

llms.txt exists but appears empty or invalid

robots.txt 파일 있음 (100/100)

robots.txt exists

robots.txt가 AI 봇 허용 (100/100)

All major AI bots allowed

sitemap.xml 있음 (0/100)

No sitemap found

Markdown for Agents 지원 (40/100) Application
&#10003; Accept: text/markdown &#10007; .md URL &#10007; <link> tag &#10007; Link header YAML frontmatter (enriched)
Content-Signal 있음 (robots.txt 또는 HTTP 헤더) (80/100)
&#10003; robots.txt &#10007; HTTP header &#10003; Policy

구조화 데이터

Schema.org / JSON-LD 있음 (100/100)

JSON-LD found: Organization, WebSite, SoftwareApplication

Open Graph 태그 있음 (100/100)

All OG tags present

메타 설명 있음 (100/100)

Meta description: 149 chars

정규 URL 있음 (0/100)

No canonical URL

lang 속성 있음 (100/100)

lang="en"

접근성

JavaScript 없이 콘텐츠 이용 가능 (100/100)

Content available without JavaScript

적절한 페이지 크기 (100/100)

Page size: 8KB

HTML에서 콘텐츠가 빠른 위치에 배치 (50/100)

Main content starts at 49% of HTML

{
  "url": "https://www.rivalr.app/",
  "timestamp": 1778798510409,
  "fetch": {
    "mode": "simple",
    "timeMs": 775,
    "htmlSizeBytes": 7876,
    "supportsMarkdown": true,
    "markdownAgents": {
      "contentNegotiation": true,
      "mdUrl": {
        "found": false,
        "url": null
      },
      "linkTag": {
        "found": false,
        "url": null
      },
      "linkHeader": {
        "found": false,
        "url": null
      },
      "responseHeaders": {
        "contentSignal": "ai-train=yes, search=yes, ai-input=yes",
        "xMarkdownTokens": "486",
        "vary": "Accept"
      },
      "frontmatter": {
        "present": true,
        "fields": [
          "title",
          "description",
          "image"
        ],
        "level": "enriched"
      },
      "level": "application"
    },
    "statusCode": 200
  },
  "extraction": {
    "title": "Rivalr — Find Players & Join Activity Groups Near You",
    "excerpt": "Find Players & Join Activity Groups Near You\n      Rivalr connects you with people for football, basketball, padel, tennis, concerts, dinners, hiking and more. Post open spots or join an existing grou",
    "byline": null,
    "siteName": null,
    "lang": "en",
    "contentLength": 3395,
    "metadata": {
      "description": "Find players for football, basketball, padel and tennis. Join concerts, dinners and hikes. Post open spots or claim one — free, no group chat needed.",
      "ogTitle": "Rivalr — Find Players & Join Activity Groups",
      "ogDescription": "Connect with people for football, basketball, padel, concerts, dinners, hikes and more. Post open spots or join an existing group — no group chat needed.",
      "ogImage": "https://www.rivalr.app/og-image.png",
      "ogType": "website",
      "canonical": null,
      "lang": "en",
      "schemas": [
        {
          "@type": "Organization",
          "@id": "https://www.rivalr.app/#organization",
          "name": "Rivalr",
          "url": "https://www.rivalr.app/",
          "logo": {
            "@type": "ImageObject",
            "url": "https://www.rivalr.app/og-image.png",
            "width": 1200,
            "height": 630
          },
          "description": "Rivalr connects people for shared activities — football, basketball, padel, concerts, dinners, hikes and more."
        },
        {
          "@type": "WebSite",
          "@id": "https://www.rivalr.app/#website",
          "url": "https://www.rivalr.app/",
          "name": "Rivalr",
          "publisher": {
            "@id": "https://www.rivalr.app/#organization"
          }
        },
        {
          "@type": "SoftwareApplication",
          "@id": "https://www.rivalr.app/#app",
          "name": "Rivalr",
          "operatingSystem": [
            "iOS",
            "Android"
          ],
          "applicationCategory": "SocialNetworkingApplication",
          "description": "Find players for football, basketball, padel and tennis. Join concerts, dinners and hikes. Post open spots or claim one — free, no group chat needed.",
          "offers": {
            "@type": "Offer",
            "price": "0",
            "priceCurrency": "EUR"
          },
          "url": "https://www.rivalr.app/"
        }
      ],
      "robotsMeta": null,
      "author": null,
      "generator": null,
      "markdownAlternateHref": null
    }
  },
  "markdown": "\n",
  "fullPageMarkdown": "Rivalr — Find Players & Join Activity Groups Near You\n",
  "markdownStats": {
    "images": 0,
    "links": 10,
    "tables": 0,
    "codeBlocks": 0,
    "headings": 1
  },
  "tokens": {
    "htmlTokens": 1936,
    "markdownTokens": 1,
    "reduction": 1935,
    "reductionPercent": 100
  },
  "score": {
    "score": 71,
    "grade": "C",
    "dimensions": {
      "semanticHtml": {
        "score": 35,
        "weight": 20,
        "grade": "F",
        "checks": {
          "uses_article_or_main": {
            "score": 0,
            "weight": 20,
            "details": "Missing <article> and <main> elements"
          },
          "proper_heading_hierarchy": {
            "score": 0,
            "weight": 25,
            "details": "No headings found"
          },
          "semantic_elements": {
            "score": 0,
            "weight": 20,
            "details": "0 semantic elements, 1 divs (ratio: 0%)"
          },
          "meaningful_alt_texts": {
            "score": 100,
            "weight": 15,
            "details": "No images found"
          },
          "low_div_nesting": {
            "score": 100,
            "weight": 20,
            "details": "Avg div depth: 0.0, max: 0"
          }
        }
      },
      "contentEfficiency": {
        "score": 93,
        "weight": 25,
        "grade": "A",
        "checks": {
          "token_reduction_ratio": {
            "score": 100,
            "weight": 40,
            "details": "100% token reduction (HTML→Markdown)"
          },
          "content_to_noise_ratio": {
            "score": 100,
            "weight": 30,
            "details": "Content ratio: 43.1% (3395 content chars / 7876 HTML bytes)"
          },
          "minimal_inline_styles": {
            "score": 50,
            "weight": 15,
            "details": "1/30 elements with inline styles (3.3%)"
          },
          "reasonable_page_weight": {
            "score": 100,
            "weight": 15,
            "details": "HTML size: 8KB"
          }
        }
      },
      "aiDiscoverability": {
        "score": 61,
        "weight": 25,
        "grade": "C",
        "checks": {
          "has_llms_txt": {
            "score": 50,
            "weight": 20,
            "details": "llms.txt exists but appears empty or invalid"
          },
          "has_robots_txt": {
            "score": 100,
            "weight": 10,
            "details": "robots.txt exists"
          },
          "robots_allows_ai_bots": {
            "score": 100,
            "weight": 15,
            "details": "All major AI bots allowed"
          },
          "has_sitemap": {
            "score": 0,
            "weight": 10,
            "details": "No sitemap found"
          },
          "supports_markdown_negotiation": {
            "score": 40,
            "weight": 25,
            "details": "Application level — Content negotiation"
          },
          "has_content_signals": {
            "score": 80,
            "weight": 20,
            "details": "robots.txt: ai-train=no, search=yes, ai-input=no | Policy included"
          }
        }
      },
      "structuredData": {
        "score": 85,
        "weight": 15,
        "grade": "B",
        "checks": {
          "has_schema_org": {
            "score": 100,
            "weight": 30,
            "details": "JSON-LD found: Organization, WebSite, SoftwareApplication"
          },
          "has_open_graph": {
            "score": 100,
            "weight": 25,
            "details": "All OG tags present"
          },
          "has_meta_description": {
            "score": 100,
            "weight": 20,
            "details": "Meta description: 149 chars"
          },
          "has_canonical_url": {
            "score": 0,
            "weight": 15,
            "details": "No canonical URL"
          },
          "has_lang_attribute": {
            "score": 100,
            "weight": 10,
            "details": "lang=\"en\""
          }
        }
      },
      "accessibility": {
        "score": 85,
        "weight": 15,
        "grade": "B",
        "checks": {
          "content_without_js": {
            "score": 100,
            "weight": 40,
            "details": "Content available without JavaScript"
          },
          "reasonable_page_size": {
            "score": 100,
            "weight": 30,
            "details": "Page size: 8KB"
          },
          "fast_content_position": {
            "score": 50,
            "weight": 30,
            "details": "Main content starts at 49% of HTML"
          }
        }
      }
    }
  },
  "recommendations": [
    {
      "id": "add_article_main",
      "priority": "critical",
      "category": "semanticHtml",
      "titleKey": "rec.add_article_main.title",
      "descriptionKey": "rec.add_article_main.description",
      "howToKey": "rec.add_article_main.howto",
      "effort": "quick-win",
      "estimatedImpact": 8,
      "checkScore": 0,
      "checkDetails": "Missing <article> and <main> elements"
    },
    {
      "id": "fix_heading_hierarchy",
      "priority": "critical",
      "category": "semanticHtml",
      "titleKey": "rec.fix_heading_hierarchy.title",
      "descriptionKey": "rec.fix_heading_hierarchy.description",
      "howToKey": "rec.fix_heading_hierarchy.howto",
      "effort": "quick-win",
      "estimatedImpact": 6,
      "checkScore": 0,
      "checkDetails": "No headings found"
    },
    {
      "id": "add_sitemap",
      "priority": "critical",
      "category": "aiDiscoverability",
      "titleKey": "rec.add_sitemap.title",
      "descriptionKey": "rec.add_sitemap.description",
      "howToKey": "rec.add_sitemap.howto",
      "effort": "quick-win",
      "estimatedImpact": 5,
      "checkScore": 0,
      "checkDetails": "No sitemap found"
    },
    {
      "id": "add_semantic_elements",
      "priority": "critical",
      "category": "semanticHtml",
      "titleKey": "rec.add_semantic_elements.title",
      "descriptionKey": "rec.add_semantic_elements.description",
      "howToKey": "rec.add_semantic_elements.howto",
      "effort": "moderate",
      "estimatedImpact": 5,
      "checkScore": 0,
      "checkDetails": "0 semantic elements, 1 divs (ratio: 0%)"
    },
    {
      "id": "add_markdown_negotiation",
      "priority": "high",
      "category": "aiDiscoverability",
      "titleKey": "rec.add_markdown_negotiation.title",
      "descriptionKey": "rec.add_markdown_negotiation.description",
      "howToKey": "rec.add_markdown_negotiation.howto",
      "effort": "significant",
      "estimatedImpact": 6,
      "checkScore": 40,
      "checkDetails": "Application level — Content negotiation"
    },
    {
      "id": "add_canonical_url",
      "priority": "high",
      "category": "structuredData",
      "titleKey": "rec.add_canonical_url.title",
      "descriptionKey": "rec.add_canonical_url.description",
      "howToKey": "rec.add_canonical_url.howto",
      "effort": "quick-win",
      "estimatedImpact": 3,
      "checkScore": 0,
      "checkDetails": "No canonical URL"
    },
    {
      "id": "remove_inline_styles",
      "priority": "medium",
      "category": "contentEfficiency",
      "titleKey": "rec.remove_inline_styles.title",
      "descriptionKey": "rec.remove_inline_styles.description",
      "howToKey": "rec.remove_inline_styles.howto",
      "effort": "moderate",
      "estimatedImpact": 3,
      "checkScore": 50,
      "checkDetails": "1/30 elements with inline styles (3.3%)"
    }
  ],
  "llmsTxtPreview": "# rivalr.app\n\n> Find players for football, basketball, padel and tennis. Join concerts, dinners and hikes. Post open spots or claim one — free, no group chat needed.\n\n## Main\n- [Rivalr — Find Players & Join Activity Groups Near You](https://www.rivalr.app/): Find players for football, basketball, padel and tennis. Join concerts, dinners and hikes. Post open spots or claim one…\n\n",
  "llmsTxtExisting": "<!doctype html>\n<html lang=\"en\">\n  <head>\n    <meta charset=\"UTF-8\" />\n    <link rel=\"icon\" type=\"image/svg+xml\" href=\"/favicon.svg\" />\n    <meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\" />\n    <meta name=\"description\" content=\"Find players for football, basketball, padel and tennis. Join concerts, dinners and hikes. Post open spots or claim one — free, no group chat needed.\" />\n\n    <!-- Open Graph -->\n    <meta property=\"og:type\" content=\"website\" />\n    <meta property=\"og:url\" content=\"https://www.rivalr.app/\" />\n    <meta property=\"og:title\" content=\"Rivalr — Find Players &amp; Join Activity Groups\" />\n    <meta property=\"og:description\" content=\"Connect with people for football, basketball, padel, concerts, dinners, hikes and more. Post open spots or join an existing group — no group chat needed.\" />\n    <meta property=\"og:image\" content=\"https://www.rivalr.app/og-image.png\" />\n    <meta property=\"og:image:width\" content=\"1200\" />\n    <meta property=\"og:image:height\" content=\"630\" />\n    <meta property=\"og:image:alt\" content=\"Rivalr — Find Players &amp; Join Activity Groups\" />\n    <meta property=\"og:site_name\" content=\"Rivalr\" />\n\n    <!-- Twitter Card -->\n    <meta name=\"twitter:card\" content=\"summary_large_image\" />\n    <meta name=\"twitter:title\" content=\"Rivalr — Find Players &amp; Join Activity Groups\" />\n    <meta name=\"twitter:description\" content=\"Connect with people for football, basketball, padel, concerts, dinners, hikes and more. Post open spots or join an existing group — no group chat needed.\" />\n    <meta name=\"twitter:image\" content=\"https://www.rivalr.app/og-image.png\" />\n\n    <title>Rivalr — Find Players &amp; Join Activity Groups Near You</title>\n\n    <!-- Schema.org JSON-LD -->\n    <script type=\"application/ld+json\">\n    {\n      \"@context\": \"https://schema.org\",\n      \"@graph\": [\n        {\n          \"@type\": \"Organization\",\n          \"@id\": \"https://www.rivalr.app/#organization\",\n          \"name\": \"Rivalr\",\n          \"url\": \"https://www.rivalr.app/\",\n          \"logo\": {\n            \"@type\": \"ImageObject\",\n            \"url\": \"https://www.rivalr.app/og-image.png\",\n            \"width\": 1200,\n            \"height\": 630\n          },\n          \"description\": \"Rivalr connects people for shared activities — football, basketball, padel, concerts, dinners, hikes and more.\"\n        },\n        {\n          \"@type\": \"WebSite\",\n          \"@id\": \"https://www.rivalr.app/#website\",\n          \"url\": \"https://www.rivalr.app/\",\n          \"name\": \"Rivalr\",\n          \"publisher\": { \"@id\": \"https://www.rivalr.app/#organization\" }\n        },\n        {\n          \"@type\": \"SoftwareApplication\",\n          \"@id\": \"https://www.rivalr.app/#app\",\n          \"name\": \"Rivalr\",\n          \"operatingSystem\": [\"iOS\", \"Android\"],\n          \"applicationCategory\": \"SocialNetworkingApplication\",\n          \"description\": \"Find players for football, basketball, padel and tennis. Join concerts, dinners and hikes. Post open spots or claim one — free, no group chat needed.\",\n          \"offers\": {\n            \"@type\": \"Offer\",\n            \"price\": \"0\",\n            \"priceCurrency\": \"EUR\"\n          },\n          \"url\": \"https://www.rivalr.app/\"\n        }\n      ]\n    }\n    </script>\n\n    <link rel=\"preconnect\" href=\"https://fonts.googleapis.com\" />\n    <link rel=\"preconnect\" href=\"https://fonts.gstatic.com\" crossorigin />\n    <link href=\"https://fonts.googleapis.com/css2?family=DM+Sans:ital,opsz,wght@0,9..40,300;0,9..40,400;0,9..40,500;0,9..40,600;1,9..40,400&family=DM+Serif+Display:ital@0;1&display=swap\" rel=\"stylesheet\" />\n    <script type=\"module\" crossorigin src=\"/assets/index-DekaQV6t.js\"></script>\n    <link rel=\"stylesheet\" crossorigin href=\"/assets/index-Bv57ANoo.css\">\n  </head>\n  <body style=\"background:#0B0F19;color:#F1F5F9\">\n    <div id=\"root\"></div>\n    <noscript>\n      <h1>Find Players &amp; Join Activity Groups Near You</h1>\n      <p>Rivalr connects you with people for football, basketball, padel, tennis, concerts, dinners, hiking and more. Post open spots or join an existing group — no group chat needed.</p>\n      <nav aria-label=\"Activity pages\">\n        <a href=\"/activities/football/athens/\">Football in Athens</a>\n        <a href=\"/activities/basketball/athens/\">Basketball in Athens</a>\n        <a href=\"/activities/padel/athens/\">Padel in Athens</a>\n        <a href=\"/activities/tennis/athens/\">Tennis in Athens</a>\n        <a href=\"/activities/concert/athens/\">Concerts in Athens</a>\n        <a href=\"/activities/hiking/athens/\">Hiking in Athens</a>\n        <a href=\"/activities/dinner/athens/\">Dinner in Athens</a>\n        <a href=\"/activities/football/thessaloniki/\">Football in Thessaloniki</a>\n        <a href=\"/activities/basketball/thessaloniki/\">Basketball in Thessaloniki</a>\n        <a href=\"/activities/padel/thessaloniki/\">Padel in Thessaloniki</a>\n      </nav>\n    </noscript>\n    <!-- WebMCP: expose Rivalr tools to AI agents via the browser -->\n    <script>\n      (function () {\n        if (!navigator.modelContext || typeof navigator.modelContext.registerTool !== 'function') return;\n        const ac = new AbortController();\n        const BASE = 'https://api.rivalr.app';\n\n        navigator.modelContext.registerTool({\n          name: 'rivalr_list_postings',\n          description: 'List open match postings and pickup game slots on Rivalr, optionally filtered by activity type, city, or date.',\n          inputSchema: {\n            type: 'object',\n            properties: {\n              activity: { type: 'string', description: 'Activity type, e.g. football, basketball, padel, tennis, concert, dinner, hiking' },\n              city: { type: 'string', description: 'City name, e.g. athens, thessaloniki' },\n              date: { type: 'string', format: 'date', description: 'ISO 8601 date to filter by' }\n            }\n          },\n          execute: async ({ activity, city, date } = {}) => {\n            const params = new URLSearchParams();\n            if (activity) params.set('activity', activity);\n            if (city) params.set('city', city);\n            if (date) params.set('date', date);\n            const res = await fetch(`${BASE}/postings?${params}`);\n            return res.json();\n          },\n          signal: ac.signal\n        });\n\n        navigator.modelContext.registerTool({\n          name: 'rivalr_get_posting',\n          description: 'Get full details about a specific Rivalr match posting including spots left, date, venue, and participants.',\n          inputSchema: {\n            type: 'object',\n            required: ['id'],\n            properties: {\n              id: { type: 'string', description: 'Match posting ID' }\n            }\n          },\n          execute: async ({ id }) => {\n            const res = await fetch(`${BASE}/postings/${encodeURIComponent(id)}`);\n            return res.json();\n          },\n          signal: ac.signal\n        });\n\n        navigator.modelContext.registerTool({\n          name: 'rivalr_list_venues',\n          description: 'List sports venues on Rivalr with location, supported activities, and availability.',\n          inputSchema: {\n            type: 'object',\n            properties: {\n              city: { type: 'string', description: 'City name to filter venues' },\n              activity: { type: 'string', description: 'Activity type supported by the venue' }\n            }\n          },\n          execute: async ({ city, activity } = {}) => {\n            const params = new URLSearchParams();\n            if (city) params.set('city', city);\n            if (activity) params.set('activity', activity);\n            const res = await fetch(`${BASE}/venues?${params}`);\n            return res.json();\n          },\n          signal: ac.signal\n        });\n\n        window.addEventListener('unload', () => ac.abort(), { once: true });\n      })();\n    </script>\n  </body>\n</html>",
  "emergingProtocols": {
    "oauthDiscovery": {
      "exists": false,
      "url": "https://www.rivalr.app/.well-known/oauth-authorization-server"
    },
    "mcpServerCard": {
      "exists": false,
      "url": "https://www.rivalr.app/.well-known/mcp.json"
    },
    "a2aAgentCard": {
      "exists": false,
      "url": "https://www.rivalr.app/.well-known/agent.json"
    },
    "count": 0
  },
  "snippets": [
    {
      "id": "fix_heading_hierarchy",
      "title": "Fix heading hierarchy",
      "description": "Your page has no <h1>. Every page needs exactly one <h1> as the main heading. Add it inside your <main> or <article>.",
      "language": "html",
      "code": "<h1>Rivalr — Find Players & Join Activity Groups Near You</h1>",
      "filename": "<main> or <article>"
    },
    {
      "id": "add_canonical_url",
      "title": "Add canonical URL",
      "description": "The canonical URL tells AI agents which version of the page is the \"official\" one, avoiding duplicate content issues.",
      "language": "html",
      "code": "<link rel=\"canonical\" href=\"https://www.rivalr.app/\">",
      "filename": "<head>"
    },
    {
      "id": "add_sitemap",
      "title": "Create /sitemap.xml",
      "description": "A sitemap helps AI agents discover all your pages. Most CMS platforms generate one automatically.",
      "language": "xml",
      "code": "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<urlset xmlns=\"http://www.sitemaps.org/schemas/sitemap/0.9\">\n  <url>\n    <loc>https://www.rivalr.app/</loc>\n    <lastmod>2026-05-14</lastmod>\n  </url>\n</urlset>",
      "filename": "/sitemap.xml"
    },
    {
      "id": "add_article_main",
      "title": "Wrap content in <main> and <article>",
      "description": "Semantic HTML landmarks help AI agents identify the main content of your page.",
      "language": "html",
      "code": "<main>\n  <article>\n    <h1>Your Page Title</h1>\n    <p>Your content here...</p>\n  </article>\n</main>",
      "filename": "<body>"
    },
    {
      "id": "add_markdown_negotiation",
      "title": "Support Markdown for Agents",
      "description": "Let AI agents request a clean Markdown version of any page via content negotiation, .md alternate URLs, link tags or Link headers.",
      "language": "html",
      "code": "<!-- Mechanism 3: link tag advertising the .md alternate -->\n<link rel=\"alternate\" type=\"text/markdown\" href=\"/page.md\">",
      "filename": "<head>",
      "stacks": [
        {
          "id": "html",
          "label": "HTML <head>",
          "language": "html",
          "filename": "<head>",
          "code": "<!-- Mechanism 3: link tag advertising the .md alternate -->\n<link rel=\"alternate\" type=\"text/markdown\" href=\"/page.md\">"
        },
        {
          "id": "express",
          "label": "Express",
          "language": "javascript",
          "filename": "server.js",
          "code": "// Mechanisms 1 + 4: content negotiation + Link header\napp.get('/page', (req, res) => {\n  res.setHeader('Vary', 'Accept');\n  res.setHeader('Link', '</page.md>; rel=\"alternate\"; type=\"text/markdown\"');\n  if ((req.headers.accept || '').includes('text/markdown')) {\n    res.type('text/markdown; charset=utf-8');\n    return res.send(renderMarkdown('page'));\n  }\n  res.render('page');\n});"
        },
        {
          "id": "fastify",
          "label": "Fastify",
          "language": "javascript",
          "filename": "server.js",
          "code": "// Mechanisms 1 + 4: content negotiation + Link header\nfastify.get('/page', async (req, reply) => {\n  reply.header('Vary', 'Accept');\n  reply.header('Link', '</page.md>; rel=\"alternate\"; type=\"text/markdown\"');\n  if ((req.headers.accept || '').includes('text/markdown')) {\n    return reply.type('text/markdown; charset=utf-8').send(renderMarkdown('page'));\n  }\n  return reply.view('/page.ejs');\n});"
        },
        {
          "id": "nextjs",
          "label": "Next.js",
          "language": "typescript",
          "filename": "app/page/route.ts",
          "code": "// Next.js App Router — Route Handler returning Markdown\nimport { NextRequest } from 'next/server';\nimport { renderMarkdown } from '@/lib/md';\nexport async function GET(req: NextRequest) {\n  const accept = req.headers.get('accept') || '';\n  if (accept.includes('text/markdown')) {\n    return new Response(await renderMarkdown('page'), {\n      headers: {\n        'Content-Type': 'text/markdown; charset=utf-8',\n        'Vary': 'Accept',\n      },\n    });\n  }\n  // Fall through to the page component\n  return new Response(null, { status: 404 });\n}"
        },
        {
          "id": "wordpress",
          "label": "WordPress",
          "language": "php",
          "filename": "functions.php",
          "code": "<?php\n// Mechanism 1: respond to Accept: text/markdown on the same URL\nadd_action('template_redirect', function () {\n    if (!is_singular()) return;\n    $accept = $_SERVER['HTTP_ACCEPT'] ?? '';\n    if (strpos($accept, 'text/markdown') === false) return;\n    header('Content-Type: text/markdown; charset=utf-8');\n    header('Vary: Accept');\n    $post = get_queried_object();\n    echo \"# \" . get_the_title($post) . \"\\n\\n\";\n    echo wp_strip_all_tags(apply_filters('the_content', $post->post_content));\n    exit;\n});"
        },
        {
          "id": "static",
          "label": "Hugo / Jekyll / Astro",
          "language": "txt",
          "filename": "static/page.md",
          "code": "# Mechanism 2: serve .md alongside .html\n# Hugo: place page.md in /static/ — built unchanged\n# Jekyll: drop page.md in /assets/ — copied as-is\n# Astro: src/pages/page.md.ts that exports a GET returning markdown\n\n# Then advertise with mechanism 3 in <head>:\n#   <link rel=\"alternate\" type=\"text/markdown\" href=\"/page.md\">"
        }
      ]
    }
  ]
}

API를 사용하여 프로그래밍 방식으로 가져올 수 있습니다 (곧 출시)

이 JSON은 내부용입니다 — Markdown 및 llms.txt 파일과 달리 사이트에 업로드하기 위한 것이 아닙니다. 시간에 따른 점수 추적을 위한 기준값으로 저장하거나, 개발팀과 공유하거나, CI/CD 파이프라인에 통합하세요.

결과 공유

Twitter LinkedIn

배지 삽입

이 배지를 사이트에 추가하세요. AI 준비도 점수가 변경되면 자동으로 업데이트됩니다.

AgentReady.md score for www.rivalr.app
Script 권장
<script src="https://agentready.md/badge.js" data-id="869e97d4-81ec-4e34-948b-49149a42d94e" data-domain="www.rivalr.app"></script>
Markdown
[![AgentReady.md score for www.rivalr.app](https://agentready.md/badge/www.rivalr.app.svg)](https://agentready.md/ko/r/869e97d4-81ec-4e34-948b-49149a42d94e)

곧 출시: 전체 도메인 분석

전체 도메인을 크롤링하고, llms.txt를 생성하고, AI 준비도 점수를 시간에 따라 모니터링하세요. 대기자 명단에 등록하여 알림을 받으세요.

명단에 등록되었습니다! 서비스 출시 시 알려드리겠습니다.