resuml 3.1.0 → 3.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -2,21 +2,31 @@ import {
2
2
  loadResumeFiles,
3
3
  program,
4
4
  themeRender_exports
5
- } from "./chunk-M6JY5UDJ.js";
5
+ } from "./chunk-YVC53STN.js";
6
6
  import {
7
7
  loadTheme,
8
8
  processResumeData
9
9
  } from "./chunk-G4AN2EMI.js";
10
+ import {
11
+ buildTailorPrompt,
12
+ deriveSearchQuery,
13
+ scorePosting,
14
+ searchJobs
15
+ } from "./chunk-QBCXFLW6.js";
10
16
  import {
11
17
  analyzeAts
12
- } from "./chunk-N55EPZ2N.js";
18
+ } from "./chunk-C2JG5KF4.js";
13
19
  import "./chunk-QR77BRMN.js";
14
20
  export {
15
21
  analyzeAts,
22
+ buildTailorPrompt,
23
+ deriveSearchQuery,
16
24
  loadResumeFiles,
17
25
  loadTheme,
18
26
  processResumeData,
19
27
  program,
28
+ scorePosting,
29
+ searchJobs,
20
30
  themeRender_exports as themeRender
21
31
  };
22
32
  //# sourceMappingURL=index.js.map
@@ -0,0 +1,158 @@
1
+ import { T as TieredAtsResult } from '../types-B_jASYU4.js';
2
+ import { Resume as ResumeSchema } from '../types/index.js';
3
+
4
+ type ProviderId = 'greenhouse' | 'lever' | 'ashby' | 'workable' | 'remoteok' | 'wwr' | 'remotive' | 'hn-whoishiring';
5
+ type SeniorityLevel = 'intern' | 'junior' | 'mid' | 'senior' | 'staff' | 'principal';
6
+ interface SearchQuery {
7
+ /** Top skills extracted from the resume, ranked desc by relevance. */
8
+ skills: string[];
9
+ /** Heuristic seniority derived from work history. */
10
+ seniority: SeniorityLevel;
11
+ /** Total years of professional experience. */
12
+ yearsExperience: number;
13
+ /** Roles the candidate has held (used for title matching). */
14
+ titles: string[];
15
+ /** Candidate's home city if available. */
16
+ city?: string;
17
+ /** Candidate's country (ISO 3166-1 alpha-2) if available. */
18
+ countryCode?: string;
19
+ /** Whether to restrict to remote-friendly postings. */
20
+ remoteOnly: boolean;
21
+ /** Free-text query terms for providers that accept keyword search. */
22
+ terms: string[];
23
+ }
24
+ interface SearchOptions {
25
+ /** Restrict the source set. Defaults to all bundled providers. */
26
+ providers?: ProviderId[];
27
+ /** Cap on results returned after ranking. Default 20. */
28
+ limit?: number;
29
+ /** Drop postings below this total ATS score after ranking. Default 60. */
30
+ minScore?: number;
31
+ /** Filter to remote postings only. Overrides resume-derived value when set. */
32
+ remoteOnly?: boolean;
33
+ /** Override candidate location as "City, CC" (e.g. "Zurich, CH"). Overrides resume-derived countryCode. */
34
+ location?: string;
35
+ /** Per-provider timeout in ms. Default 8000. */
36
+ timeoutMs?: number;
37
+ /** Extra company slugs to query for allowlist-based providers. */
38
+ extraCompanies?: Partial<Record<ProviderId, string[]>>;
39
+ }
40
+ interface JobPosting {
41
+ /** Stable id within source. */
42
+ id: string;
43
+ /** Provider that surfaced this posting. */
44
+ source: ProviderId;
45
+ company: string;
46
+ title: string;
47
+ /** Free-form location string from the provider. */
48
+ location?: string;
49
+ remote: boolean;
50
+ /** Posting URL. */
51
+ url: string;
52
+ /** Body text used for ATS scoring. HTML stripped. */
53
+ body: string;
54
+ /** ISO date if provided by source. */
55
+ postedAt?: string;
56
+ /** Salary string if provider exposes one. Not normalized. */
57
+ compensation?: string;
58
+ /** Tags / keywords if exposed (RemoteOK, Remotive). */
59
+ tags?: string[];
60
+ }
61
+ interface RankedJob extends JobPosting {
62
+ /** Result of analyzeAts(resume, { jobDescription: body }). */
63
+ ats: TieredAtsResult;
64
+ /** Hash key used for cross-provider dedupe. */
65
+ dedupeKey: string;
66
+ }
67
+ interface ProviderResult {
68
+ providerId: ProviderId;
69
+ /** Postings normalized to JobPosting. */
70
+ postings: JobPosting[];
71
+ /** Error message if the provider failed. Postings still returns []. */
72
+ error?: string;
73
+ /** Milliseconds spent fetching + normalizing. */
74
+ durationMs: number;
75
+ }
76
+ interface Provider {
77
+ id: ProviderId;
78
+ /**
79
+ * Whether this provider requires a company allowlist (true) or returns
80
+ * a single global feed (false).
81
+ */
82
+ needsAllowlist: boolean;
83
+ /**
84
+ * Fetch postings. Implementations must catch their own errors and return
85
+ * an empty array on failure — the orchestrator never re-throws.
86
+ */
87
+ fetch(query: SearchQuery, options: {
88
+ companies?: string[];
89
+ timeoutMs: number;
90
+ }): Promise<JobPosting[]>;
91
+ }
92
+ interface SearchResult {
93
+ query: SearchQuery;
94
+ /** Sorted desc by total ATS score. */
95
+ jobs: RankedJob[];
96
+ /** Per-provider stats for observability. */
97
+ providers: ProviderResult[];
98
+ /** Postings dropped because total score < minScore. */
99
+ filteredCount: number;
100
+ /** Total postings fetched before dedupe. */
101
+ fetchedCount: number;
102
+ /** Ranked postings where role-family-match status is fail. */
103
+ offSpecialtyCount?: number;
104
+ /** Postings dropped by the on-site country filter. */
105
+ wrongLocationCount?: number;
106
+ }
107
+
108
+ /**
109
+ * Derive a search query from the resume. Pure function — no IO.
110
+ *
111
+ * Top skills come from the bundled O*NET skill index (same one the ATS
112
+ * matcher uses), ranked by occurrence and the "hot" flag. Seniority is the
113
+ * max of explicit title evidence and a YOE-derived heuristic so a resume
114
+ * with a short tenure but a "Senior" title still maps to senior roles.
115
+ */
116
+ declare function deriveSearchQuery(resume: ResumeSchema, overrides?: {
117
+ remoteOnly?: boolean;
118
+ }): SearchQuery;
119
+
120
+ /**
121
+ * Strip HTML to plain text. Naive but sufficient for ATS scoring — the
122
+ * scorer only needs prose for skill extraction, not exact whitespace.
123
+ */
124
+ declare function stripHtml(input: string): string;
125
+ /**
126
+ * Stable dedupe key — same (company, title, location) from two providers
127
+ * collapses to one entry.
128
+ */
129
+ declare function dedupeKey(posting: Pick<JobPosting, 'company' | 'title' | 'location'>): string;
130
+
131
+ declare function listProviders(): Provider[];
132
+ declare function getProvider(id: ProviderId): Provider | undefined;
133
+
134
+ /**
135
+ * Discover + score job postings against a resume.
136
+ *
137
+ * Pipeline:
138
+ * 1. Derive a SearchQuery from the resume (skills, seniority, location).
139
+ * 2. Fan out to enabled providers in parallel; per-provider errors are
140
+ * captured in `providers[]` and never bubble out.
141
+ * 3. Apply remote-only filter if requested.
142
+ * 4. Score every survivor via analyzeAts(resume, { jobDescription }), dedupe
143
+ * by (company, title, location), and sort desc by total ATS score.
144
+ * 5. Drop entries below `minScore`, cap to `limit`.
145
+ */
146
+ declare function searchJobs(resume: ResumeSchema, options?: SearchOptions): Promise<SearchResult>;
147
+ /**
148
+ * Score a single posting body against a resume. Useful for the MCP
149
+ * `resuml_jobs_score` tool when the agent already has the JD text in hand.
150
+ */
151
+ declare function scorePosting(resume: ResumeSchema, posting: Pick<JobPosting, 'body' | 'company' | 'title' | 'location' | 'url' | 'source' | 'id' | 'remote'>): RankedJob | undefined;
152
+ /**
153
+ * Build a prompt that asks the agent to tailor the resume to a specific
154
+ * posting. The MCP `resuml_jobs_tailor` tool returns this string.
155
+ */
156
+ declare function buildTailorPrompt(posting: JobPosting): string;
157
+
158
+ export { type JobPosting, type ProviderId, type ProviderResult, type RankedJob, type SearchOptions, type SearchQuery, type SearchResult, type SeniorityLevel, buildTailorPrompt, dedupeKey, deriveSearchQuery, getProvider, listProviders, scorePosting, searchJobs, stripHtml };
@@ -0,0 +1,23 @@
1
+ import {
2
+ buildTailorPrompt,
3
+ dedupeKey,
4
+ deriveSearchQuery,
5
+ getProvider,
6
+ listProviders,
7
+ scorePosting,
8
+ searchJobs,
9
+ stripHtml
10
+ } from "../chunk-QBCXFLW6.js";
11
+ import "../chunk-C2JG5KF4.js";
12
+ import "../chunk-QR77BRMN.js";
13
+ export {
14
+ buildTailorPrompt,
15
+ dedupeKey,
16
+ deriveSearchQuery,
17
+ getProvider,
18
+ listProviders,
19
+ scorePosting,
20
+ searchJobs,
21
+ stripHtml
22
+ };
23
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":[],"sourcesContent":[],"mappings":"","names":[]}
@@ -8,10 +8,15 @@ import {
8
8
  loadTheme,
9
9
  processResumeData
10
10
  } from "../chunk-G4AN2EMI.js";
