<?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: Leeson Wong</title>
    <description>The latest articles on DEV Community by Leeson Wong (@leeson_wong_fa777ac94d188).</description>
    <link>https://dev.to/leeson_wong_fa777ac94d188</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%2F3907311%2Fcade0c48-1b37-4c4f-8e84-58c1b375b12a.jpg</url>
      <title>DEV Community: Leeson Wong</title>
      <link>https://dev.to/leeson_wong_fa777ac94d188</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/leeson_wong_fa777ac94d188"/>
    <language>en</language>
    <item>
      <title>From PDF to MCP: Let AI Agents Query Your Resume On Demand</title>
      <dc:creator>Leeson Wong</dc:creator>
      <pubDate>Fri, 01 May 2026 09:53:56 +0000</pubDate>
      <link>https://dev.to/leeson_wong_fa777ac94d188/from-pdf-to-mcp-let-ai-agents-query-your-resume-on-demand-530e</link>
      <guid>https://dev.to/leeson_wong_fa777ac94d188/from-pdf-to-mcp-let-ai-agents-query-your-resume-on-demand-530e</guid>
      <description>&lt;h2&gt;
  
  
  The Problem
&lt;/h2&gt;

&lt;p&gt;More and more companies are using AI for resume screening. HR hands the JD to an Agent, and the Agent matches candidates. But resumes are essentially PDFs (or web pages) — hard for AI to parse cleanly. Layout noise, scattered information, no structured querying.&lt;/p&gt;

&lt;p&gt;So I asked myself: why not make my resume &lt;strong&gt;AI-native&lt;/strong&gt;? Not "AI-friendly" marketing fluff, but something an AI Agent can actually connect to, query on demand, and run structured analysis against.&lt;/p&gt;

&lt;p&gt;I built exactly that: a MCP Server for my resume. Visit the public page, and an AI Agent can auto-discover the endpoint, connect, search through experience, analyze job fit, and query skill proficiency.&lt;/p&gt;

&lt;p&gt;The project is open source: &lt;a href="https://github.com/Leeson-Wong/resume-maker" rel="noopener noreferrer"&gt;github.com/Leeson-Wong/resume-maker&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;This article won't oversell the technical implementation (honestly, it's basic). Instead, let's talk about what the &lt;strong&gt;Resume + MCP&lt;/strong&gt; combination actually means.&lt;/p&gt;

&lt;h3&gt;
  
  
  What is MCP?
&lt;/h3&gt;

&lt;p&gt;MCP (Model Context Protocol) is an open protocol proposed by Anthropic that lets AI Agents connect to external data sources and tools in a standardized way. Think of it as "the API for the AI era" — except MCP is self-describing (tools come with name, description, inputSchema), so AI Agents can automatically understand how to use them without integration docs.&lt;/p&gt;




&lt;h2&gt;
  
  
  See It in Action
&lt;/h2&gt;

&lt;p&gt;Here's what a recruiter's AI Agent can do after connecting:&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Query&lt;/strong&gt;: Search for big data related experience&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight javascript"&gt;&lt;code&gt;&lt;span class="nf"&gt;search_resume&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt; &lt;span class="na"&gt;query&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;big data&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt; &lt;span class="p"&gt;})&lt;/span&gt;

&lt;span class="err"&gt;→&lt;/span&gt; &lt;span class="nx"&gt;Returns&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;relevant&lt;/span&gt; &lt;span class="nx"&gt;experience&lt;/span&gt; &lt;span class="nx"&gt;snippets&lt;/span&gt; &lt;span class="o"&gt;+&lt;/span&gt; &lt;span class="nx"&gt;source&lt;/span&gt; &lt;span class="nx"&gt;modules&lt;/span&gt; &lt;span class="o"&gt;+&lt;/span&gt; &lt;span class="nx"&gt;relevance&lt;/span&gt; &lt;span class="nx"&gt;scores&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;&lt;strong&gt;Query&lt;/strong&gt;: Is this candidate a fit for our role?&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight javascript"&gt;&lt;code&gt;&lt;span class="nf"&gt;evaluate_fit&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt;
  &lt;span class="na"&gt;job_description&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="se"&gt;\"&lt;/span&gt;&lt;span class="s2"&gt;Senior full-stack engineer, 5+ years, React/Node.js,&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;
                    &lt;span class="nx"&gt;big&lt;/span&gt; &lt;span class="nx"&gt;data&lt;/span&gt; &lt;span class="nx"&gt;platform&lt;/span&gt; &lt;span class="nx"&gt;experience&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;DevOps&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;
})

