<?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: Giovanni Sizino Ennes</title>
    <description>The latest articles on DEV Community by Giovanni Sizino Ennes (@goofypluto999).</description>
    <link>https://dev.to/goofypluto999</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%2F3902218%2F0db799ce-30a0-44c2-a280-f024589a3773.jpeg</url>
      <title>DEV Community: Giovanni Sizino Ennes</title>
      <link>https://dev.to/goofypluto999</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/goofypluto999"/>
    <language>en</language>
    <item>
      <title>I built an open-source MCP server that lints a CV against 5 real ATS parsers</title>
      <dc:creator>Giovanni Sizino Ennes</dc:creator>
      <pubDate>Tue, 28 Apr 2026 10:51:57 +0000</pubDate>
      <link>https://dev.to/goofypluto999/i-built-an-open-source-mcp-server-that-lints-a-cv-against-5-real-ats-parsers-pjb</link>
      <guid>https://dev.to/goofypluto999/i-built-an-open-source-mcp-server-that-lints-a-cv-against-5-real-ats-parsers-pjb</guid>
      <description>&lt;p&gt;Most "ATS scanners" online give you a 0-100 score that's invented. After talking to recruiters at three companies on different ATS platforms, I confirmed: there is no real ATS score. Different parsers extract different fields, and the scoring tools just guess.&lt;/p&gt;

&lt;p&gt;So I built a tool that does it differently — and shipped it as an MCP server so any agent (Claude Code, Cursor, Windsurf, Zed, etc.) can use it inline.&lt;/p&gt;

&lt;p&gt;Repo: &lt;a href="https://github.com/goofypluto999/cv-mirror-mcp" rel="noopener noreferrer"&gt;goofypluto999/cv-mirror-mcp&lt;/a&gt;&lt;br&gt;
License: MIT.&lt;/p&gt;
&lt;h2&gt;
  
  
  What it actually does
&lt;/h2&gt;

&lt;p&gt;Three tools exposed over the Model Context Protocol:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;code&gt;analyze_cv(path)&lt;/code&gt; — full report. Takes a CV (PDF or DOCX), returns per-vendor lint findings across &lt;strong&gt;Workday, Greenhouse, Lever, Taleo, iCIMS&lt;/strong&gt; with severity (&lt;code&gt;error&lt;/code&gt; / &lt;code&gt;warn&lt;/code&gt; / &lt;code&gt;info&lt;/code&gt;) and concrete fixes.&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;lint_for_vendor(path, vendor)&lt;/code&gt; — single-vendor lint. Faster when the user only cares about, say, Workday.&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;get_express_url()&lt;/code&gt; — returns the URL of a visual web companion (live in-browser parse view) for users who want the side-by-side overlay.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Sample output:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;[ERROR] Workday
  - WORKDAY_MULTI_COLUMN: 35% of lines look multi-column. Workday's parser
    reads left-to-right and interleaves both columns into one stream.
    Fix: Convert to single-column layout. Move sidebars (Skills, Tools)
    above or below the main content.

[OK] Greenhouse
  No issues detected by the simulated parser.

[WARN] Lever
  - LEVER_HEADER_FOOTER: Header/footer-like text detected ("Page 1 of 2").
    Lever historically drops content placed in PDF headers/footers.
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  Why per-vendor instead of a single score
&lt;/h2&gt;

&lt;p&gt;The five top enterprise ATSes do parsing differently:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Workday&lt;/strong&gt; reads PDFs in document-stream order — multi-column layouts get interleaved&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Greenhouse&lt;/strong&gt; strips most emoji codepoints; "Projects 🚀" can lose its surrounding context&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Lever&lt;/strong&gt; historically drops content in PDF headers/footers&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Taleo&lt;/strong&gt; prefers Month-Year date format; ISO dates often fail to populate employment durations&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;iCIMS&lt;/strong&gt; has the worst multi-column handling — sidebar Skills sections often merge with the line above&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;A single score collapses all of that into noise. Per-vendor surfaces what each one actually sees.&lt;/p&gt;

