<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: pengspirit</title>
    <description>The latest articles on DEV Community by pengspirit (@incultnitollc).</description>
    <link>https://dev.to/incultnitollc</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F3914325%2Fb37d323f-e828-4db9-a08f-3eb6b60fbaaf.png</url>
      <title>DEV Community: pengspirit</title>
      <link>https://dev.to/incultnitollc</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/incultnitollc"/>
    <language>en</language>
    <item>
      <title>Schema descriptions are load-bearing: why missing parameter descriptions break MCP clients</title>
      <dc:creator>pengspirit</dc:creator>
      <pubDate>Tue, 05 May 2026 16:16:49 +0000</pubDate>
      <link>https://dev.to/incultnitollc/schema-descriptions-are-load-bearing-why-missing-parameter-descriptions-break-mcp-clients-4l42</link>
      <guid>https://dev.to/incultnitollc/schema-descriptions-are-load-bearing-why-missing-parameter-descriptions-break-mcp-clients-4l42</guid>
      <description>&lt;p&gt;I shipped &lt;a href="https://www.npmjs.com/package/@incultnitollc/mcp-probe" rel="noopener noreferrer"&gt;&lt;code&gt;mcp-probe&lt;/code&gt;&lt;/a&gt; — a CLI that points at any MCP server, enumerates every tool, resource, and prompt, calls each with auto-generated arguments, validates against declared schemas, prints a pass/fail scorecard, and exits 0/1 for CI.&lt;/p&gt;

&lt;p&gt;The plan for launch week: run it against the official Node MCP servers and post results. The first run made me look like I'd broken half the ecosystem. The second, after I read my own output, told a different story — most failures were bugs in my client, not the servers. The rest collapsed into one finding about schema design.&lt;/p&gt;

&lt;p&gt;This post is the corrected version. Three sections: what mcp-probe does, what the scorecards say, and the three bugs I fixed in my own client first.&lt;/p&gt;

&lt;h2&gt;
  
  
  1. What mcp-probe does
&lt;/h2&gt;

&lt;p&gt;One command. stdio, SSE, or Streamable HTTP transport. No config file required.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx @incultnitollc/mcp-probe &lt;span class="nb"&gt;test&lt;/span&gt; &lt;span class="s2"&gt;"npx -y @modelcontextprotocol/server-memory"&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Output is a scorecard:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;Tools callable:      9/9
Resources readable:  n/a
Prompts callable:    n/a
Schema warnings:     4
ALL CHECKS PASSED
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Exit code 0 if everything passes, 1 if anything fails. Drop it in CI:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;run&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;npx -y @incultnitollc/mcp-probe test "node dist/index.js"&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Install globally if you'd rather not &lt;code&gt;npx&lt;/code&gt; every time:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npm &lt;span class="nb"&gt;install&lt;/span&gt; &lt;span class="nt"&gt;-g&lt;/span&gt; @incultnitollc/mcp-probe
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;The mental model is &lt;code&gt;curl&lt;/code&gt; for MCP servers. You don't open Claude Desktop, hand-write a config, restart the app, and stare at the tool list to see whether anything broke. You run one command and get a scorecard.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fuuvitnyao76ow5kklqn2.gif" class="article-body-image-wrapper"&gt;&lt;img src="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fuuvitnyao76ow5kklqn2.gif" alt="mcp-probe demo" width="" height=""&gt;&lt;/a&gt;&lt;/p&gt;

&lt;h2&gt;
  
  
  2. What I found across the four official Node servers
&lt;/h2&gt;

&lt;p&gt;Here is the actual scorecard from &lt;code&gt;docs/scorecards/SUMMARY.md&lt;/code&gt;, re-run on &lt;code&gt;@incultnitollc/mcp-probe@1.0.1&lt;/code&gt;:&lt;/p&gt;

&lt;div class="table-wrapper-paragraph"&gt;&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Server&lt;/th&gt;
&lt;th&gt;Tools&lt;/th&gt;
&lt;th&gt;Resources&lt;/th&gt;
&lt;th&gt;Prompts&lt;/th&gt;
&lt;th&gt;Schema warns&lt;/th&gt;
&lt;th&gt;Status&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;@modelcontextprotocol/server-memory&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;9 / 9&lt;/td&gt;
&lt;td&gt;n/a&lt;/td&gt;
&lt;td&gt;n/a&lt;/td&gt;
&lt;td&gt;4&lt;/td&gt;
&lt;td&gt;PASS&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;@modelcontextprotocol/server-sequential-thinking&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;1 / 1&lt;/td&gt;
&lt;td&gt;n/a&lt;/td&gt;
&lt;td&gt;n/a&lt;/td&gt;
&lt;td&gt;0&lt;/td&gt;
&lt;td&gt;PASS&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;@modelcontextprotocol/server-everything&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;12 / 13&lt;/td&gt;
&lt;td&gt;7 / 7&lt;/td&gt;
&lt;td&gt;3 / 4&lt;/td&gt;
&lt;td&gt;1&lt;/td&gt;
&lt;td&gt;partial&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;@modelcontextprotocol/server-filesystem&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;8 / 14&lt;/td&gt;
&lt;td&gt;n/a&lt;/td&gt;
&lt;td&gt;n/a&lt;/td&gt;
&lt;td&gt;18&lt;/td&gt;
&lt;td&gt;partial&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;&lt;/div&gt;