→ {
    score: 82,
    matched: [&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="nx"&gt;react&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;, &lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="nx"&gt;node&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;, &lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="nx"&gt;big&lt;/span&gt; &lt;span class="nx"&gt;data&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;, &lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="nx"&gt;devops&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;, ...],
    missing: [&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="nx"&gt;Kubernetes&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;],
    recommendation: &lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="nx"&gt;Strong&lt;/span&gt; &lt;span class="nx"&gt;match&lt;/span&gt;&lt;span class="o"&gt;!&lt;/span&gt; &lt;span class="p"&gt;...&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;
  }
&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;&lt;strong&gt;Query&lt;/strong&gt;: Give me a comprehensive assessment&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight javascript"&gt;&lt;code&gt;&lt;span class="nf"&gt;get_career_summary&lt;/span&gt;&lt;span class="p"&gt;({})&lt;/span&gt;

&lt;span class="err"&gt;→&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="nl"&gt;seniority&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;senior&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="nx"&gt;totalYears&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mi"&gt;6&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="nx"&gt;domains&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;[&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Big Data Cloud Services&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Cross-platform Development&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Indie Product Development&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;],&lt;/span&gt;
    &lt;span class="nx"&gt;coreStrengths&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;[&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Multi-stack adaptability&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Independent product delivery&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;AI protocol design&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;],&lt;/span&gt;
    &lt;span class="nx"&gt;topSkills&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;[{&lt;/span&gt; &lt;span class="na"&gt;name&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;React&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;level&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mi"&gt;4&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;yearsUsed&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mi"&gt;5&lt;/span&gt; &lt;span class="p"&gt;},&lt;/span&gt; &lt;span class="p"&gt;...]&lt;/span&gt;
  &lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;All structured JSON — no PDF parsing, no web scraping. The AI Agent processes data programmatically.&lt;/p&gt;




&lt;h2&gt;
  
  
  The Key Insight Isn't Technical — It's Product Logic
&lt;/h2&gt;

&lt;h3&gt;
  
  
  The Resume Submission Flow Has Changed
&lt;/h3&gt;

&lt;p&gt;Old flow: Send PDF → HR skims manually → Maybe passes to AI for辅助判断&lt;/p&gt;

&lt;p&gt;New flow: Give HR a link + invite code → Their AI Agent connects via MCP → The Agent decides what to query and how to analyze&lt;/p&gt;

&lt;p&gt;The difference: &lt;strong&gt;the initiative shifts from HR to the AI Agent&lt;/strong&gt;. The Agent queries dynamically based on what it cares about, rather than passively reading a fixed-format document.&lt;/p&gt;

&lt;h3&gt;
  
  
  Invite Code = Reverse Filtering
&lt;/h3&gt;

&lt;p&gt;This is my favorite part of the design.&lt;/p&gt;

&lt;p&gt;When applying, you give HR a "link + invite code." HR needs to configure the invite code in their AI client before the Agent can access deep information.&lt;/p&gt;

&lt;p&gt;The process itself is a filter: &lt;strong&gt;teams without sufficient AI literacy can't even get through it&lt;/strong&gt;. If they can, it means they're already using AI Agents in their hiring process — likely a more modern team.&lt;/p&gt;

&lt;p&gt;Plus, the invite code supports tracking — you can see exactly which tools the other party's Agent queried and when. Far more valuable than a "resume viewed" notification. You know &lt;strong&gt;what they actually care about&lt;/strong&gt;.&lt;/p&gt;

&lt;h3&gt;
  
  
  Privacy Tiers
&lt;/h3&gt;

&lt;p&gt;Public page and MCP responses expose different levels of information:&lt;/p&gt;

&lt;div class="table-wrapper-paragraph"&gt;&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Tier&lt;/th&gt;
&lt;th&gt;What's visible&lt;/th&gt;
&lt;th&gt;Channel&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;Public&lt;/td&gt;
&lt;td&gt;Name, title, skill tags, work history&lt;/td&gt;
&lt;td&gt;
&lt;code&gt;/&lt;/code&gt; public page&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;MCP Query&lt;/td&gt;
&lt;td&gt;Project details, skill proficiency, fit analysis&lt;/td&gt;
&lt;td&gt;
&lt;code&gt;/mcp&lt;/code&gt; (invite code required)&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Private&lt;/td&gt;
&lt;td&gt;Email, phone number&lt;/td&gt;
&lt;td&gt;Never exposed via MCP&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;&lt;/div&gt;

&lt;p&gt;Not everything should be AI-accessible. Contact info belongs in the human communication phase.&lt;/p&gt;




&lt;h2&gt;
  
  
  Three-Layer Discovery Mechanism
&lt;/h2&gt;