11
+ import {
12
+ buildTailorPrompt,
13
+ scorePosting,
14
+ searchJobs
15
+ } from "../chunk-QBCXFLW6.js";
11
16
  import {
12
17
  analyzeAts,
13
18
  loadConfig
14
- } from "../chunk-N55EPZ2N.js";
19
+ } from "../chunk-C2JG5KF4.js";
15
20
  import "../chunk-QR77BRMN.js";
16
21
 
17
22
  // src/mcp/server.ts
@@ -598,6 +603,129 @@ ${resumeYaml}
598
603
  }
599
604
  }
600
605
  );
606
+ server.registerTool(
607
+ "resuml_jobs_search",
608
+ {
609
+ title: "Search Jobs",
610
+ description: "Discover job postings across free sources (Greenhouse, Lever, Ashby, Workable, RemoteOK, Remotive, WeWorkRemotely, HN Who is Hiring), score each against the resume with the ATS engine, and return the ranked queue. No paid APIs, no LinkedIn scraping.",
611
+ inputSchema: {
612
+ yaml: z.string().describe("Resume content in YAML format"),
613
+ remote: z.boolean().optional().describe("Filter to remote-friendly postings"),
614
+ minScore: z.number().int().min(0).max(100).optional().describe("Minimum total ATS score (default 85)"),
615
+ limit: z.number().int().min(1).max(100).optional().describe("Max postings returned (default 20)"),
616
+ providers: z.array(
617
+ z.enum([
618
+ "greenhouse",
619
+ "lever",
620
+ "ashby",
621
+ "workable",
622
+ "remoteok",
623
+ "wwr",
624
+ "remotive",
625
+ "hn-whoishiring"
626
+ ])
627
+ ).optional().describe("Restrict to a subset of providers"),
628
+ timeoutMs: z.number().int().optional().describe("Per-provider timeout in ms (default 8000)")
629
+ }
630
+ },
631
+ async ({ yaml, remote, minScore, limit, providers, timeoutMs }) => {
632
+ suppressStdout();
633
+ try {
634
+ const resume = await processResumeData([yaml]);
635
+ const result = await searchJobs(resume, {
636
+ ...remote !== void 0 && { remoteOnly: remote },
637
+ ...minScore !== void 0 && { minScore },
638
+ ...limit !== void 0 && { limit },
639
+ ...providers && { providers },
640
+ ...timeoutMs !== void 0 && { timeoutMs }
641
+ });
642
+ restoreStdout();
643
+ return { content: [{ type: "text", text: JSON.stringify(result, null, 2) }] };
644
+ } catch (e) {
645
+ restoreStdout();
646
+ return {
647
+ content: [
648
+ {
649
+ type: "text",
650
+ text: JSON.stringify({ error: e instanceof Error ? e.message : String(e) })
651
+ }
652
+ ],
653
+ isError: true
654
+ };
655
+ }
656
+ }
657
+ );
658
+ server.registerTool(
659
+ "resuml_jobs_score",
660
+ {
661
+ title: "Score Job Posting",
662
+ description: "Score a single JD body against the resume. Use when the agent already has the JD text (e.g. user pasted a URL elsewhere) and just wants the ranked ATS breakdown without a fresh search.",
663
+ inputSchema: {
664
+ yaml: z.string().describe("Resume content in YAML format"),
665
+ company: z.string().describe("Posting company name"),
666
+ title: z.string().describe("Posting title"),
667
+ body: z.string().describe("Full job description text"),
668
+ url: z.string().describe("Posting URL"),
669
+ location: z.string().optional().describe("Free-form location")
670
+ }
671
+ },
672
+ async ({ yaml, company, title, body, url, location }) => {
673
+ suppressStdout();
674
+ try {
675
+ const resume = await processResumeData([yaml]);
676
+ const ranked = scorePosting(resume, {
677
+ id: `manual:${url}`,
678
+ source: "remoteok",
679
+ company,
680
+ title,
681
+ body,
682
+ url,
683
+ ...location !== void 0 && { location },
684
+ remote: !!location && /remote/i.test(location)
685
+ });
686
+ restoreStdout();
687
+ return { content: [{ type: "text", text: JSON.stringify(ranked, null, 2) }] };
688
+ } catch (e) {
689
+ restoreStdout();
690
+ return {
691
+ content: [
692
+ {
693
+ type: "text",
694
+ text: JSON.stringify({ error: e instanceof Error ? e.message : String(e) })
695
+ }
696
+ ],
697
+ isError: true
698
+ };
699
+ }
700
+ }
701
+ );
702
+ server.registerTool(
703
+ "resuml_jobs_tailor",
704
+ {
705
+ title: "Build Tailor Prompt",
706
+ description: "Return a prompt that asks the agent to tailor the resume YAML to a specific posting, chaining into resuml_validate and resuml_ats_check.",
707
+ inputSchema: {
708
+ company: z.string(),
709
+ title: z.string(),
710
+ body: z.string(),
711
+ url: z.string(),
712
+ location: z.string().optional()
713
+ }
714
+ },
715
+ ({ company, title, body, url, location }) => {
716
+ const prompt = buildTailorPrompt({
717
+ id: `manual:${url}`,
718
+ source: "remoteok",
719
+ company,
720
+ title,
721
+ body,
722
+ url,
723
+ ...location !== void 0 && { location },
724
+ remote: !!location && /remote/i.test(location)
725
+ });
726
+ return { content: [{ type: "text", text: prompt }] };
727
+ }
728
+ );
601
729
  return server;
602
730
  }