&lt;p&gt;Aggregate: 30 of 37 tools callable across four servers, 81%. Two servers fully pass. The other two have a single failure pattern between them.&lt;/p&gt;

&lt;p&gt;A scope note before the finding, because I got this wrong the first time: Anthropic's &lt;code&gt;fetch&lt;/code&gt; MCP server is Python-only, installed via &lt;code&gt;uvx mcp-server-fetch&lt;/code&gt;. It has never been published to npm. mcp-probe runs against any stdio MCP server regardless of language — only this scorecard is scoped to the official Node servers. Earlier launch copy of mine that called &lt;code&gt;server-fetch&lt;/code&gt; "broken on npm" was wrong, and I want to flag it explicitly here because I almost shipped that draft.&lt;/p&gt;

&lt;p&gt;Now the real finding. Every remaining failure on the partial-pass servers traces to the same root cause: &lt;strong&gt;missing &lt;code&gt;description&lt;/code&gt; fields on schema properties&lt;/strong&gt;.&lt;/p&gt;

&lt;p&gt;On &lt;code&gt;server-filesystem&lt;/code&gt;, six of the fourteen tools fail because mcp-probe doesn't know which arguments are supposed to be file paths versus directory paths versus arbitrary strings. The &lt;code&gt;path&lt;/code&gt; parameter on &lt;code&gt;read_file&lt;/code&gt;, &lt;code&gt;read_text_file&lt;/code&gt;, &lt;code&gt;read_media_file&lt;/code&gt;, &lt;code&gt;edit_file&lt;/code&gt;, and &lt;code&gt;write_file&lt;/code&gt; has no description in the schema, so my client defaults to the allowed sandbox directory itself. The server correctly returns &lt;code&gt;EISDIR&lt;/code&gt; (you tried to read a directory as a file) or &lt;code&gt;EACCES&lt;/code&gt; (you tried to write to one). &lt;code&gt;move_file&lt;/code&gt; fails the same way — both &lt;code&gt;source&lt;/code&gt; and &lt;code&gt;destination&lt;/code&gt; resolve to the same directory, and the server correctly refuses the no-op rename. The server is doing its job. The schema is the gap.&lt;/p&gt;

&lt;p&gt;On &lt;code&gt;server-everything&lt;/code&gt;, one prompt fails because the &lt;code&gt;resourceType&lt;/code&gt; argument has no description. It's an enum — &lt;code&gt;"Text"&lt;/code&gt; or &lt;code&gt;"Blob"&lt;/code&gt; — but with no description and no examples, my client passes the literal string &lt;code&gt;"test"&lt;/code&gt; and the server correctly returns &lt;code&gt;Invalid resourceType: test&lt;/code&gt;. The schema validator inside mcp-probe even raises a warning on this property before the call fires:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;WARN  get-resource-reference — Property "resourceType" missing description
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;That warning is the diagnostic working as intended — mcp-probe still attempts the call, then surfaces both the warning and the resulting failure side-by-side so you can see the connection.&lt;/p&gt;

&lt;p&gt;The substantive insight, and the line I'll repeat at every MCP-related event for the next year: &lt;strong&gt;when an MCP server ships parameter properties without descriptions, no automated tool can guess valid arguments.&lt;/strong&gt; Not mcp-probe. Not your IDE's autocomplete. Not an LLM trying to call the tool from Claude Desktop. Schema descriptions aren't documentation polish. They're the instruction manual the model is reading every time it picks an argument. They're load-bearing.&lt;/p&gt;

&lt;p&gt;If you maintain an MCP server and you want a quick win, add &lt;code&gt;"description"&lt;/code&gt; to every property in every input schema. The 18 schema warnings on &lt;code&gt;server-filesystem&lt;/code&gt; are not 18 separate problems — they're 18 instances of the same one-line fix.&lt;/p&gt;

&lt;h2&gt;
  
  
  3. The three bugs I fixed in my own client first
&lt;/h2&gt;

&lt;p&gt;Here's the part I want to be honest about. The first time I ran mcp-probe against &lt;code&gt;server-filesystem&lt;/code&gt;, I got 2 of 14 tools passing and a scorecard that screamed FAIL. My instinct was to write a launch post saying "the official filesystem server is broken." I almost did.&lt;/p&gt;