&lt;h2&gt;
  
  
  Stack
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;code&gt;pdf-parse&lt;/code&gt; for PDF text extraction&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;mammoth&lt;/code&gt; for DOCX&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;@modelcontextprotocol/sdk&lt;/code&gt; for the MCP server transport (stdio)&lt;/li&gt;
&lt;li&gt;Pure-JS vendor simulators in &lt;code&gt;src/lint.mjs&lt;/code&gt; — no AI inference, just heuristics derived from public vendor docs&lt;/li&gt;
&lt;li&gt;19 unit tests covering signal extraction and per-vendor rule firing&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;The vendor rules are intentionally conservative. Every rule cites its public source in &lt;a href="https://github.com/goofypluto999/cv-mirror-mcp/blob/main/docs/vendor-sources.md" rel="noopener noreferrer"&gt;docs/vendor-sources.md&lt;/a&gt;. I'd rather miss an issue than false-positive on a real job page.&lt;/p&gt;

&lt;h2&gt;
  
  
  Install + use
&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; cv-mirror-mcp
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Add to your MCP client config:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight json"&gt;&lt;code&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"mcpServers"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"cv-mirror"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
      &lt;/span&gt;&lt;span class="nl"&gt;"command"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"npx"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
      &lt;/span&gt;&lt;span class="nl"&gt;"args"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;[&lt;/span&gt;&lt;span class="s2"&gt;"-y"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"cv-mirror-mcp"&lt;/span&gt;&lt;span class="p"&gt;]&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt;
&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt;
&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Restart the client. Then ask your agent:&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;Scan my CV at ~/Documents/resume.pdf and tell me what each ATS would do to it.&lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;The agent calls &lt;code&gt;analyze_cv&lt;/code&gt;, the lint engine runs locally inside the agent's process — no upload, no telemetry — and the agent surfaces the per-vendor breakdown.&lt;/p&gt;

&lt;h2&gt;
  
  
  Privacy by architecture
&lt;/h2&gt;

&lt;p&gt;The express path (web tool) runs entirely client-side via pdf.js. The MCP server runs locally inside your agent's process. There is no upload endpoint and no network call from the lint engine. You can verify in the source — it's MIT licensed.&lt;/p&gt;

&lt;p&gt;This isn't a marketing claim. The architecture makes the data exfiltration impossible, not just unlikely.&lt;/p&gt;

&lt;h2&gt;
  
  
  Roadmap
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;[ ] BambooHR simulator&lt;/li&gt;
&lt;li&gt;[ ] SmartRecruiters simulator&lt;/li&gt;
&lt;li&gt;[ ] Real-time JD scraping for major ATS public job boards (so the user can also pass a JD URL and get keyword-match findings)&lt;/li&gt;
&lt;li&gt;[ ] Browser extension that overlays parser output on any job application form&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Sister project (paid, optional)
&lt;/h2&gt;

&lt;p&gt;The same team also makes &lt;a href="https://vantage-livid.vercel.app" rel="noopener noreferrer"&gt;Vantage AI&lt;/a&gt; — once your CV is past the parser, the next problem is the application: tailored cover letter, mock interview, fit analysis. That's a paid SaaS, separate from this MCP server. Genuinely free signup with 3 free analyses included if you want to try it.&lt;/p&gt;

&lt;p&gt;The MCP server stands alone — it's MIT, it's free, it's not a teaser.&lt;/p&gt;

&lt;h2&gt;
  
  
  Contributing
&lt;/h2&gt;

&lt;p&gt;PRs welcome. Particularly interested in:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;More vendor simulators (BambooHR, SmartRecruiters, JazzHR, Recruitee, etc.)&lt;/li&gt;
&lt;li&gt;Updated rules when vendors change their parsing behaviour (open an issue with the source link)&lt;/li&gt;
&lt;li&gt;Translations of the lint output&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;The repo is at &lt;a href="https://github.com/goofypluto999/cv-mirror-mcp" rel="noopener noreferrer"&gt;goofypluto999/cv-mirror-mcp&lt;/a&gt;. Built solo over a couple of weeks.&lt;/p&gt;

</description>
      <category>mcp</category>
      <category>opensource</category>
      <category>career</category>
      <category>node</category>
    </item>
  </channel>
</rss>