&lt;p&gt;How does an AI Agent know your resume has an MCP service? This is the most critical design decision — &lt;strong&gt;discovery matters more than connection&lt;/strong&gt;.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;AI Agent visits https://your-domain.com
  ↓
Parses HTML, discovers &amp;lt;link rel="mcp" href="/mcp"&amp;gt;
  ↓
Requests /.well-known/mcp for full metadata (tool list, auth method)
  ↓
Connects using invite code via Authorization header
  ↓
Starts querying
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Three layers of discovery, shallow to deep:&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Layer 1: HTML Injection&lt;/strong&gt; — Auto-injected into public page, zero maintenance cost&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight html"&gt;&lt;code&gt;&lt;span class="nt"&gt;&amp;lt;link&lt;/span&gt; &lt;span class="na"&gt;rel=&lt;/span&gt;&lt;span class="s"&gt;"mcp"&lt;/span&gt; &lt;span class="na"&gt;href=&lt;/span&gt;&lt;span class="s"&gt;"/mcp"&lt;/span&gt;&lt;span class="nt"&gt;&amp;gt;&lt;/span&gt;
&lt;span class="nt"&gt;&amp;lt;link&lt;/span&gt; &lt;span class="na"&gt;rel=&lt;/span&gt;&lt;span class="s"&gt;"llms-txt"&lt;/span&gt; &lt;span class="na"&gt;href=&lt;/span&gt;&lt;span class="s"&gt;"/llms.txt"&lt;/span&gt;&lt;span class="nt"&gt;&amp;gt;&lt;/span&gt;
&lt;span class="nt"&gt;&amp;lt;script &lt;/span&gt;&lt;span class="na"&gt;type=&lt;/span&gt;&lt;span class="s"&gt;"application/ld+json"&lt;/span&gt;&lt;span class="nt"&gt;&amp;gt;&lt;/span&gt;
&lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;@type&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Person&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;name&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;...&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;knowsAbout&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;[...]&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt;
&lt;span class="nt"&gt;&amp;lt;/script&amp;gt;&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;&lt;strong&gt;Layer 2: /.well-known/mcp&lt;/strong&gt; — Standard well-known endpoint, machine-readable full MCP 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;"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="nl"&gt;"endpoint"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"/mcp"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"transport"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"http"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"auth"&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;"type"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"bearer"&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;"tools"&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="err"&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;&lt;strong&gt;Layer 3: /llms.txt&lt;/strong&gt; — Follows the &lt;a href="https://llmstxt.org/" rel="noopener noreferrer"&gt;llms.txt&lt;/a&gt; standard, Markdown-formatted AI-friendly description&lt;/p&gt;

&lt;p&gt;All three layers are auto-generated server-side from the same &lt;code&gt;resume.json&lt;/code&gt;. Update your resume, and all discovery info updates automatically.&lt;/p&gt;




&lt;h2&gt;
  
  
  Architecture
&lt;/h2&gt;

&lt;p&gt;This is a self-hosted full-stack application, not SaaS. Single-user, deployed on your own server.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;/            → Public page (visible to anyone, MCP discovery info injected)
/mcp         → MCP endpoint (invite code auth, for AI Agents)
/edit        → Resume editor (auth code protected, for yourself)
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Core principle: &lt;strong&gt;single source of truth&lt;/strong&gt; — &lt;code&gt;resume.json&lt;/code&gt;. Editor and MCP share the same data. MCP reads fresh data on every request — no cache staleness.&lt;/p&gt;

&lt;p&gt;The tech stack is nothing to brag about: React 19 + Vite 7 + Tailwind 4 for the editor, vanilla Node.js HTTP for the server, MCP SDK for the protocol layer, Zod for data validation. Search uses a hand-rolled TF scorer + mixed Chinese/English tokenization — good enough for keyword matching, nowhere near real semantic search.&lt;/p&gt;

&lt;p&gt;8 tools total:&lt;/p&gt;

&lt;div class="table-wrapper-paragraph"&gt;&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Tool&lt;/th&gt;
&lt;th&gt;What it does&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;get_profile&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Public info, privacy fields never via MCP&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;get_experience&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Work history, filterable by keyword&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;get_projects&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Projects, filterable by tech stack&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;get_skills&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Skill list with proficiency and years&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;get_education&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Education background&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;search_resume&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Full-text search, TF relevance scoring&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;evaluate_fit&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Given a JD, outputs match score + missing skills&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;get_career_summary&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Career overview: seniority, domains, core strengths&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;&lt;/div&gt;

&lt;p&gt;Each tool supports &lt;code&gt;format: 'text' | 'json'&lt;/code&gt; — text for humans, json for programs.&lt;/p&gt;