&lt;p&gt;Then I actually read my own output. Most of those failures were because my client was sending arguments the server had no way to accept. A diagnostic tool is only credible if it can distinguish "your server is broken" from "I sent garbage." Stress-testing forced that distinction, and three commits came out of it before I trusted the scorecard.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Commit &lt;code&gt;3825170&lt;/code&gt; — show the args we sent on every failure.&lt;/strong&gt; When a tool or prompt call fails, mcp-probe now prints the exact JSON it sent alongside the server's error response. Before this, a failure looked like &lt;code&gt;MCP error -32603: Invalid resourceType: test&lt;/code&gt; with no indication that &lt;code&gt;"test"&lt;/code&gt; was something my client had auto-generated. After this, you can read the failure and immediately tell whether the server rejected something reasonable or something nonsense. This is the smallest of the three changes and the most important one for the trust story.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Commit &lt;code&gt;ce4f55e&lt;/code&gt; — sandbox-aware paths.&lt;/strong&gt; &lt;code&gt;server-filesystem&lt;/code&gt; enforces an allowed-directory sandbox. mcp-probe now calls &lt;code&gt;list_allowed_directories&lt;/code&gt; before generating sample arguments and uses one of those directories as the default for any &lt;code&gt;path&lt;/code&gt;-shaped parameter. On macOS, where &lt;code&gt;/tmp&lt;/code&gt; is a symlink to &lt;code&gt;/private/tmp&lt;/code&gt;, it normalizes via &lt;code&gt;realpath&lt;/code&gt; so the path the server receives matches what the sandbox check expects. This single commit moved &lt;code&gt;server-filesystem&lt;/code&gt; from 2 of 14 passing to 8 of 14. The remaining 6 are the missing-description cases I already covered — the bugs that aren't mine.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Prompt-argument enum extractor.&lt;/strong&gt; When a prompt argument is described in prose like &lt;code&gt;"one of: Text, Blob"&lt;/code&gt; instead of as a JSON Schema enum, mcp-probe now tries to parse the allowed values out of the description string and pick one. Partial — it works on the prompts that have prose-level documentation, and it does nothing for arguments like &lt;code&gt;resourceType&lt;/code&gt; on &lt;code&gt;server-everything&lt;/code&gt; that have neither schema enum nor prose description. This is why the schema-description finding above isn't theoretical: I built the workaround, and the workaround can't help when there's no text to read.&lt;/p&gt;

&lt;p&gt;The loop, in one sentence: I had to make my client honest about what it was sending before I could call any server's failure a server bug.&lt;/p&gt;

&lt;h2&gt;
  
  
  Try it
&lt;/h2&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npm &lt;span class="nb"&gt;install&lt;/span&gt; &lt;span class="nt"&gt;-g&lt;/span&gt; @incultnitollc/mcp-probe
mcp-probe &lt;span class="nb"&gt;test&lt;/span&gt; &lt;span class="s2"&gt;"npx -y @modelcontextprotocol/server-memory"&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;ul&gt;
&lt;li&gt;Repo: &lt;a href="https://github.com/incultnitollc/mcp-probe" rel="noopener noreferrer"&gt;github.com/incultnitollc/mcp-probe&lt;/a&gt;
&lt;/li&gt;
&lt;li&gt;npm: &lt;a href="https://www.npmjs.com/package/@incultnitollc/mcp-probe" rel="noopener noreferrer"&gt;@incultnitollc/mcp-probe&lt;/a&gt;
&lt;/li&gt;
&lt;li&gt;Raw scorecards from this post: &lt;a href="https://github.com/incultnitollc/mcp-probe/tree/main/docs/scorecards" rel="noopener noreferrer"&gt;&lt;code&gt;docs/scorecards/&lt;/code&gt;&lt;/a&gt;
&lt;/li&gt;
&lt;li&gt;Pre-publish checklist for MCP server maintainers: &lt;a href="https://github.com/incultnitollc/mcp-probe/blob/main/docs/checklist.md" rel="noopener noreferrer"&gt;&lt;code&gt;docs/checklist.md&lt;/code&gt;&lt;/a&gt;
&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;If you maintain an MCP server and you want a scorecard run against it, open an issue with the &lt;a href="https://github.com/incultnitollc/mcp-probe/issues/new?template=test_my_server.yml" rel="noopener noreferrer"&gt;test-my-server template&lt;/a&gt; and I'll post the results as a comment. If mcp-probe reports something that looks like a server bug and isn't, open an issue against mcp-probe instead — that's the loop that produced commits &lt;code&gt;3825170&lt;/code&gt; and &lt;code&gt;ce4f55e&lt;/code&gt;, and it's the only way the diagnostic gets more trustworthy.&lt;/p&gt;

</description>
      <category>mcp</category>
      <category>claude</category>
      <category>devtools</category>
      <category>testing</category>
    </item>
  </channel>
</rss>