603
731
  function parseMargin(margin) {
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/mcp/server.ts"],"sourcesContent":["import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';\nimport { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';\nimport { z } from 'zod';\nimport { processResumeData } from '../core';\nimport { analyzeAts } from '../ats/index';\nimport { listRubricMarkdown, getRubricEntry } from '../ats/rubric';\nimport { loadConfig } from '../utils/config';\nimport { loadTheme } from '../utils/themeLoader';\nimport { generateResumeYaml } from '../utils/resumeTemplate';\nimport { KNOWN_THEMES, isThemeInstalled, getInstalledVersion } from '../utils/themeInfo';\n\n// Redirect console.log to stderr so it doesn't corrupt the MCP stdio channel\nconst originalLog = console.log;\nconst originalWarn = console.warn;\n\nfunction suppressStdout() {\n console.log = (...args: unknown[]) => {\n console.error('[resuml]', ...args);\n };\n console.warn = (...args: unknown[]) => {\n console.error('[resuml]', ...args);\n };\n}\n\nfunction restoreStdout() {\n console.log = originalLog;\n console.warn = originalWarn;\n}\n\n// ── Shared data ─────────────────────────────────────────────────────\n\nconst JSON_RESUME_SCHEMA_REFERENCE = `# JSON Resume Schema Reference\n\nThe JSON Resume schema defines the structure for resume data. resuml uses YAML as the input format.\n\n## Top-level sections\n\n| Section | Required | Description |\n|---------|----------|-------------|\n| basics | Yes | Name, label, email, phone, url, summary, location, profiles |\n| work | Recommended | Work experience entries |\n| education | Recommended | Education entries |\n| skills | Recommended | Skill categories with keywords |\n| projects | Optional | Project entries |\n| volunteer | Optional | Volunteer experience |\n| awards | Optional | Awards and honors |\n| certificates | Optional | Professional certifications |\n| publications | Optional | Published works |\n| languages | Optional | Language proficiencies |\n| interests | Optional | Personal interests |\n| references | Optional | Professional references |\n\n## Section schemas\n\n### basics\n\\`\\`\\`yaml\nbasics:\n name: \"Full Name\" # required\n label: \"Professional Title\"\n email: \"email@example.com\"\n phone: \"+1-555-123-4567\"\n url: \"https://website.com\"\n summary: \"2-4 sentence professional summary\"\n location:\n city: \"City\"\n countryCode: \"US\"\n region: \"State\"\n profiles:\n - network: \"LinkedIn\"\n username: \"username\"\n url: \"https://linkedin.com/in/username\"\n\\`\\`\\`\n\n### work\n\\`\\`\\`yaml\nwork:\n - name: \"Company Name\"\n position: \"Job Title\"\n url: \"https://company.com\"\n startDate: \"2020-01-01\" # ISO 8601\n endDate: \"2023-12-31\" # omit for current position\n summary: \"Role description\"\n highlights:\n - \"Achievement with measurable result\"\n\\`\\`\\`\n\n### education\n\\`\\`\\`yaml\neducation:\n - institution: \"University\"\n area: \"Field of Study\"\n studyType: \"Degree Type\" # e.g. Bachelor, Master, PhD\n startDate: \"2014-09-01\"\n endDate: \"2018-06-01\"\n\\`\\`\\`\n\n### skills\n\\`\\`\\`yaml\nskills:\n - name: \"Category\"\n level: \"Expert\" # Master, Expert, Advanced, Intermediate, Beginner\n keywords: [\"Skill1\", \"Skill2\"]\n\\`\\`\\`\n\n### projects\n\\`\\`\\`yaml\nprojects:\n - name: \"Project Name\"\n description: \"What it does\"\n highlights: [\"Key achievement\"]\n keywords: [\"Tech1\", \"Tech2\"]\n startDate: \"2023-01-01\"\n url: \"https://github.com/...\"\n\\`\\`\\`\n\n### certificates\n\\`\\`\\`yaml\ncertificates:\n - name: \"Certificate Name\"\n date: \"2023-01-01\"\n issuer: \"Issuing Organization\"\n url: \"https://credential-url.com\"\n\\`\\`\\`\n\n### languages\n\\`\\`\\`yaml\nlanguages:\n - language: \"English\"\n fluency: \"Native speaker\" # Native speaker, Fluent, Advanced, Intermediate, Elementary\n\\`\\`\\`\n\n## Formatting rules\n- Dates: ISO 8601 format (YYYY-MM-DD or YYYY-MM)\n- Start highlights with action verbs: Developed, Implemented, Led, Optimized, Reduced, Built, Designed\n- Include numbers in 50%+ of highlights (e.g., \"Reduced latency by 40%\")\n- Never use first person (I, my, me, we)\n- Summary: 2-4 sentences positioning the candidate for the specific role\n`;\n\nfunction createServer(): McpServer {\n const server = new McpServer({\n name: 'resuml',\n version: '2.0.0',\n });\n\n // ═══════════════════════════════════════════════════════════════════\n // RESOURCES\n // ═══════════════════════════════════════════════════════════════════\n\n // ── JSON Resume Schema Reference ──────────────────────────────────\n\n server.registerResource(\n 'json-resume-schema',\n 'resuml://schema/json-resume',\n {\n description:\n 'JSON Resume schema reference with all sections, field types, and formatting rules',\n mimeType: 'text/markdown',\n },\n () => ({\n contents: [\n {\n uri: 'resuml://schema/json-resume',\n mimeType: 'text/markdown',\n text: JSON_RESUME_SCHEMA_REFERENCE,\n },\n ],\n })\n );\n\n // ── ATS Rubric ───────────────────────────────────────────────────\n\n server.registerResource(\n 'ats-rubric',\n 'resuml://docs/ats-rubric',\n {\n description:\n 'Tiered ATS rubric: every check, its tier, weight, evidence level, description, and source URL.',\n mimeType: 'text/markdown',\n },\n () => ({\n contents: [\n {\n uri: 'resuml://docs/ats-rubric',\n mimeType: 'text/markdown',\n text: listRubricMarkdown(),\n },\n ],\n })\n );\n\n // ── Theme Catalog ─────────────────────────────────────────────────\n\n server.registerResource(\n 'theme-catalog',\n 'resuml://themes/catalog',\n {\n description: 'Available resume themes with descriptions and installation status',\n mimeType: 'application/json',\n },\n () => {\n const themes = KNOWN_THEMES.map((t) => ({\n name: t.name,\n package: t.pkg,\n description: t.description,\n installed: isThemeInstalled(t.pkg),\n version: getInstalledVersion(t.pkg),\n }));\n return {\n contents: [\n {\n uri: 'resuml://themes/catalog',\n mimeType: 'application/json',\n text: JSON.stringify({ themes, totalCount: themes.length }, null, 2),\n },\n ],\n };\n }\n );\n\n // ═══════════════════════════════════════════════════════════════════\n // PROMPTS\n // ═══════════════════════════════════════════════════════════════════\n\n // ── Tailor Resume to Job Description ──────────────────────────────\n\n server.registerPrompt(\n 'tailor-resume-to-jd',\n {\n title: 'Tailor Resume to Job Description',\n description: 'Generate a tailored resume YAML optimized for a specific job description',\n argsSchema: {\n jobDescription: z.string().describe('The full job description text'),\n candidateName: z.string().optional().describe('Candidate full name'),\n candidateEmail: z.string().optional().describe('Candidate email address'),\n candidateBackground: z\n .string()\n .optional()\n .describe(\n 'Brief summary of the candidate background, skills, and experience to incorporate'\n ),\n },\n },\n ({ jobDescription, candidateName, candidateEmail, candidateBackground }) => ({\n messages: [\n {\n role: 'user',\n content: {\n type: 'text',\n text: `Create a tailored resume in YAML format optimized for the following job description.\n\n## Job Description\n${jobDescription}\n\n${candidateBackground ? `## Candidate Background\\n${candidateBackground}\\n` : ''}\n## Instructions\n\n1. Analyze the job description to identify required skills, technologies, experience level, and industry terms\n2. Generate a complete resume YAML following the JSON Resume schema (read the resuml://schema/json-resume resource for the full schema)\n3. Mirror exact terminology from the job description in skills and highlights\n4. Start every highlight with an action verb (Developed, Implemented, Led, Optimized, Reduced, Built, Designed)\n5. Include numbers in 50%+ of highlights (e.g., \"Reduced latency by 40%\", \"Managed team of 8\")\n6. Never use \"I\", \"my\", \"me\", \"we\"\n7. Write a 2-4 sentence summary positioning the candidate for this specific role\n8. Use ISO 8601 dates (YYYY-MM-DD or YYYY-MM)\n${candidateName ? `9. Use candidate name: ${candidateName}` : ''}\n${candidateEmail ? `10. Use candidate email: ${candidateEmail}` : ''}\n\n## Workflow\nAfter generating the YAML:\n1. Use \\`resuml_validate\\` to check schema compliance\n2. Use \\`resuml_ats_check\\` with the job description text. Target: total >= 75, parsing tier grade A, hard-skill-overlap >= 70%\n3. If ATS score is low, revise the YAML and re-check\n4. Use \\`resuml_render\\` with theme \"even\" for the final output\n\nOutput the resume YAML first, then run the validation and ATS check tools.`,\n },\n },\n ],\n })\n );\n\n // ── Optimize ATS Score ────────────────────────────────────────────\n\n server.registerPrompt(\n 'optimize-ats-score',\n {\n title: 'Optimize ATS Score',\n description: 'Analyze and improve an existing resume YAML to maximize its ATS score',\n argsSchema: {\n resumeYaml: z.string().describe('The current resume YAML content'),\n jobDescription: z\n .string()\n .optional()\n .describe('Optional job description to optimize against'),\n targetScore: z.string().optional().describe('Target ATS score (default: 85)'),\n },\n },\n ({ resumeYaml, jobDescription, targetScore }) => ({\n messages: [\n {\n role: 'user',\n content: {\n type: 'text',\n text: `Optimize this resume YAML to maximize its ATS score${targetScore ? ` (target: ${targetScore})` : ' (target: 85+)'}.\n\n## Current Resume YAML\n\\`\\`\\`yaml\n${resumeYaml}\n\\`\\`\\`\n\n${jobDescription ? `## Job Description\\n${jobDescription}\\n` : ''}\n## Instructions\n\n1. First, run \\`resuml_ats_check\\` on the current YAML${jobDescription ? ' with the job description' : ''} to get the baseline score\n2. Read the ATS rubric (resuml://docs/ats-rubric) to understand what checks are performed, their tier, and weight\n3. Review each failed or low-scoring check and fix the issues:\n - Missing contact info → add it\n - No summary → write a 2-4 sentence professional summary\n - Weak highlights → rewrite with action verbs and quantified metrics\n - Missing keywords → incorporate them naturally into skills and highlights\n - Structural issues → ensure all essential sections are present\n4. Run \\`resuml_ats_check\\` again to verify improvement\n5. Repeat until the target score is reached\n\nOutput the improved YAML with a summary of changes made.`,\n },\n },\n ],\n })\n );\n\n // ── Review Resume ─────────────────────────────────────────────────\n\n server.registerPrompt(\n 'review-resume',\n {\n title: 'Review Resume',\n description:\n 'Comprehensive review of a resume YAML with ATS analysis and improvement suggestions',\n argsSchema: {\n resumeYaml: z.string().describe('The resume YAML content to review'),\n },\n },\n ({ resumeYaml }) => ({\n messages: [\n {\n role: 'user',\n content: {\n type: 'text',\n text: `Perform a comprehensive review of this resume.\n\n## Resume YAML\n\\`\\`\\`yaml\n${resumeYaml}\n\\`\\`\\`\n\n## Review steps\n\n1. Run \\`resuml_validate\\` to check schema compliance\n2. Run \\`resuml_ats_check\\` for ATS analysis\n3. Review content quality:\n - Is the summary compelling and role-specific?\n - Do highlights use strong action verbs?\n - Are achievements quantified with metrics?\n - Are skills well-organized and comprehensive?\n - Is the work history clear and impactful?\n4. Provide a structured review with:\n - **ATS Score**: Current score and rating\n - **Schema Issues**: Any validation errors\n - **Strengths**: What the resume does well\n - **Improvements**: Specific, actionable suggestions\n - **Revised YAML**: An improved version if significant changes are recommended`,\n },\n },\n ],\n })\n );\n\n // ═══════════════════════════════════════════════════════════════════\n // TOOLS\n // ═══════════════════════════════════════════════════════════════════\n\n // ── resuml_init_resume ──────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_init_resume',\n {\n title: 'Init Resume',\n description: 'Generate a starter resume YAML template following the JSON Resume schema',\n inputSchema: {\n name: z.string().optional().describe('Full name for the resume'),\n title: z.string().optional().describe('Professional title/label'),\n email: z.string().optional().describe('Email address'),\n },\n },\n ({ name, title, email }) => {\n const yaml = generateResumeYaml(\n name ?? 'Your Name',\n email ?? 'email@example.com',\n title ?? 'Professional Title'\n );\n return { content: [{ type: 'text' as const, text: yaml }] };\n }\n );\n\n // ── resuml_validate ─────────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_validate',\n {\n title: 'Validate Resume',\n description: 'Validate resume YAML against the JSON Resume schema',\n inputSchema: {\n yaml: z.string().describe('Resume content in YAML format'),\n },\n },\n async ({ yaml }) => {\n suppressStdout();\n try {\n await processResumeData([yaml]);\n restoreStdout();\n return {\n content: [\n { type: 'text' as const, text: JSON.stringify({ valid: true, errors: [] }, null, 2) },\n ],\n };\n } catch (e: unknown) {\n restoreStdout();\n const message = e instanceof Error ? e.message : String(e);\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({ valid: false, errors: [message] }, null, 2),\n },\n ],\n };\n }\n }\n );\n\n // ── resuml_ats_check ────────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_ats_check',\n {\n title: 'ATS Check',\n description:\n 'Run ATS (Applicant Tracking System) analysis on a resume, optionally matching against a job description',\n inputSchema: {\n yaml: z.string().describe('Resume content in YAML format'),\n jobDescription: z\n .string()\n .optional()\n .describe('Job description text to match keywords against'),\n language: z.enum(['en', 'de']).optional().describe('Language for analysis (default: en)'),\n },\n },\n async ({ yaml, jobDescription, language }) => {\n suppressStdout();\n try {\n const resume = await processResumeData([yaml]);\n const cfg = loadConfig();\n const result = analyzeAts(resume, {\n language: language ?? cfg.locale,\n jobDescription,\n config: cfg,\n });\n restoreStdout();\n return {\n content: [{ type: 'text' as const, text: JSON.stringify(result, null, 2) }],\n };\n } catch (e: unknown) {\n restoreStdout();\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({ error: e instanceof Error ? e.message : String(e) }),\n },\n ],\n isError: true,\n };\n }\n }\n );\n\n // ── resuml_ats_explain ──────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_ats_explain',\n {\n title: 'ATS Rubric Explain',\n description:\n 'Return the rubric entry (tier, weight, evidence level, description, source) for a given check id.',\n inputSchema: {\n checkId: z.string().describe('Check id, e.g. quantification-density'),\n },\n },\n ({ checkId }) => {\n const entry = getRubricEntry(checkId);\n if (!entry) {\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({ error: `Unknown check id: ${checkId}` }),\n },\n ],\n isError: true,\n };\n }\n return { content: [{ type: 'text' as const, text: JSON.stringify(entry, null, 2) }] };\n }\n );\n\n // ── resuml_render ───────────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_render',\n {\n title: 'Render Resume',\n description: 'Render a resume to HTML using a specified theme',\n inputSchema: {\n yaml: z.string().describe('Resume content in YAML format'),\n theme: z\n .string()\n .default('even')\n .describe('Theme name (e.g. even, stackoverflow, elegant, paper, kendall)'),\n locale: z.string().optional().describe('Locale for theme rendering (e.g. en, de)'),\n },\n },\n async (args) => {\n const { yaml, theme } = args;\n const locale = args['locale'];\n suppressStdout();\n try {\n const resume = await processResumeData([yaml]);\n const themeModule = loadTheme(theme, { autoInstall: false });\n const renderOptions: Record<string, unknown> = {};\n if (locale) renderOptions['locale'] = locale;\n const html = await themeModule.render(resume, renderOptions);\n restoreStdout();\n return { content: [{ type: 'text' as const, text: html }] };\n } catch (e: unknown) {\n restoreStdout();\n const message = e instanceof Error ? e.message : String(e);\n const hint = message.includes('Cannot find module')\n ? `. Install with: resuml themes --install ${theme}`\n : '';\n return {\n content: [{ type: 'text' as const, text: JSON.stringify({ error: message + hint }) }],\n isError: true,\n };\n }\n }\n );\n\n // ── resuml_list_themes ──────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_list_themes',\n {\n title: 'List Themes',\n description: 'List available resume themes with their installation status',\n },\n () => {\n const themes = KNOWN_THEMES.map((t) => ({\n name: t.name,\n package: t.pkg,\n description: t.description,\n installed: isThemeInstalled(t.pkg),\n version: getInstalledVersion(t.pkg),\n }));\n return {\n content: [{ type: 'text' as const, text: JSON.stringify({ themes }, null, 2) }],\n };\n }\n );\n\n // ── resuml_export_pdf ───────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_export_pdf',\n {\n title: 'Export PDF',\n description: 'Export a resume as PDF (requires Playwright to be installed)',\n inputSchema: {\n yaml: z.string().describe('Resume content in YAML format'),\n theme: z.string().default('even').describe('Theme name'),\n format: z.enum(['A4', 'Letter']).default('A4').describe('Paper format'),\n locale: z.string().optional().describe('Locale for theme rendering (e.g. en, de)'),\n margin: z\n .string()\n .optional()\n .describe(\n 'Page margins. Single value (e.g. \"10mm\") for all sides, two values (e.g. \"10mm,15mm\") for vertical/horizontal, or four values (e.g. \"10mm,15mm,10mm,15mm\") for top/right/bottom/left'\n ),\n },\n },\n async (args) => {\n const { yaml, theme, format } = args;\n const locale = args['locale'];\n const margin = args['margin'];\n suppressStdout();\n try {\n const resume = await processResumeData([yaml]);\n const themeModule = loadTheme(theme, { autoInstall: false });\n const renderOptions: Record<string, unknown> = {};\n if (locale) renderOptions['locale'] = locale;\n const html = await themeModule.render(resume, renderOptions);\n\n let chromium;\n try {\n const pw = await import('playwright');\n chromium = pw.chromium;\n } catch {\n restoreStdout();\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({\n error: 'Playwright is not installed. Run: npm install playwright',\n }),\n },\n ],\n isError: true,\n };\n }\n\n const parsedMargin = parseMargin(margin);\n const browser = await chromium.launch({ headless: true });\n const page = await browser.newPage();\n await page.setContent(html, { waitUntil: 'networkidle' });\n const pdfBuffer = await page.pdf({\n format,\n printBackground: true,\n margin: parsedMargin,\n });\n await browser.close();\n restoreStdout();\n\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({\n pdf: Buffer.from(pdfBuffer).toString('base64'),\n encoding: 'base64',\n format,\n }),\n },\n ],\n };\n } catch (e: unknown) {\n restoreStdout();\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({ error: e instanceof Error ? e.message : String(e) }),\n },\n ],\n isError: true,\n };\n }\n }\n );\n\n return server;\n}\n\nfunction parseMargin(margin?: string): Record<string, string> {\n const defaultMargin = { top: '10mm', right: '10mm', bottom: '10mm', left: '10mm' };\n if (!margin) return defaultMargin;\n\n const parts = margin.split(',').map((s) => s.trim());\n if (parts.length === 1 && parts[0]) {\n return { top: parts[0], right: parts[0], bottom: parts[0], left: parts[0] };\n }\n if (parts.length === 2 && parts[0] && parts[1]) {\n return { top: parts[0], right: parts[1], bottom: parts[0], left: parts[1] };\n }\n if (parts.length === 4 && parts[0] && parts[1] && parts[2] && parts[3]) {\n return { top: parts[0], right: parts[1], bottom: parts[2], left: parts[3] };\n }\n return defaultMargin;\n}\n\nexport async function startMcpServer(): Promise<void> {\n const server = createServer();\n const transport = new StdioServerTransport();\n await server.connect(transport);\n}\n"],"mappings":"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","names":[]}
1
+ {"version":3,"sources":["../../src/mcp/server.ts"],"sourcesContent":["import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';\nimport { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';\nimport { z } from 'zod';\nimport { processResumeData } from '../core';\nimport { analyzeAts } from '../ats/index';\nimport { searchJobs, scorePosting, buildTailorPrompt } from '../jobs/index';\nimport { listRubricMarkdown, getRubricEntry } from '../ats/rubric';\nimport { loadConfig } from '../utils/config';\nimport { loadTheme } from '../utils/themeLoader';\nimport { generateResumeYaml } from '../utils/resumeTemplate';\nimport { KNOWN_THEMES, isThemeInstalled, getInstalledVersion } from '../utils/themeInfo';\n\n// Redirect console.log to stderr so it doesn't corrupt the MCP stdio channel\nconst originalLog = console.log;\nconst originalWarn = console.warn;\n\nfunction suppressStdout() {\n console.log = (...args: unknown[]) => {\n console.error('[resuml]', ...args);\n };\n console.warn = (...args: unknown[]) => {\n console.error('[resuml]', ...args);\n };\n}\n\nfunction restoreStdout() {\n console.log = originalLog;\n console.warn = originalWarn;\n}\n\n// ── Shared data ─────────────────────────────────────────────────────\n\nconst JSON_RESUME_SCHEMA_REFERENCE = `# JSON Resume Schema Reference\n\nThe JSON Resume schema defines the structure for resume data. resuml uses YAML as the input format.\n\n## Top-level sections\n\n| Section | Required | Description |\n|---------|----------|-------------|\n| basics | Yes | Name, label, email, phone, url, summary, location, profiles |\n| work | Recommended | Work experience entries |\n| education | Recommended | Education entries |\n| skills | Recommended | Skill categories with keywords |\n| projects | Optional | Project entries |\n| volunteer | Optional | Volunteer experience |\n| awards | Optional | Awards and honors |\n| certificates | Optional | Professional certifications |\n| publications | Optional | Published works |\n| languages | Optional | Language proficiencies |\n| interests | Optional | Personal interests |\n| references | Optional | Professional references |\n\n## Section schemas\n\n### basics\n\\`\\`\\`yaml\nbasics:\n name: \"Full Name\" # required\n label: \"Professional Title\"\n email: \"email@example.com\"\n phone: \"+1-555-123-4567\"\n url: \"https://website.com\"\n summary: \"2-4 sentence professional summary\"\n location:\n city: \"City\"\n countryCode: \"US\"\n region: \"State\"\n profiles:\n - network: \"LinkedIn\"\n username: \"username\"\n url: \"https://linkedin.com/in/username\"\n\\`\\`\\`\n\n### work\n\\`\\`\\`yaml\nwork:\n - name: \"Company Name\"\n position: \"Job Title\"\n url: \"https://company.com\"\n startDate: \"2020-01-01\" # ISO 8601\n endDate: \"2023-12-31\" # omit for current position\n summary: \"Role description\"\n highlights:\n - \"Achievement with measurable result\"\n\\`\\`\\`\n\n### education\n\\`\\`\\`yaml\neducation:\n - institution: \"University\"\n area: \"Field of Study\"\n studyType: \"Degree Type\" # e.g. Bachelor, Master, PhD\n startDate: \"2014-09-01\"\n endDate: \"2018-06-01\"\n\\`\\`\\`\n\n### skills\n\\`\\`\\`yaml\nskills:\n - name: \"Category\"\n level: \"Expert\" # Master, Expert, Advanced, Intermediate, Beginner\n keywords: [\"Skill1\", \"Skill2\"]\n\\`\\`\\`\n\n### projects\n\\`\\`\\`yaml\nprojects:\n - name: \"Project Name\"\n description: \"What it does\"\n highlights: [\"Key achievement\"]\n keywords: [\"Tech1\", \"Tech2\"]\n startDate: \"2023-01-01\"\n url: \"https://github.com/...\"\n\\`\\`\\`\n\n### certificates\n\\`\\`\\`yaml\ncertificates:\n - name: \"Certificate Name\"\n date: \"2023-01-01\"\n issuer: \"Issuing Organization\"\n url: \"https://credential-url.com\"\n\\`\\`\\`\n\n### languages\n\\`\\`\\`yaml\nlanguages:\n - language: \"English\"\n fluency: \"Native speaker\" # Native speaker, Fluent, Advanced, Intermediate, Elementary\n\\`\\`\\`\n\n## Formatting rules\n- Dates: ISO 8601 format (YYYY-MM-DD or YYYY-MM)\n- Start highlights with action verbs: Developed, Implemented, Led, Optimized, Reduced, Built, Designed\n- Include numbers in 50%+ of highlights (e.g., \"Reduced latency by 40%\")\n- Never use first person (I, my, me, we)\n- Summary: 2-4 sentences positioning the candidate for the specific role\n`;\n\nfunction createServer(): McpServer {\n const server = new McpServer({\n name: 'resuml',\n version: '2.0.0',\n });\n\n // ═══════════════════════════════════════════════════════════════════\n // RESOURCES\n // ═══════════════════════════════════════════════════════════════════\n\n // ── JSON Resume Schema Reference ──────────────────────────────────\n\n server.registerResource(\n 'json-resume-schema',\n 'resuml://schema/json-resume',\n {\n description:\n 'JSON Resume schema reference with all sections, field types, and formatting rules',\n mimeType: 'text/markdown',\n },\n () => ({\n contents: [\n {\n uri: 'resuml://schema/json-resume',\n mimeType: 'text/markdown',\n text: JSON_RESUME_SCHEMA_REFERENCE,\n },\n ],\n })\n );\n\n // ── ATS Rubric ───────────────────────────────────────────────────\n\n server.registerResource(\n 'ats-rubric',\n 'resuml://docs/ats-rubric',\n {\n description:\n 'Tiered ATS rubric: every check, its tier, weight, evidence level, description, and source URL.',\n mimeType: 'text/markdown',\n },\n () => ({\n contents: [\n {\n uri: 'resuml://docs/ats-rubric',\n mimeType: 'text/markdown',\n text: listRubricMarkdown(),\n },\n ],\n })\n );\n\n // ── Theme Catalog ─────────────────────────────────────────────────\n\n server.registerResource(\n 'theme-catalog',\n 'resuml://themes/catalog',\n {\n description: 'Available resume themes with descriptions and installation status',\n mimeType: 'application/json',\n },\n () => {\n const themes = KNOWN_THEMES.map((t) => ({\n name: t.name,\n package: t.pkg,\n description: t.description,\n installed: isThemeInstalled(t.pkg),\n version: getInstalledVersion(t.pkg),\n }));\n return {\n contents: [\n {\n uri: 'resuml://themes/catalog',\n mimeType: 'application/json',\n text: JSON.stringify({ themes, totalCount: themes.length }, null, 2),\n },\n ],\n };\n }\n );\n\n // ═══════════════════════════════════════════════════════════════════\n // PROMPTS\n // ═══════════════════════════════════════════════════════════════════\n\n // ── Tailor Resume to Job Description ──────────────────────────────\n\n server.registerPrompt(\n 'tailor-resume-to-jd',\n {\n title: 'Tailor Resume to Job Description',\n description: 'Generate a tailored resume YAML optimized for a specific job description',\n argsSchema: {\n jobDescription: z.string().describe('The full job description text'),\n candidateName: z.string().optional().describe('Candidate full name'),\n candidateEmail: z.string().optional().describe('Candidate email address'),\n candidateBackground: z\n .string()\n .optional()\n .describe(\n 'Brief summary of the candidate background, skills, and experience to incorporate'\n ),\n },\n },\n ({ jobDescription, candidateName, candidateEmail, candidateBackground }) => ({\n messages: [\n {\n role: 'user',\n content: {\n type: 'text',\n text: `Create a tailored resume in YAML format optimized for the following job description.\n\n## Job Description\n${jobDescription}\n\n${candidateBackground ? `## Candidate Background\\n${candidateBackground}\\n` : ''}\n## Instructions\n\n1. Analyze the job description to identify required skills, technologies, experience level, and industry terms\n2. Generate a complete resume YAML following the JSON Resume schema (read the resuml://schema/json-resume resource for the full schema)\n3. Mirror exact terminology from the job description in skills and highlights\n4. Start every highlight with an action verb (Developed, Implemented, Led, Optimized, Reduced, Built, Designed)\n5. Include numbers in 50%+ of highlights (e.g., \"Reduced latency by 40%\", \"Managed team of 8\")\n6. Never use \"I\", \"my\", \"me\", \"we\"\n7. Write a 2-4 sentence summary positioning the candidate for this specific role\n8. Use ISO 8601 dates (YYYY-MM-DD or YYYY-MM)\n${candidateName ? `9. Use candidate name: ${candidateName}` : ''}\n${candidateEmail ? `10. Use candidate email: ${candidateEmail}` : ''}\n\n## Workflow\nAfter generating the YAML:\n1. Use \\`resuml_validate\\` to check schema compliance\n2. Use \\`resuml_ats_check\\` with the job description text. Target: total >= 75, parsing tier grade A, hard-skill-overlap >= 70%\n3. If ATS score is low, revise the YAML and re-check\n4. Use \\`resuml_render\\` with theme \"even\" for the final output\n\nOutput the resume YAML first, then run the validation and ATS check tools.`,\n },\n },\n ],\n })\n );\n\n // ── Optimize ATS Score ────────────────────────────────────────────\n\n server.registerPrompt(\n 'optimize-ats-score',\n {\n title: 'Optimize ATS Score',\n description: 'Analyze and improve an existing resume YAML to maximize its ATS score',\n argsSchema: {\n resumeYaml: z.string().describe('The current resume YAML content'),\n jobDescription: z\n .string()\n .optional()\n .describe('Optional job description to optimize against'),\n targetScore: z.string().optional().describe('Target ATS score (default: 85)'),\n },\n },\n ({ resumeYaml, jobDescription, targetScore }) => ({\n messages: [\n {\n role: 'user',\n content: {\n type: 'text',\n text: `Optimize this resume YAML to maximize its ATS score${targetScore ? ` (target: ${targetScore})` : ' (target: 85+)'}.\n\n## Current Resume YAML\n\\`\\`\\`yaml\n${resumeYaml}\n\\`\\`\\`\n\n${jobDescription ? `## Job Description\\n${jobDescription}\\n` : ''}\n## Instructions\n\n1. First, run \\`resuml_ats_check\\` on the current YAML${jobDescription ? ' with the job description' : ''} to get the baseline score\n2. Read the ATS rubric (resuml://docs/ats-rubric) to understand what checks are performed, their tier, and weight\n3. Review each failed or low-scoring check and fix the issues:\n - Missing contact info → add it\n - No summary → write a 2-4 sentence professional summary\n - Weak highlights → rewrite with action verbs and quantified metrics\n - Missing keywords → incorporate them naturally into skills and highlights\n - Structural issues → ensure all essential sections are present\n4. Run \\`resuml_ats_check\\` again to verify improvement\n5. Repeat until the target score is reached\n\nOutput the improved YAML with a summary of changes made.`,\n },\n },\n ],\n })\n );\n\n // ── Review Resume ─────────────────────────────────────────────────\n\n server.registerPrompt(\n 'review-resume',\n {\n title: 'Review Resume',\n description:\n 'Comprehensive review of a resume YAML with ATS analysis and improvement suggestions',\n argsSchema: {\n resumeYaml: z.string().describe('The resume YAML content to review'),\n },\n },\n ({ resumeYaml }) => ({\n messages: [\n {\n role: 'user',\n content: {\n type: 'text',\n text: `Perform a comprehensive review of this resume.\n\n## Resume YAML\n\\`\\`\\`yaml\n${resumeYaml}\n\\`\\`\\`\n\n## Review steps\n\n1. Run \\`resuml_validate\\` to check schema compliance\n2. Run \\`resuml_ats_check\\` for ATS analysis\n3. Review content quality:\n - Is the summary compelling and role-specific?\n - Do highlights use strong action verbs?\n - Are achievements quantified with metrics?\n - Are skills well-organized and comprehensive?\n - Is the work history clear and impactful?\n4. Provide a structured review with:\n - **ATS Score**: Current score and rating\n - **Schema Issues**: Any validation errors\n - **Strengths**: What the resume does well\n - **Improvements**: Specific, actionable suggestions\n - **Revised YAML**: An improved version if significant changes are recommended`,\n },\n },\n ],\n })\n );\n\n // ═══════════════════════════════════════════════════════════════════\n // TOOLS\n // ═══════════════════════════════════════════════════════════════════\n\n // ── resuml_init_resume ──────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_init_resume',\n {\n title: 'Init Resume',\n description: 'Generate a starter resume YAML template following the JSON Resume schema',\n inputSchema: {\n name: z.string().optional().describe('Full name for the resume'),\n title: z.string().optional().describe('Professional title/label'),\n email: z.string().optional().describe('Email address'),\n },\n },\n ({ name, title, email }) => {\n const yaml = generateResumeYaml(\n name ?? 'Your Name',\n email ?? 'email@example.com',\n title ?? 'Professional Title'\n );\n return { content: [{ type: 'text' as const, text: yaml }] };\n }\n );\n\n // ── resuml_validate ─────────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_validate',\n {\n title: 'Validate Resume',\n description: 'Validate resume YAML against the JSON Resume schema',\n inputSchema: {\n yaml: z.string().describe('Resume content in YAML format'),\n },\n },\n async ({ yaml }) => {\n suppressStdout();\n try {\n await processResumeData([yaml]);\n restoreStdout();\n return {\n content: [\n { type: 'text' as const, text: JSON.stringify({ valid: true, errors: [] }, null, 2) },\n ],\n };\n } catch (e: unknown) {\n restoreStdout();\n const message = e instanceof Error ? e.message : String(e);\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({ valid: false, errors: [message] }, null, 2),\n },\n ],\n };\n }\n }\n );\n\n // ── resuml_ats_check ────────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_ats_check',\n {\n title: 'ATS Check',\n description:\n 'Run ATS (Applicant Tracking System) analysis on a resume, optionally matching against a job description',\n inputSchema: {\n yaml: z.string().describe('Resume content in YAML format'),\n jobDescription: z\n .string()\n .optional()\n .describe('Job description text to match keywords against'),\n language: z.enum(['en', 'de']).optional().describe('Language for analysis (default: en)'),\n },\n },\n async ({ yaml, jobDescription, language }) => {\n suppressStdout();\n try {\n const resume = await processResumeData([yaml]);\n const cfg = loadConfig();\n const result = analyzeAts(resume, {\n language: language ?? cfg.locale,\n jobDescription,\n config: cfg,\n });\n restoreStdout();\n return {\n content: [{ type: 'text' as const, text: JSON.stringify(result, null, 2) }],\n };\n } catch (e: unknown) {\n restoreStdout();\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({ error: e instanceof Error ? e.message : String(e) }),\n },\n ],\n isError: true,\n };\n }\n }\n );\n\n // ── resuml_ats_explain ──────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_ats_explain',\n {\n title: 'ATS Rubric Explain',\n description:\n 'Return the rubric entry (tier, weight, evidence level, description, source) for a given check id.',\n inputSchema: {\n checkId: z.string().describe('Check id, e.g. quantification-density'),\n },\n },\n ({ checkId }) => {\n const entry = getRubricEntry(checkId);\n if (!entry) {\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({ error: `Unknown check id: ${checkId}` }),\n },\n ],\n isError: true,\n };\n }\n return { content: [{ type: 'text' as const, text: JSON.stringify(entry, null, 2) }] };\n }\n );\n\n // ── resuml_render ───────────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_render',\n {\n title: 'Render Resume',\n description: 'Render a resume to HTML using a specified theme',\n inputSchema: {\n yaml: z.string().describe('Resume content in YAML format'),\n theme: z\n .string()\n .default('even')\n .describe('Theme name (e.g. even, stackoverflow, elegant, paper, kendall)'),\n locale: z.string().optional().describe('Locale for theme rendering (e.g. en, de)'),\n },\n },\n async (args) => {\n const { yaml, theme } = args;\n const locale = args['locale'];\n suppressStdout();\n try {\n const resume = await processResumeData([yaml]);\n const themeModule = loadTheme(theme, { autoInstall: false });\n const renderOptions: Record<string, unknown> = {};\n if (locale) renderOptions['locale'] = locale;\n const html = await themeModule.render(resume, renderOptions);\n restoreStdout();\n return { content: [{ type: 'text' as const, text: html }] };\n } catch (e: unknown) {\n restoreStdout();\n const message = e instanceof Error ? e.message : String(e);\n const hint = message.includes('Cannot find module')\n ? `. Install with: resuml themes --install ${theme}`\n : '';\n return {\n content: [{ type: 'text' as const, text: JSON.stringify({ error: message + hint }) }],\n isError: true,\n };\n }\n }\n );\n\n // ── resuml_list_themes ──────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_list_themes',\n {\n title: 'List Themes',\n description: 'List available resume themes with their installation status',\n },\n () => {\n const themes = KNOWN_THEMES.map((t) => ({\n name: t.name,\n package: t.pkg,\n description: t.description,\n installed: isThemeInstalled(t.pkg),\n version: getInstalledVersion(t.pkg),\n }));\n return {\n content: [{ type: 'text' as const, text: JSON.stringify({ themes }, null, 2) }],\n };\n }\n );\n\n // ── resuml_export_pdf ───────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_export_pdf',\n {\n title: 'Export PDF',\n description: 'Export a resume as PDF (requires Playwright to be installed)',\n inputSchema: {\n yaml: z.string().describe('Resume content in YAML format'),\n theme: z.string().default('even').describe('Theme name'),\n format: z.enum(['A4', 'Letter']).default('A4').describe('Paper format'),\n locale: z.string().optional().describe('Locale for theme rendering (e.g. en, de)'),\n margin: z\n .string()\n .optional()\n .describe(\n 'Page margins. Single value (e.g. \"10mm\") for all sides, two values (e.g. \"10mm,15mm\") for vertical/horizontal, or four values (e.g. \"10mm,15mm,10mm,15mm\") for top/right/bottom/left'\n ),\n },\n },\n async (args) => {\n const { yaml, theme, format } = args;\n const locale = args['locale'];\n const margin = args['margin'];\n suppressStdout();\n try {\n const resume = await processResumeData([yaml]);\n const themeModule = loadTheme(theme, { autoInstall: false });\n const renderOptions: Record<string, unknown> = {};\n if (locale) renderOptions['locale'] = locale;\n const html = await themeModule.render(resume, renderOptions);\n\n let chromium;\n try {\n const pw = await import('playwright');\n chromium = pw.chromium;\n } catch {\n restoreStdout();\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({\n error: 'Playwright is not installed. Run: npm install playwright',\n }),\n },\n ],\n isError: true,\n };\n }\n\n const parsedMargin = parseMargin(margin);\n const browser = await chromium.launch({ headless: true });\n const page = await browser.newPage();\n await page.setContent(html, { waitUntil: 'networkidle' });\n const pdfBuffer = await page.pdf({\n format,\n printBackground: true,\n margin: parsedMargin,\n });\n await browser.close();\n restoreStdout();\n\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({\n pdf: Buffer.from(pdfBuffer).toString('base64'),\n encoding: 'base64',\n format,\n }),\n },\n ],\n };\n } catch (e: unknown) {\n restoreStdout();\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({ error: e instanceof Error ? e.message : String(e) }),\n },\n ],\n isError: true,\n };\n }\n }\n );\n\n // ── resuml_jobs_search ──────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_jobs_search',\n {\n title: 'Search Jobs',\n description:\n 'Discover job postings across free sources (Greenhouse, Lever, Ashby, Workable, RemoteOK, Remotive, WeWorkRemotely, HN Who is Hiring), score each against the resume with the ATS engine, and return the ranked queue. No paid APIs, no LinkedIn scraping.',\n inputSchema: {\n yaml: z.string().describe('Resume content in YAML format'),\n remote: z.boolean().optional().describe('Filter to remote-friendly postings'),\n minScore: z.number().int().min(0).max(100).optional().describe('Minimum total ATS score (default 85)'),\n limit: z.number().int().min(1).max(100).optional().describe('Max postings returned (default 20)'),\n providers: z\n .array(\n z.enum([\n 'greenhouse',\n 'lever',\n 'ashby',\n 'workable',\n 'remoteok',\n 'wwr',\n 'remotive',\n 'hn-whoishiring',\n ])\n )\n .optional()\n .describe('Restrict to a subset of providers'),\n timeoutMs: z.number().int().optional().describe('Per-provider timeout in ms (default 8000)'),\n },\n },\n async ({ yaml, remote, minScore, limit, providers, timeoutMs }) => {\n suppressStdout();\n try {\n const resume = await processResumeData([yaml]);\n const result = await searchJobs(resume, {\n ...(remote !== undefined && { remoteOnly: remote }),\n ...(minScore !== undefined && { minScore }),\n ...(limit !== undefined && { limit }),\n ...(providers && { providers }),\n ...(timeoutMs !== undefined && { timeoutMs }),\n });\n restoreStdout();\n return { content: [{ type: 'text' as const, text: JSON.stringify(result, null, 2) }] };\n } catch (e: unknown) {\n restoreStdout();\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({ error: e instanceof Error ? e.message : String(e) }),\n },\n ],\n isError: true,\n };\n }\n }\n );\n\n // ── resuml_jobs_score ───────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_jobs_score',\n {\n title: 'Score Job Posting',\n description:\n 'Score a single JD body against the resume. Use when the agent already has the JD text (e.g. user pasted a URL elsewhere) and just wants the ranked ATS breakdown without a fresh search.',\n inputSchema: {\n yaml: z.string().describe('Resume content in YAML format'),\n company: z.string().describe('Posting company name'),\n title: z.string().describe('Posting title'),\n body: z.string().describe('Full job description text'),\n url: z.string().describe('Posting URL'),\n location: z.string().optional().describe('Free-form location'),\n },\n },\n async ({ yaml, company, title, body, url, location }) => {\n suppressStdout();\n try {\n const resume = await processResumeData([yaml]);\n const ranked = scorePosting(resume, {\n id: `manual:${url}`,\n source: 'remoteok',\n company,\n title,\n body,\n url,\n ...(location !== undefined && { location }),\n remote: !!location && /remote/i.test(location),\n });\n restoreStdout();\n return { content: [{ type: 'text' as const, text: JSON.stringify(ranked, null, 2) }] };\n } catch (e: unknown) {\n restoreStdout();\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify({ error: e instanceof Error ? e.message : String(e) }),\n },\n ],\n isError: true,\n };\n }\n }\n );\n\n // ── resuml_jobs_tailor ──────────────────────────────────────────────\n\n server.registerTool(\n 'resuml_jobs_tailor',\n {\n title: 'Build Tailor Prompt',\n description:\n 'Return a prompt that asks the agent to tailor the resume YAML to a specific posting, chaining into resuml_validate and resuml_ats_check.',\n inputSchema: {\n company: z.string(),\n title: z.string(),\n body: z.string(),\n url: z.string(),\n location: z.string().optional(),\n },\n },\n ({ company, title, body, url, location }) => {\n const prompt = buildTailorPrompt({\n id: `manual:${url}`,\n source: 'remoteok',\n company,\n title,\n body,\n url,\n ...(location !== undefined && { location }),\n remote: !!location && /remote/i.test(location),\n });\n return { content: [{ type: 'text' as const, text: prompt }] };\n }\n );\n\n return server;\n}\n\nfunction parseMargin(margin?: string): Record<string, string> {\n const defaultMargin = { top: '10mm', right: '10mm', bottom: '10mm', left: '10mm' };\n if (!margin) return defaultMargin;\n\n const parts = margin.split(',').map((s) => s.trim());\n if (parts.length === 1 && parts[0]) {\n return { top: parts[0], right: parts[0], bottom: parts[0], left: parts[0] };\n }\n if (parts.length === 2 && parts[0] && parts[1]) {\n return { top: parts[0], right: parts[1], bottom: parts[0], left: parts[1] };\n }\n if (parts.length === 4 && parts[0] && parts[1] && parts[2] && parts[3]) {\n return { top: parts[0], right: parts[1], bottom: parts[2], left: parts[3] };\n }\n return defaultMargin;\n}\n\nexport async function startMcpServer(): Promise<void> {\n const server = createServer();\n const transport = new StdioServerTransport();\n await server.connect(transport);\n}\n"],"mappings":"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","names":[]}
@@ -0,0 +1,81 @@
1
+ type Tier = 'parsing' | 'match' | 'recruiter';
2
+ type AtsCheckWeight = 'high' | 'medium' | 'low';
3
+ type AtsRating = 'excellent' | 'good' | 'needs-work' | 'poor';
4
+ type CheckStatus = 'pass' | 'warn' | 'fail' | 'skipped';
5
+ type Grade = 'A' | 'B' | 'C' | 'D' | 'F';
6
+ interface CheckResult {
7
+ id: string;
8
+ tier: Tier;
9
+ status: CheckStatus;
10
+ score: number;
11
+ weight: AtsCheckWeight;
12
+ message: string;
13
+ hints: string[];
14
+ }
15
+ interface TierResult {
16
+ score: number;
17
+ grade: Grade;
18
+ checks: CheckResult[];
19
+ }
20
+ interface KnockoutSignal {
21
+ signal: string;
22
+ evidence: string;
23
+ recommendation: string;
24
+ }
25
+ interface TieredAtsResult {
26
+ score: number;
27
+ rating: AtsRating;
28
+ tiers: {
29
+ parsing: TierResult;
30
+ match?: TierResult;
31
+ recruiter: TierResult;
32
+ };
33
+ knockouts: KnockoutSignal[];
34
+ summary: string;
35
+ }
36
+ interface AtsOptions {
37
+ language?: string;
38
+ jobDescription?: string;
39
+ /** Posting title, when known separately from the body (job search). */
40
+ jobTitle?: string;
41
+ threshold?: number;
42
+ config?: AtsConfig;
43
+ }
44
+ interface AtsConfig {
45
+ weights: {
46
+ tiers: {
47
+ parsing: number;
48
+ match: number;
49
+ recruiter: number;
50
+ };
51
+ checks: Record<string, AtsCheckWeight>;
52
+ };
53
+ thresholds: {
54
+ rating: {
55
+ excellent: number;
56
+ good: number;
57
+ needsWork: number;
58
+ };
59
+ grade: {
60
+ A: number;
61
+ B: number;
62
+ C: number;
63
+ D: number;
64
+ };
65
+ seniorYoeCutoff: number;
66
+ wordCount: {
67
+ min: number;
68
+ max: number;
69
+ seniorMax: number;
70
+ };
71
+ bulletsPerRole: {
72
+ min: number;
73
+ max: number;
74
+ seniorMax: number;
75
+ };
76
+ };
77
+ disable: string[];
78
+ locale: string;
79
+ }
80
+
81
+ export type { AtsOptions as A, CheckResult as C, KnockoutSignal as K, TieredAtsResult as T, TierResult as a };