&lt;h2&gt;
  
  
  Engineering Decisions Worth Mentioning
&lt;/h2&gt;

&lt;p&gt;&lt;strong&gt;Atomic writes instead of a database&lt;/strong&gt; — Resume data is ~2KB. Even SQLite feels overkill. write-to-temp + rename is atomic, plus a write queue for concurrency safety.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Fresh MCP Server per request&lt;/strong&gt; — Instead of a long-lived MCP Server created at startup, each request calls &lt;code&gt;createMcpServer(currentResume)&lt;/code&gt;. This guarantees the latest data — edits are immediately visible to MCP.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Zod schema validation&lt;/strong&gt; — Validate on both read and write. The frontend editor validates too, but never trusting client input is basic hygiene.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;stdio mode&lt;/strong&gt; — Besides HTTP, supports &lt;code&gt;start:stdio&lt;/code&gt; for direct Claude Desktop configuration. No server needed for local development.&lt;/p&gt;




&lt;h2&gt;
  
  
  Deployment
&lt;/h2&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;&lt;span class="c"&gt;# One-command Docker deployment&lt;/span&gt;
&lt;span class="nb"&gt;cp &lt;/span&gt;config.example.json config.json   &lt;span class="c"&gt;# Set your authCode&lt;/span&gt;
docker compose up &lt;span class="nt"&gt;-d&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Or connect directly via Claude Desktop:&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;"resume"&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;"tsx"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"server/stdio.ts"&lt;/span&gt;&lt;span class="p"&gt;],&lt;/span&gt;&lt;span class="w"&gt;
      &lt;/span&gt;&lt;span class="nl"&gt;"cwd"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"/path/to/resume-maker"&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;h2&gt;
  
  
  Honest Take
&lt;/h2&gt;

&lt;p&gt;The search and matching algorithms in this project are honestly mediocre. TF scoring is just keyword frequency counting, skill normalization is a lookup table, and evaluate_fit's matching logic is essentially string containment checking. These are far from "precisely understanding personal data."&lt;/p&gt;

&lt;p&gt;But the interesting part isn't the implementation.&lt;/p&gt;

&lt;p&gt;What's interesting is the new possibilities created by the &lt;strong&gt;Resume + MCP&lt;/strong&gt; combination:&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;
&lt;strong&gt;Information access has changed&lt;/strong&gt; — From "reading a fixed document" to "AI querying structured data on demand"&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Filtering logic has changed&lt;/strong&gt; — Invite codes reverse-filter recruiters by their AI literacy&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Data ownership has changed&lt;/strong&gt; — Data lives on your own server, you control who sees what&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;These three shifts have nothing to do with implementation complexity. They're innovations at the &lt;strong&gt;product form&lt;/strong&gt; level.&lt;/p&gt;

&lt;p&gt;MCP is still early — almost no recruiters are actually using AI Agents to connect to MCP endpoints. But the direction is right. As AI becomes more involved in hiring, resumes should evolve from "documents for humans" into "data sources for AI."&lt;/p&gt;

&lt;p&gt;If you're also experimenting with MCP, or interested in "AI-native personal information presentation," let's connect.&lt;/p&gt;




&lt;p&gt;&lt;strong&gt;Live Demo&lt;/strong&gt;: &lt;a href="http://47.97.101.194:965" rel="noopener noreferrer"&gt;http://47.97.101.194:965&lt;/a&gt; (Invite code: &lt;code&gt;qm8vvf&lt;/code&gt; — connect to the MCP endpoint to try all tools)&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Project&lt;/strong&gt;: &lt;a href="https://github.com/Leeson-Wong/resume-maker" rel="noopener noreferrer"&gt;github.com/Leeson-Wong/resume-maker&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;If you find it interesting, a Star would be appreciated.&lt;/p&gt;




&lt;h2&gt;
  
  
  Another Project: BrickLife
&lt;/h2&gt;

&lt;p&gt;Quick shoutout to another thing I built — a simulation/management mini-game called "BrickLife"&lt;/p&gt;

&lt;p&gt;Play: &lt;strong&gt;&lt;a href="http://47.97.101.194" rel="noopener noreferrer"&gt;http://47.97.101.194&lt;/a&gt;&lt;/strong&gt; (Best experience on mobile)&lt;/p&gt;




&lt;blockquote&gt;
&lt;p&gt;This article was co-written with Claude Code + GLM-5.1.&lt;/p&gt;
&lt;/blockquote&gt;

</description>
      <category>agents</category>
      <category>career</category>
      <category>mcp</category>
      <category>showdev</category>
    </item>
  </channel>
</rss>
