@dolusoft/hirebase-mcp 1.1.20 → 1.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/application/use-cases/compare-candidates.d.ts +31 -0
- package/dist/application/use-cases/compare-candidates.js +78 -0
- package/dist/application/use-cases/compare-candidates.js.map +1 -0
- package/dist/application/use-cases/generate-screening-report.d.ts +39 -0
- package/dist/application/use-cases/generate-screening-report.js +70 -0
- package/dist/application/use-cases/generate-screening-report.js.map +1 -0
- package/dist/application/use-cases/get-cv-chunks.d.ts +11 -1
- package/dist/application/use-cases/get-cv-chunks.js +33 -1
- package/dist/application/use-cases/get-cv-chunks.js.map +1 -1
- package/dist/application/use-cases/get-screening-history.d.ts +26 -0
- package/dist/application/use-cases/get-screening-history.js +46 -0
- package/dist/application/use-cases/get-screening-history.js.map +1 -0
- package/dist/application/use-cases/index.d.ts +3 -0
- package/dist/application/use-cases/index.js +3 -0
- package/dist/application/use-cases/index.js.map +1 -1
- package/dist/application/use-cases/match-candidates.d.ts +13 -1
- package/dist/application/use-cases/match-candidates.js +31 -7
- package/dist/application/use-cases/match-candidates.js.map +1 -1
- package/dist/domain/entities/application.d.ts +1 -1
- package/dist/interface/dashboard/public/200.html +1 -1
- package/dist/interface/dashboard/public/404.html +1 -1
- package/dist/interface/dashboard/public/_nuxt/builds/latest.json +1 -1
- package/dist/interface/dashboard/public/_nuxt/builds/meta/05323eb6-c50f-4fab-aec8-17a93e7adeb3.json +1 -0
- package/dist/interface/dashboard/public/_nuxt/builds/meta/0d6e812f-4393-4ccc-a3d3-5f84e10cef1f.json +1 -0
- package/dist/interface/dashboard/public/_nuxt/builds/meta/1056fc86-712c-4f08-82e7-b0195da2d885.json +1 -0
- package/dist/interface/dashboard/public/_nuxt/builds/meta/287f4183-8718-470c-a30e-21776a882e1d.json +1 -0
- package/dist/interface/dashboard/public/_nuxt/builds/meta/47653a05-51ca-47b0-8728-ea69a473e926.json +1 -0
- package/dist/interface/dashboard/public/_nuxt/builds/meta/4c1f6063-4cfa-4abb-8f92-2172e917f88b.json +1 -0
- package/dist/interface/dashboard/public/_nuxt/builds/meta/76aaca87-0e82-4fa7-a19a-155439e56038.json +1 -0
- package/dist/interface/dashboard/public/_nuxt/builds/meta/8d10e489-6195-4ab5-9a19-ac5b6bfb3e21.json +1 -0
- package/dist/interface/dashboard/public/_nuxt/builds/meta/8ede37fe-3543-45b5-84a3-21350596eef7.json +1 -0
- package/dist/interface/dashboard/public/_nuxt/builds/meta/b4071b37-04ea-4aad-89c1-665c0a33ff14.json +1 -0
- package/dist/interface/dashboard/public/_nuxt/builds/meta/f020f3fc-f5ab-48bf-a28d-516004359cc2.json +1 -0
- package/dist/interface/dashboard/public/index.html +1 -1
- package/dist/interface/mcp/server.js +7 -1
- package/dist/interface/mcp/server.js.map +1 -1
- package/dist/interface/mcp/tools/compare-candidates.d.ts +3 -0
- package/dist/interface/mcp/tools/compare-candidates.js +23 -0
- package/dist/interface/mcp/tools/compare-candidates.js.map +1 -0
- package/dist/interface/mcp/tools/generate-screening-report.d.ts +3 -0
- package/dist/interface/mcp/tools/generate-screening-report.js +23 -0
- package/dist/interface/mcp/tools/generate-screening-report.js.map +1 -0
- package/dist/interface/mcp/tools/get-cv-chunks.js +23 -4
- package/dist/interface/mcp/tools/get-cv-chunks.js.map +1 -1
- package/dist/interface/mcp/tools/get-screening-history.d.ts +3 -0
- package/dist/interface/mcp/tools/get-screening-history.js +22 -0
- package/dist/interface/mcp/tools/get-screening-history.js.map +1 -0
- package/dist/interface/mcp/tools/match-candidates.js +16 -3
- package/dist/interface/mcp/tools/match-candidates.js.map +1 -1
- package/dist/interface/mcp/tools/update-pipeline-status.js +3 -1
- package/dist/interface/mcp/tools/update-pipeline-status.js.map +1 -1
- package/package.json +1 -1
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import type { ICandidateRepository } from '../../domain/repositories/candidate-repository.js';
|
|
2
|
+
import type { IJobPostingRepository } from '../../domain/repositories/job-posting-repository.js';
|
|
3
|
+
interface CandidateComparison {
|
|
4
|
+
candidateId: string;
|
|
5
|
+
name: string;
|
|
6
|
+
location?: string;
|
|
7
|
+
experienceYears?: number;
|
|
8
|
+
skills: string[];
|
|
9
|
+
loyaltyScore?: string;
|
|
10
|
+
avgTenureMonths?: number;
|
|
11
|
+
shortStintCount?: number;
|
|
12
|
+
education: string | null;
|
|
13
|
+
currentRole: string | null;
|
|
14
|
+
languages: string[];
|
|
15
|
+
}
|
|
16
|
+
interface JobMatchEntry {
|
|
17
|
+
compositeScore: number;
|
|
18
|
+
matchedSkills: string[];
|
|
19
|
+
missingSkills: string[];
|
|
20
|
+
}
|
|
21
|
+
export interface CompareResult {
|
|
22
|
+
candidates: CandidateComparison[];
|
|
23
|
+
job_match?: Record<string, JobMatchEntry>;
|
|
24
|
+
}
|
|
25
|
+
export declare class CompareCandidatesUseCase {
|
|
26
|
+
private candidateRepo;
|
|
27
|
+
private jobPostingRepo;
|
|
28
|
+
constructor(candidateRepo: ICandidateRepository, jobPostingRepo: IJobPostingRepository);
|
|
29
|
+
execute(candidateIds: string[], jobPostingId?: string): Promise<CompareResult>;
|
|
30
|
+
}
|
|
31
|
+
export {};
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
const LOYALTY_FACTORS = {
|
|
2
|
+
stable: 1.0,
|
|
3
|
+
moderate: 0.7,
|
|
4
|
+
flight_risk: 0.4,
|
|
5
|
+
};
|
|
6
|
+
export class CompareCandidatesUseCase {
|
|
7
|
+
candidateRepo;
|
|
8
|
+
jobPostingRepo;
|
|
9
|
+
constructor(candidateRepo, jobPostingRepo) {
|
|
10
|
+
this.candidateRepo = candidateRepo;
|
|
11
|
+
this.jobPostingRepo = jobPostingRepo;
|
|
12
|
+
}
|
|
13
|
+
async execute(candidateIds, jobPostingId) {
|
|
14
|
+
if (candidateIds.length < 2 || candidateIds.length > 5) {
|
|
15
|
+
throw new Error('Provide between 2 and 5 candidate IDs');
|
|
16
|
+
}
|
|
17
|
+
const candidates = [];
|
|
18
|
+
for (const id of candidateIds) {
|
|
19
|
+
const candidate = await this.candidateRepo.findById(id);
|
|
20
|
+
if (!candidate) {
|
|
21
|
+
throw new Error(`Candidate not found: ${id}`);
|
|
22
|
+
}
|
|
23
|
+
const firstExp = candidate.experience[0];
|
|
24
|
+
const firstEdu = candidate.education[0];
|
|
25
|
+
candidates.push({
|
|
26
|
+
candidateId: candidate.id,
|
|
27
|
+
name: candidate.name,
|
|
28
|
+
location: candidate.contact.location,
|
|
29
|
+
experienceYears: candidate.experienceYears,
|
|
30
|
+
skills: candidate.skills,
|
|
31
|
+
loyaltyScore: candidate.loyaltyScore,
|
|
32
|
+
avgTenureMonths: candidate.avgTenureMonths,
|
|
33
|
+
shortStintCount: candidate.shortStintCount,
|
|
34
|
+
education: firstEdu
|
|
35
|
+
? [firstEdu.institution, firstEdu.field].filter(Boolean).join(' - ')
|
|
36
|
+
: null,
|
|
37
|
+
currentRole: firstExp
|
|
38
|
+
? `${firstExp.position} @ ${firstExp.company}`
|
|
39
|
+
: null,
|
|
40
|
+
languages: candidate.languages.map((l) => l.language),
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
let jobMatch;
|
|
44
|
+
if (jobPostingId) {
|
|
45
|
+
const jobPosting = await this.jobPostingRepo.findById(jobPostingId);
|
|
46
|
+
if (!jobPosting) {
|
|
47
|
+
throw new Error(`Job posting not found: ${jobPostingId}`);
|
|
48
|
+
}
|
|
49
|
+
const requiredSkillsLower = jobPosting.requiredSkills.map((s) => s.toLowerCase());
|
|
50
|
+
jobMatch = {};
|
|
51
|
+
for (const c of candidates) {
|
|
52
|
+
const candidateSkillsLower = c.skills.map((s) => s.toLowerCase());
|
|
53
|
+
const matchedSkills = [];
|
|
54
|
+
const missingSkills = [];
|
|
55
|
+
for (let i = 0; i < requiredSkillsLower.length; i++) {
|
|
56
|
+
if (candidateSkillsLower.includes(requiredSkillsLower[i])) {
|
|
57
|
+
matchedSkills.push(jobPosting.requiredSkills[i]);
|
|
58
|
+
}
|
|
59
|
+
else {
|
|
60
|
+
missingSkills.push(jobPosting.requiredSkills[i]);
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
const skillOverlap = requiredSkillsLower.length > 0
|
|
64
|
+
? matchedSkills.length / requiredSkillsLower.length
|
|
65
|
+
: 1;
|
|
66
|
+
const loyaltyFactor = LOYALTY_FACTORS[c.loyaltyScore ?? ''] ?? 0.7;
|
|
67
|
+
const compositeScore = 0.3 * skillOverlap + 0.2 * loyaltyFactor;
|
|
68
|
+
jobMatch[c.candidateId] = {
|
|
69
|
+
compositeScore: Math.round(compositeScore * 1000) / 1000,
|
|
70
|
+
matchedSkills,
|
|
71
|
+
missingSkills,
|
|
72
|
+
};
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
return { candidates, job_match: jobMatch };
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
//# sourceMappingURL=compare-candidates.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"compare-candidates.js","sourceRoot":"","sources":["../../../src/application/use-cases/compare-candidates.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import type { IApplicationRepository } from '../../domain/repositories/application-repository.js';
|
|
2
|
+
import type { IApplicationEventRepository } from '../../domain/repositories/application-event-repository.js';
|
|
3
|
+
import type { ICandidateRepository } from '../../domain/repositories/candidate-repository.js';
|
|
4
|
+
import type { IJobPostingRepository } from '../../domain/repositories/job-posting-repository.js';
|
|
5
|
+
interface ReportEntry {
|
|
6
|
+
candidateId: string;
|
|
7
|
+
name: string;
|
|
8
|
+
note: string;
|
|
9
|
+
date: string;
|
|
10
|
+
}
|
|
11
|
+
interface PendingEntry extends ReportEntry {
|
|
12
|
+
status: string;
|
|
13
|
+
}
|
|
14
|
+
export interface ScreeningReport {
|
|
15
|
+
jobPosting: {
|
|
16
|
+
id: string;
|
|
17
|
+
title: string;
|
|
18
|
+
requiredSkills: string[];
|
|
19
|
+
};
|
|
20
|
+
summary: {
|
|
21
|
+
totalScreened: number;
|
|
22
|
+
shortlisted: number;
|
|
23
|
+
screenedOut: number;
|
|
24
|
+
inProgress: number;
|
|
25
|
+
shortlistRate: string;
|
|
26
|
+
};
|
|
27
|
+
shortlisted: ReportEntry[];
|
|
28
|
+
screened_out: ReportEntry[];
|
|
29
|
+
pending: PendingEntry[];
|
|
30
|
+
}
|
|
31
|
+
export declare class GenerateScreeningReportUseCase {
|
|
32
|
+
private applicationRepo;
|
|
33
|
+
private applicationEventRepo;
|
|
34
|
+
private candidateRepo;
|
|
35
|
+
private jobPostingRepo;
|
|
36
|
+
constructor(applicationRepo: IApplicationRepository, applicationEventRepo: IApplicationEventRepository, candidateRepo: ICandidateRepository, jobPostingRepo: IJobPostingRepository);
|
|
37
|
+
execute(jobPostingId: string, since?: string): Promise<ScreeningReport>;
|
|
38
|
+
}
|
|
39
|
+
export {};
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
export class GenerateScreeningReportUseCase {
|
|
2
|
+
applicationRepo;
|
|
3
|
+
applicationEventRepo;
|
|
4
|
+
candidateRepo;
|
|
5
|
+
jobPostingRepo;
|
|
6
|
+
constructor(applicationRepo, applicationEventRepo, candidateRepo, jobPostingRepo) {
|
|
7
|
+
this.applicationRepo = applicationRepo;
|
|
8
|
+
this.applicationEventRepo = applicationEventRepo;
|
|
9
|
+
this.candidateRepo = candidateRepo;
|
|
10
|
+
this.jobPostingRepo = jobPostingRepo;
|
|
11
|
+
}
|
|
12
|
+
async execute(jobPostingId, since) {
|
|
13
|
+
const jobPosting = await this.jobPostingRepo.findById(jobPostingId);
|
|
14
|
+
if (!jobPosting) {
|
|
15
|
+
throw new Error(`Job posting not found: ${jobPostingId}`);
|
|
16
|
+
}
|
|
17
|
+
let applications = await this.applicationRepo.findByJobPostingId(jobPostingId);
|
|
18
|
+
if (since) {
|
|
19
|
+
applications = applications.filter((app) => app.updatedAt >= since);
|
|
20
|
+
}
|
|
21
|
+
const shortlisted = [];
|
|
22
|
+
const screenedOut = [];
|
|
23
|
+
const pending = [];
|
|
24
|
+
for (const app of applications) {
|
|
25
|
+
const candidate = await this.candidateRepo.findById(app.candidateId);
|
|
26
|
+
const candidateName = candidate?.name ?? 'Unknown';
|
|
27
|
+
const events = await this.applicationEventRepo.findByApplicationId(app.id);
|
|
28
|
+
const latestNote = events
|
|
29
|
+
.filter((e) => e.note)
|
|
30
|
+
.sort((a, b) => b.createdAt.localeCompare(a.createdAt))[0];
|
|
31
|
+
const entry = {
|
|
32
|
+
candidateId: app.candidateId,
|
|
33
|
+
name: candidateName,
|
|
34
|
+
note: latestNote?.note ?? app.notes ?? '',
|
|
35
|
+
date: app.updatedAt,
|
|
36
|
+
};
|
|
37
|
+
if (app.status === 'shortlisted') {
|
|
38
|
+
shortlisted.push(entry);
|
|
39
|
+
}
|
|
40
|
+
else if (app.status === 'screened_out') {
|
|
41
|
+
screenedOut.push(entry);
|
|
42
|
+
}
|
|
43
|
+
else {
|
|
44
|
+
pending.push({ ...entry, status: app.status });
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
const totalScreened = applications.length;
|
|
48
|
+
const shortlistRate = totalScreened > 0
|
|
49
|
+
? `${Math.round((shortlisted.length / totalScreened) * 100)}%`
|
|
50
|
+
: '0%';
|
|
51
|
+
return {
|
|
52
|
+
jobPosting: {
|
|
53
|
+
id: jobPosting.id,
|
|
54
|
+
title: jobPosting.title,
|
|
55
|
+
requiredSkills: jobPosting.requiredSkills,
|
|
56
|
+
},
|
|
57
|
+
summary: {
|
|
58
|
+
totalScreened,
|
|
59
|
+
shortlisted: shortlisted.length,
|
|
60
|
+
screenedOut: screenedOut.length,
|
|
61
|
+
inProgress: totalScreened - shortlisted.length - screenedOut.length,
|
|
62
|
+
shortlistRate,
|
|
63
|
+
},
|
|
64
|
+
shortlisted,
|
|
65
|
+
screened_out: screenedOut,
|
|
66
|
+
pending,
|
|
67
|
+
};
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
//# sourceMappingURL=generate-screening-report.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"generate-screening-report.js","sourceRoot":"","sources":["../../../src/application/use-cases/generate-screening-report.ts"],"names":[],"mappings":"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"}
|
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
import type { IChunkRepository } from '../../domain/repositories/chunk-repository.js';
|
|
2
|
+
import type { ICandidateRepository } from '../../domain/repositories/candidate-repository.js';
|
|
2
3
|
import type { SectionType } from '../../shared/types/index.js';
|
|
3
4
|
export interface ChunkDto {
|
|
4
5
|
id: string;
|
|
@@ -8,8 +9,17 @@ export interface ChunkDto {
|
|
|
8
9
|
metadata: Record<string, unknown>;
|
|
9
10
|
createdAt: string;
|
|
10
11
|
}
|
|
12
|
+
export interface BatchChunkResult {
|
|
13
|
+
results: Record<string, {
|
|
14
|
+
name: string;
|
|
15
|
+
chunks: ChunkDto[];
|
|
16
|
+
}>;
|
|
17
|
+
not_found: string[];
|
|
18
|
+
}
|
|
11
19
|
export declare class GetCvChunksUseCase {
|
|
12
20
|
private chunkRepo;
|
|
13
|
-
|
|
21
|
+
private candidateRepo?;
|
|
22
|
+
constructor(chunkRepo: IChunkRepository, candidateRepo?: ICandidateRepository | undefined);
|
|
14
23
|
execute(candidateId: string, sectionType?: SectionType): Promise<ChunkDto[]>;
|
|
24
|
+
executeBatch(candidateIds: string[], sectionType?: SectionType): Promise<BatchChunkResult>;
|
|
15
25
|
}
|
|
@@ -1,7 +1,9 @@
|
|
|
1
1
|
export class GetCvChunksUseCase {
|
|
2
2
|
chunkRepo;
|
|
3
|
-
|
|
3
|
+
candidateRepo;
|
|
4
|
+
constructor(chunkRepo, candidateRepo) {
|
|
4
5
|
this.chunkRepo = chunkRepo;
|
|
6
|
+
this.candidateRepo = candidateRepo;
|
|
5
7
|
}
|
|
6
8
|
async execute(candidateId, sectionType) {
|
|
7
9
|
const chunks = await this.chunkRepo.findByCandidateId(candidateId, sectionType);
|
|
@@ -14,5 +16,35 @@ export class GetCvChunksUseCase {
|
|
|
14
16
|
createdAt: c.createdAt,
|
|
15
17
|
}));
|
|
16
18
|
}
|
|
19
|
+
async executeBatch(candidateIds, sectionType) {
|
|
20
|
+
if (!this.candidateRepo) {
|
|
21
|
+
throw new Error('CandidateRepository is required for batch operations');
|
|
22
|
+
}
|
|
23
|
+
if (candidateIds.length > 10) {
|
|
24
|
+
throw new Error('Maximum 10 candidates per batch request');
|
|
25
|
+
}
|
|
26
|
+
const results = {};
|
|
27
|
+
const notFound = [];
|
|
28
|
+
for (const id of candidateIds) {
|
|
29
|
+
const candidate = await this.candidateRepo.findById(id);
|
|
30
|
+
if (!candidate) {
|
|
31
|
+
notFound.push(id);
|
|
32
|
+
continue;
|
|
33
|
+
}
|
|
34
|
+
const chunks = await this.chunkRepo.findByCandidateId(id, sectionType);
|
|
35
|
+
results[id] = {
|
|
36
|
+
name: candidate.name,
|
|
37
|
+
chunks: chunks.map((c) => ({
|
|
38
|
+
id: c.id,
|
|
39
|
+
candidateId: c.candidateId,
|
|
40
|
+
sectionType: c.sectionType,
|
|
41
|
+
content: c.content,
|
|
42
|
+
metadata: c.metadata,
|
|
43
|
+
createdAt: c.createdAt,
|
|
44
|
+
})),
|
|
45
|
+
};
|
|
46
|
+
}
|
|
47
|
+
return { results, not_found: notFound };
|
|
48
|
+
}
|
|
17
49
|
}
|
|
18
50
|
//# sourceMappingURL=get-cv-chunks.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"get-cv-chunks.js","sourceRoot":"","sources":["../../../src/application/use-cases/get-cv-chunks.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"get-cv-chunks.js","sourceRoot":"","sources":["../../../src/application/use-cases/get-cv-chunks.ts"],"names":[],"mappings":"AAkBA,MAAM,OAAO,kBAAkB;IAEnB;IACA;IAFV,YACU,SAA2B,EAC3B,aAAoC;QADpC,cAAS,GAAT,SAAS,CAAkB;QAC3B,kBAAa,GAAb,aAAa,CAAuB;IAC3C,CAAC;IAEJ,KAAK,CAAC,OAAO,CAAC,WAAmB,EAAE,WAAyB;QAC1D,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,iBAAiB,CAAC,WAAW,EAAE,WAAW,CAAC,CAAC;QAChF,OAAO,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YACxB,EAAE,EAAE,CAAC,CAAC,EAAE;YACR,WAAW,EAAE,CAAC,CAAC,WAAW;YAC1B,WAAW,EAAE,CAAC,CAAC,WAAW;YAC1B,OAAO,EAAE,CAAC,CAAC,OAAO;YAClB,QAAQ,EAAE,CAAC,CAAC,QAAQ;YACpB,SAAS,EAAE,CAAC,CAAC,SAAS;SACvB,CAAC,CAAC,CAAC;IACN,CAAC;IAED,KAAK,CAAC,YAAY,CAAC,YAAsB,EAAE,WAAyB;QAClE,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC;YACxB,MAAM,IAAI,KAAK,CAAC,sDAAsD,CAAC,CAAC;QAC1E,CAAC;QACD,IAAI,YAAY,CAAC,MAAM,GAAG,EAAE,EAAE,CAAC;YAC7B,MAAM,IAAI,KAAK,CAAC,yCAAyC,CAAC,CAAC;QAC7D,CAAC;QAED,MAAM,OAAO,GAAyD,EAAE,CAAC;QACzE,MAAM,QAAQ,GAAa,EAAE,CAAC;QAE9B,KAAK,MAAM,EAAE,IAAI,YAAY,EAAE,CAAC;YAC9B,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;YACxD,IAAI,CAAC,SAAS,EAAE,CAAC;gBACf,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;gBAClB,SAAS;YACX,CAAC;YAED,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,iBAAiB,CAAC,EAAE,EAAE,WAAW,CAAC,CAAC;YACvE,OAAO,CAAC,EAAE,CAAC,GAAG;gBACZ,IAAI,EAAE,SAAS,CAAC,IAAI;gBACpB,MAAM,EAAE,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;oBACzB,EAAE,EAAE,CAAC,CAAC,EAAE;oBACR,WAAW,EAAE,CAAC,CAAC,WAAW;oBAC1B,WAAW,EAAE,CAAC,CAAC,WAAW;oBAC1B,OAAO,EAAE,CAAC,CAAC,OAAO;oBAClB,QAAQ,EAAE,CAAC,CAAC,QAAQ;oBACpB,SAAS,EAAE,CAAC,CAAC,SAAS;iBACvB,CAAC,CAAC;aACJ,CAAC;QACJ,CAAC;QAED,OAAO,EAAE,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,CAAC;IAC1C,CAAC;CACF"}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import type { IApplicationRepository } from '../../domain/repositories/application-repository.js';
|
|
2
|
+
import type { IApplicationEventRepository } from '../../domain/repositories/application-event-repository.js';
|
|
3
|
+
import type { ICandidateRepository } from '../../domain/repositories/candidate-repository.js';
|
|
4
|
+
interface ScreeningEntry {
|
|
5
|
+
candidateId: string;
|
|
6
|
+
candidateName: string;
|
|
7
|
+
note: string;
|
|
8
|
+
date: string;
|
|
9
|
+
}
|
|
10
|
+
interface PipelineEntry extends ScreeningEntry {
|
|
11
|
+
status: string;
|
|
12
|
+
}
|
|
13
|
+
export interface ScreeningHistoryResult {
|
|
14
|
+
shortlisted: ScreeningEntry[];
|
|
15
|
+
screened_out: ScreeningEntry[];
|
|
16
|
+
in_pipeline: PipelineEntry[];
|
|
17
|
+
total_screened: number;
|
|
18
|
+
}
|
|
19
|
+
export declare class GetScreeningHistoryUseCase {
|
|
20
|
+
private applicationRepo;
|
|
21
|
+
private applicationEventRepo;
|
|
22
|
+
private candidateRepo;
|
|
23
|
+
constructor(applicationRepo: IApplicationRepository, applicationEventRepo: IApplicationEventRepository, candidateRepo: ICandidateRepository);
|
|
24
|
+
execute(jobPostingId: string): Promise<ScreeningHistoryResult>;
|
|
25
|
+
}
|
|
26
|
+
export {};
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
export class GetScreeningHistoryUseCase {
|
|
2
|
+
applicationRepo;
|
|
3
|
+
applicationEventRepo;
|
|
4
|
+
candidateRepo;
|
|
5
|
+
constructor(applicationRepo, applicationEventRepo, candidateRepo) {
|
|
6
|
+
this.applicationRepo = applicationRepo;
|
|
7
|
+
this.applicationEventRepo = applicationEventRepo;
|
|
8
|
+
this.candidateRepo = candidateRepo;
|
|
9
|
+
}
|
|
10
|
+
async execute(jobPostingId) {
|
|
11
|
+
const applications = await this.applicationRepo.findByJobPostingId(jobPostingId);
|
|
12
|
+
const shortlisted = [];
|
|
13
|
+
const screenedOut = [];
|
|
14
|
+
const inPipeline = [];
|
|
15
|
+
for (const app of applications) {
|
|
16
|
+
const candidate = await this.candidateRepo.findById(app.candidateId);
|
|
17
|
+
const candidateName = candidate?.name ?? 'Unknown';
|
|
18
|
+
const events = await this.applicationEventRepo.findByApplicationId(app.id);
|
|
19
|
+
const latestNote = events
|
|
20
|
+
.filter((e) => e.note)
|
|
21
|
+
.sort((a, b) => b.createdAt.localeCompare(a.createdAt))[0];
|
|
22
|
+
const entry = {
|
|
23
|
+
candidateId: app.candidateId,
|
|
24
|
+
candidateName,
|
|
25
|
+
note: latestNote?.note ?? app.notes ?? '',
|
|
26
|
+
date: app.updatedAt,
|
|
27
|
+
};
|
|
28
|
+
if (app.status === 'shortlisted') {
|
|
29
|
+
shortlisted.push(entry);
|
|
30
|
+
}
|
|
31
|
+
else if (app.status === 'screened_out') {
|
|
32
|
+
screenedOut.push(entry);
|
|
33
|
+
}
|
|
34
|
+
else {
|
|
35
|
+
inPipeline.push({ ...entry, status: app.status });
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
return {
|
|
39
|
+
shortlisted,
|
|
40
|
+
screened_out: screenedOut,
|
|
41
|
+
in_pipeline: inPipeline,
|
|
42
|
+
total_screened: applications.length,
|
|
43
|
+
};
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
//# sourceMappingURL=get-screening-history.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"get-screening-history.js","sourceRoot":"","sources":["../../../src/application/use-cases/get-screening-history.ts"],"names":[],"mappings":"AAsBA,MAAM,OAAO,0BAA0B;IAE3B;IACA;IACA;IAHV,YACU,eAAuC,EACvC,oBAAiD,EACjD,aAAmC;QAFnC,oBAAe,GAAf,eAAe,CAAwB;QACvC,yBAAoB,GAApB,oBAAoB,CAA6B;QACjD,kBAAa,GAAb,aAAa,CAAsB;IAC1C,CAAC;IAEJ,KAAK,CAAC,OAAO,CAAC,YAAoB;QAChC,MAAM,YAAY,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,kBAAkB,CAAC,YAAY,CAAC,CAAC;QAEjF,MAAM,WAAW,GAAqB,EAAE,CAAC;QACzC,MAAM,WAAW,GAAqB,EAAE,CAAC;QACzC,MAAM,UAAU,GAAoB,EAAE,CAAC;QAEvC,KAAK,MAAM,GAAG,IAAI,YAAY,EAAE,CAAC;YAC/B,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;YACrE,MAAM,aAAa,GAAG,SAAS,EAAE,IAAI,IAAI,SAAS,CAAC;YAEnD,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,oBAAoB,CAAC,mBAAmB,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YAC3E,MAAM,UAAU,GAAG,MAAM;iBACtB,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC;iBACrB,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YAE7D,MAAM,KAAK,GAAG;gBACZ,WAAW,EAAE,GAAG,CAAC,WAAW;gBAC5B,aAAa;gBACb,IAAI,EAAE,UAAU,EAAE,IAAI,IAAI,GAAG,CAAC,KAAK,IAAI,EAAE;gBACzC,IAAI,EAAE,GAAG,CAAC,SAAS;aACpB,CAAC;YAEF,IAAI,GAAG,CAAC,MAAM,KAAK,aAAa,EAAE,CAAC;gBACjC,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC1B,CAAC;iBAAM,IAAI,GAAG,CAAC,MAAM,KAAK,cAAc,EAAE,CAAC;gBACzC,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC1B,CAAC;iBAAM,CAAC;gBACN,UAAU,CAAC,IAAI,CAAC,EAAE,GAAG,KAAK,EAAE,MAAM,EAAE,GAAG,CAAC,MAAM,EAAE,CAAC,CAAC;YACpD,CAAC;QACH,CAAC;QAED,OAAO;YACL,WAAW;YACX,YAAY,EAAE,WAAW;YACzB,WAAW,EAAE,UAAU;YACvB,cAAc,EAAE,YAAY,CAAC,MAAM;SACpC,CAAC;IACJ,CAAC;CACF"}
|
|
@@ -21,3 +21,6 @@ export { SetPendingActionUseCase } from './set-pending-action.js';
|
|
|
21
21
|
export { GetPipelineUseCase } from './get-pipeline.js';
|
|
22
22
|
export { GetCandidateHistoryUseCase } from './get-candidate-history.js';
|
|
23
23
|
export { GetPendingActionsUseCase } from './get-pending-actions.js';
|
|
24
|
+
export { CompareCandidatesUseCase } from './compare-candidates.js';
|
|
25
|
+
export { GetScreeningHistoryUseCase } from './get-screening-history.js';
|
|
26
|
+
export { GenerateScreeningReportUseCase } from './generate-screening-report.js';
|
|
@@ -21,4 +21,7 @@ export { SetPendingActionUseCase } from './set-pending-action.js';
|
|
|
21
21
|
export { GetPipelineUseCase } from './get-pipeline.js';
|
|
22
22
|
export { GetCandidateHistoryUseCase } from './get-candidate-history.js';
|
|
23
23
|
export { GetPendingActionsUseCase } from './get-pending-actions.js';
|
|
24
|
+
export { CompareCandidatesUseCase } from './compare-candidates.js';
|
|
25
|
+
export { GetScreeningHistoryUseCase } from './get-screening-history.js';
|
|
26
|
+
export { GenerateScreeningReportUseCase } from './generate-screening-report.js';
|
|
24
27
|
//# sourceMappingURL=index.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/application/use-cases/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAC3C,OAAO,EAAE,eAAe,EAAE,MAAM,gBAAgB,CAAC;AACjD,OAAO,EAAE,qBAAqB,EAAE,MAAM,sBAAsB,CAAC;AAC7D,OAAO,EAAE,uBAAuB,EAAE,MAAM,wBAAwB,CAAC;AACjE,OAAO,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAC;AACxD,OAAO,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAC;AACxD,OAAO,EAAE,cAAc,EAAE,MAAM,eAAe,CAAC;AAC/C,OAAO,EAAE,eAAe,EAAE,MAAM,gBAAgB,CAAC;AACjD,OAAO,EAAE,iBAAiB,EAAE,MAAM,kBAAkB,CAAC;AACrD,OAAO,EAAE,oBAAoB,EAAE,MAAM,sBAAsB,CAAC;AAC5D,OAAO,EAAE,eAAe,EAAE,MAAM,gBAAgB,CAAC;AACjD,OAAO,EAAE,oBAAoB,EAAE,MAAM,qBAAqB,CAAC;AAC3D,OAAO,EAAE,oBAAoB,EAAE,MAAM,sBAAsB,CAAC;AAC5D,OAAO,EAAE,uBAAuB,EAAE,MAAM,yBAAyB,CAAC;AAClE,OAAO,EAAE,sBAAsB,EAAE,MAAM,wBAAwB,CAAC;AAChE,OAAO,EAAE,sBAAsB,EAAE,MAAM,uBAAuB,CAAC;AAC/D,OAAO,EAAE,oBAAoB,EAAE,MAAM,sBAAsB,CAAC;AAC5D,OAAO,EAAE,2BAA2B,EAAE,MAAM,6BAA6B,CAAC;AAC1E,OAAO,EAAE,sBAAsB,EAAE,MAAM,wBAAwB,CAAC;AAChE,OAAO,EAAE,uBAAuB,EAAE,MAAM,yBAAyB,CAAC;AAClE,OAAO,EAAE,kBAAkB,EAAE,MAAM,mBAAmB,CAAC;AACvD,OAAO,EAAE,0BAA0B,EAAE,MAAM,4BAA4B,CAAC;AACxE,OAAO,EAAE,wBAAwB,EAAE,MAAM,0BAA0B,CAAC"}
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/application/use-cases/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAC3C,OAAO,EAAE,eAAe,EAAE,MAAM,gBAAgB,CAAC;AACjD,OAAO,EAAE,qBAAqB,EAAE,MAAM,sBAAsB,CAAC;AAC7D,OAAO,EAAE,uBAAuB,EAAE,MAAM,wBAAwB,CAAC;AACjE,OAAO,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAC;AACxD,OAAO,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAC;AACxD,OAAO,EAAE,cAAc,EAAE,MAAM,eAAe,CAAC;AAC/C,OAAO,EAAE,eAAe,EAAE,MAAM,gBAAgB,CAAC;AACjD,OAAO,EAAE,iBAAiB,EAAE,MAAM,kBAAkB,CAAC;AACrD,OAAO,EAAE,oBAAoB,EAAE,MAAM,sBAAsB,CAAC;AAC5D,OAAO,EAAE,eAAe,EAAE,MAAM,gBAAgB,CAAC;AACjD,OAAO,EAAE,oBAAoB,EAAE,MAAM,qBAAqB,CAAC;AAC3D,OAAO,EAAE,oBAAoB,EAAE,MAAM,sBAAsB,CAAC;AAC5D,OAAO,EAAE,uBAAuB,EAAE,MAAM,yBAAyB,CAAC;AAClE,OAAO,EAAE,sBAAsB,EAAE,MAAM,wBAAwB,CAAC;AAChE,OAAO,EAAE,sBAAsB,EAAE,MAAM,uBAAuB,CAAC;AAC/D,OAAO,EAAE,oBAAoB,EAAE,MAAM,sBAAsB,CAAC;AAC5D,OAAO,EAAE,2BAA2B,EAAE,MAAM,6BAA6B,CAAC;AAC1E,OAAO,EAAE,sBAAsB,EAAE,MAAM,wBAAwB,CAAC;AAChE,OAAO,EAAE,uBAAuB,EAAE,MAAM,yBAAyB,CAAC;AAClE,OAAO,EAAE,kBAAkB,EAAE,MAAM,mBAAmB,CAAC;AACvD,OAAO,EAAE,0BAA0B,EAAE,MAAM,4BAA4B,CAAC;AACxE,OAAO,EAAE,wBAAwB,EAAE,MAAM,0BAA0B,CAAC;AACpE,OAAO,EAAE,wBAAwB,EAAE,MAAM,yBAAyB,CAAC;AACnE,OAAO,EAAE,0BAA0B,EAAE,MAAM,4BAA4B,CAAC;AACxE,OAAO,EAAE,8BAA8B,EAAE,MAAM,gCAAgC,CAAC"}
|
|
@@ -4,6 +4,15 @@ import type { IChunkRepository } from '../../domain/repositories/chunk-repositor
|
|
|
4
4
|
import type { IApplicationRepository } from '../../domain/repositories/application-repository.js';
|
|
5
5
|
import type { IEmbeddingService } from '../ports/embedding-service.js';
|
|
6
6
|
import type { ApplicationStatus } from '../../domain/entities/application.js';
|
|
7
|
+
export interface MatchCandidatesInput {
|
|
8
|
+
jobPostingId: string;
|
|
9
|
+
limit?: number;
|
|
10
|
+
excludePipeline?: boolean;
|
|
11
|
+
excludeScreened?: boolean;
|
|
12
|
+
location?: string;
|
|
13
|
+
requiredSkills?: string[];
|
|
14
|
+
includeLoyaltyDetail?: boolean;
|
|
15
|
+
}
|
|
7
16
|
export interface MatchResult {
|
|
8
17
|
candidateId: string;
|
|
9
18
|
candidateName: string;
|
|
@@ -14,6 +23,9 @@ export interface MatchResult {
|
|
|
14
23
|
matchedSkills: string[];
|
|
15
24
|
missingSkills: string[];
|
|
16
25
|
pipelineStatus?: ApplicationStatus;
|
|
26
|
+
location?: string;
|
|
27
|
+
avgTenureMonths?: number;
|
|
28
|
+
shortStintCount?: number;
|
|
17
29
|
}
|
|
18
30
|
export declare class MatchCandidatesUseCase {
|
|
19
31
|
private jobPostingRepo;
|
|
@@ -22,6 +34,6 @@ export declare class MatchCandidatesUseCase {
|
|
|
22
34
|
private embeddingService;
|
|
23
35
|
private applicationRepo?;
|
|
24
36
|
constructor(jobPostingRepo: IJobPostingRepository, candidateRepo: ICandidateRepository, chunkRepo: IChunkRepository, embeddingService: IEmbeddingService, applicationRepo?: IApplicationRepository | undefined);
|
|
25
|
-
execute(
|
|
37
|
+
execute(input: MatchCandidatesInput): Promise<MatchResult[]>;
|
|
26
38
|
private computeSkillOverlap;
|
|
27
39
|
}
|
|
@@ -16,7 +16,8 @@ export class MatchCandidatesUseCase {
|
|
|
16
16
|
this.embeddingService = embeddingService;
|
|
17
17
|
this.applicationRepo = applicationRepo;
|
|
18
18
|
}
|
|
19
|
-
async execute(
|
|
19
|
+
async execute(input) {
|
|
20
|
+
const { jobPostingId, limit = 10, excludePipeline = false, excludeScreened = false, location, requiredSkills, includeLoyaltyDetail = false, } = input;
|
|
20
21
|
const jobPosting = await this.jobPostingRepo.findById(jobPostingId);
|
|
21
22
|
if (!jobPosting) {
|
|
22
23
|
throw new Error(`Job posting not found: ${jobPostingId}`);
|
|
@@ -25,7 +26,7 @@ export class MatchCandidatesUseCase {
|
|
|
25
26
|
const searchText = [jobPosting.rawText, jobPosting.description, jobPosting.requirements].filter(Boolean).join('\n');
|
|
26
27
|
const [queryVector] = await this.embeddingService.embedBatch([searchText]);
|
|
27
28
|
// Search chunks (fetch more than needed to aggregate by candidate)
|
|
28
|
-
const chunkLimit = limit *
|
|
29
|
+
const chunkLimit = limit * 8;
|
|
29
30
|
const searchResults = await this.chunkRepo.semanticSearch(queryVector, chunkLimit);
|
|
30
31
|
// Aggregate by candidateId — keep best chunk score per candidate
|
|
31
32
|
const candidateScores = new Map();
|
|
@@ -43,17 +44,36 @@ export class MatchCandidatesUseCase {
|
|
|
43
44
|
const candidate = await this.candidateRepo.findById(candidateId);
|
|
44
45
|
if (!candidate)
|
|
45
46
|
continue;
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
47
|
+
// Location filter
|
|
48
|
+
if (location) {
|
|
49
|
+
const candidateLocation = candidate.contact.location?.toLowerCase() ?? '';
|
|
50
|
+
if (!candidateLocation.includes(location.toLowerCase()))
|
|
51
|
+
continue;
|
|
52
|
+
}
|
|
53
|
+
// Required skills hard filter (case-insensitive exact match)
|
|
54
|
+
if (requiredSkills && requiredSkills.length > 0) {
|
|
55
|
+
const candidateSkillsLower = candidate.skills.map((s) => s.toLowerCase());
|
|
56
|
+
const hasAll = requiredSkills.every((rs) => candidateSkillsLower.includes(rs.toLowerCase()));
|
|
57
|
+
if (!hasAll)
|
|
58
|
+
continue;
|
|
59
|
+
}
|
|
60
|
+
// Pipeline check
|
|
50
61
|
let pipelineStatus;
|
|
51
62
|
if (this.applicationRepo) {
|
|
52
|
-
const existing = await this.applicationRepo.findByJobAndCandidate(
|
|
63
|
+
const existing = await this.applicationRepo.findByJobAndCandidate(jobPostingId, candidateId);
|
|
53
64
|
if (existing) {
|
|
54
65
|
pipelineStatus = existing.status;
|
|
55
66
|
}
|
|
56
67
|
}
|
|
68
|
+
// Exclude filters
|
|
69
|
+
if (excludePipeline && pipelineStatus)
|
|
70
|
+
continue;
|
|
71
|
+
if (excludeScreened && (pipelineStatus === 'shortlisted' || pipelineStatus === 'screened_out'))
|
|
72
|
+
continue;
|
|
73
|
+
// Compute scores
|
|
74
|
+
const { skillOverlap, matchedSkills, missingSkills } = this.computeSkillOverlap(candidate, requiredSkillsLower, jobPosting.requiredSkills);
|
|
75
|
+
const loyaltyFactor = LOYALTY_FACTORS[candidate.loyaltyScore ?? ''] ?? 0.7;
|
|
76
|
+
const compositeScore = 0.5 * vectorScore + 0.3 * skillOverlap + 0.2 * loyaltyFactor;
|
|
57
77
|
results.push({
|
|
58
78
|
candidateId,
|
|
59
79
|
candidateName: candidate.name,
|
|
@@ -64,6 +84,10 @@ export class MatchCandidatesUseCase {
|
|
|
64
84
|
matchedSkills,
|
|
65
85
|
missingSkills,
|
|
66
86
|
pipelineStatus,
|
|
87
|
+
location: candidate.contact.location,
|
|
88
|
+
...(includeLoyaltyDetail
|
|
89
|
+
? { avgTenureMonths: candidate.avgTenureMonths, shortStintCount: candidate.shortStintCount }
|
|
90
|
+
: {}),
|
|
67
91
|
});
|
|
68
92
|
}
|
|
69
93
|
// Sort by composite score descending, return top N
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"match-candidates.js","sourceRoot":"","sources":["../../../src/application/use-cases/match-candidates.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"match-candidates.js","sourceRoot":"","sources":["../../../src/application/use-cases/match-candidates.ts"],"names":[],"mappings":"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"}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
export type ApplicationStatus = 'new' | 'contacted' | 'unreachable' | 'not_interested' | 'interview_scheduled' | 'no_show' | 'interviewed' | 'rejected' | 'offer_sent' | 'hired';
|
|
1
|
+
export type ApplicationStatus = 'new' | 'contacted' | 'unreachable' | 'not_interested' | 'interview_scheduled' | 'no_show' | 'interviewed' | 'rejected' | 'offer_sent' | 'hired' | 'shortlisted' | 'screened_out';
|
|
2
2
|
export interface Application {
|
|
3
3
|
readonly id: string;
|
|
4
4
|
readonly jobPostingId: string;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
<!DOCTYPE html><html><head><meta charset="utf-8"><meta name="viewport" content="width=device-width, initial-scale=1"><link rel="stylesheet" href="/_nuxt/entry.Cjj6Q9Tz.css" crossorigin><link rel="modulepreload" as="script" crossorigin href="/_nuxt/FR49ZCl_.js"><script type="module" src="/_nuxt/FR49ZCl_.js" crossorigin></script><script>"use strict";(()=>{const t=window,e=document.documentElement,c=["dark","light"],n=getStorageValue("localStorage","nuxt-color-mode")||"system";let i=n==="system"?u():n;const r=e.getAttribute("data-color-mode-forced");r&&(i=r),l(i),t["__NUXT_COLOR_MODE__"]={preference:n,value:i,getColorScheme:u,addColorScheme:l,removeColorScheme:d};function l(o){const s=""+o+"",a="";e.classList?e.classList.add(s):e.className+=" "+s,a&&e.setAttribute("data-"+a,o)}function d(o){const s=""+o+"",a="";e.classList?e.classList.remove(s):e.className=e.className.replace(new RegExp(s,"g"),""),a&&e.removeAttribute("data-"+a)}function f(o){return t.matchMedia("(prefers-color-scheme"+o+")")}function u(){if(t.matchMedia&&f("").media!=="not all"){for(const o of c)if(f(":"+o).matches)return o}return"light"}})();function getStorageValue(t,e){switch(t){case"localStorage":return window.localStorage.getItem(e);case"sessionStorage":return window.sessionStorage.getItem(e);case"cookie":return getCookie(e);default:return null}}function getCookie(t){const c=("; "+window.document.cookie).split("; "+t+"=");if(c.length===2)return c.pop()?.split(";").shift()}</script></head><body><div id="__nuxt" class="isolate"></div><div id="teleports"></div><script>window.__NUXT__={};window.__NUXT__.config={public:{},app:{baseURL:"/",buildId:"
|
|
1
|
+
<!DOCTYPE html><html><head><meta charset="utf-8"><meta name="viewport" content="width=device-width, initial-scale=1"><link rel="stylesheet" href="/_nuxt/entry.Cjj6Q9Tz.css" crossorigin><link rel="modulepreload" as="script" crossorigin href="/_nuxt/FR49ZCl_.js"><script type="module" src="/_nuxt/FR49ZCl_.js" crossorigin></script><script>"use strict";(()=>{const t=window,e=document.documentElement,c=["dark","light"],n=getStorageValue("localStorage","nuxt-color-mode")||"system";let i=n==="system"?u():n;const r=e.getAttribute("data-color-mode-forced");r&&(i=r),l(i),t["__NUXT_COLOR_MODE__"]={preference:n,value:i,getColorScheme:u,addColorScheme:l,removeColorScheme:d};function l(o){const s=""+o+"",a="";e.classList?e.classList.add(s):e.className+=" "+s,a&&e.setAttribute("data-"+a,o)}function d(o){const s=""+o+"",a="";e.classList?e.classList.remove(s):e.className=e.className.replace(new RegExp(s,"g"),""),a&&e.removeAttribute("data-"+a)}function f(o){return t.matchMedia("(prefers-color-scheme"+o+")")}function u(){if(t.matchMedia&&f("").media!=="not all"){for(const o of c)if(f(":"+o).matches)return o}return"light"}})();function getStorageValue(t,e){switch(t){case"localStorage":return window.localStorage.getItem(e);case"sessionStorage":return window.sessionStorage.getItem(e);case"cookie":return getCookie(e);default:return null}}function getCookie(t){const c=("; "+window.document.cookie).split("; "+t+"=");if(c.length===2)return c.pop()?.split(";").shift()}</script></head><body><div id="__nuxt" class="isolate"></div><div id="teleports"></div><script>window.__NUXT__={};window.__NUXT__.config={public:{},app:{baseURL:"/",buildId:"b4071b37-04ea-4aad-89c1-665c0a33ff14",buildAssetsDir:"/_nuxt/",cdnURL:""}}</script><script type="application/json" data-nuxt-data="nuxt-app" data-ssr="false" id="__NUXT_DATA__">[{"prerenderedAt":1,"serverRendered":2},1773502792240,false]</script></body></html>
|
|
@@ -1 +1 @@
|
|
|
1
|
-
<!DOCTYPE html><html><head><meta charset="utf-8"><meta name="viewport" content="width=device-width, initial-scale=1"><link rel="stylesheet" href="/_nuxt/entry.Cjj6Q9Tz.css" crossorigin><link rel="modulepreload" as="script" crossorigin href="/_nuxt/FR49ZCl_.js"><script type="module" src="/_nuxt/FR49ZCl_.js" crossorigin></script><script>"use strict";(()=>{const t=window,e=document.documentElement,c=["dark","light"],n=getStorageValue("localStorage","nuxt-color-mode")||"system";let i=n==="system"?u():n;const r=e.getAttribute("data-color-mode-forced");r&&(i=r),l(i),t["__NUXT_COLOR_MODE__"]={preference:n,value:i,getColorScheme:u,addColorScheme:l,removeColorScheme:d};function l(o){const s=""+o+"",a="";e.classList?e.classList.add(s):e.className+=" "+s,a&&e.setAttribute("data-"+a,o)}function d(o){const s=""+o+"",a="";e.classList?e.classList.remove(s):e.className=e.className.replace(new RegExp(s,"g"),""),a&&e.removeAttribute("data-"+a)}function f(o){return t.matchMedia("(prefers-color-scheme"+o+")")}function u(){if(t.matchMedia&&f("").media!=="not all"){for(const o of c)if(f(":"+o).matches)return o}return"light"}})();function getStorageValue(t,e){switch(t){case"localStorage":return window.localStorage.getItem(e);case"sessionStorage":return window.sessionStorage.getItem(e);case"cookie":return getCookie(e);default:return null}}function getCookie(t){const c=("; "+window.document.cookie).split("; "+t+"=");if(c.length===2)return c.pop()?.split(";").shift()}</script></head><body><div id="__nuxt" class="isolate"></div><div id="teleports"></div><script>window.__NUXT__={};window.__NUXT__.config={public:{},app:{baseURL:"/",buildId:"
|
|
1
|
+
<!DOCTYPE html><html><head><meta charset="utf-8"><meta name="viewport" content="width=device-width, initial-scale=1"><link rel="stylesheet" href="/_nuxt/entry.Cjj6Q9Tz.css" crossorigin><link rel="modulepreload" as="script" crossorigin href="/_nuxt/FR49ZCl_.js"><script type="module" src="/_nuxt/FR49ZCl_.js" crossorigin></script><script>"use strict";(()=>{const t=window,e=document.documentElement,c=["dark","light"],n=getStorageValue("localStorage","nuxt-color-mode")||"system";let i=n==="system"?u():n;const r=e.getAttribute("data-color-mode-forced");r&&(i=r),l(i),t["__NUXT_COLOR_MODE__"]={preference:n,value:i,getColorScheme:u,addColorScheme:l,removeColorScheme:d};function l(o){const s=""+o+"",a="";e.classList?e.classList.add(s):e.className+=" "+s,a&&e.setAttribute("data-"+a,o)}function d(o){const s=""+o+"",a="";e.classList?e.classList.remove(s):e.className=e.className.replace(new RegExp(s,"g"),""),a&&e.removeAttribute("data-"+a)}function f(o){return t.matchMedia("(prefers-color-scheme"+o+")")}function u(){if(t.matchMedia&&f("").media!=="not all"){for(const o of c)if(f(":"+o).matches)return o}return"light"}})();function getStorageValue(t,e){switch(t){case"localStorage":return window.localStorage.getItem(e);case"sessionStorage":return window.sessionStorage.getItem(e);case"cookie":return getCookie(e);default:return null}}function getCookie(t){const c=("; "+window.document.cookie).split("; "+t+"=");if(c.length===2)return c.pop()?.split(";").shift()}</script></head><body><div id="__nuxt" class="isolate"></div><div id="teleports"></div><script>window.__NUXT__={};window.__NUXT__.config={public:{},app:{baseURL:"/",buildId:"b4071b37-04ea-4aad-89c1-665c0a33ff14",buildAssetsDir:"/_nuxt/",cdnURL:""}}</script><script type="application/json" data-nuxt-data="nuxt-app" data-ssr="false" id="__NUXT_DATA__">[{"prerenderedAt":1,"serverRendered":2},1773502792240,false]</script></body></html>
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"id":"
|
|
1
|
+
{"id":"b4071b37-04ea-4aad-89c1-665c0a33ff14","timestamp":1773502786903}
|
package/dist/interface/dashboard/public/_nuxt/builds/meta/05323eb6-c50f-4fab-aec8-17a93e7adeb3.json
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"id":"05323eb6-c50f-4fab-aec8-17a93e7adeb3","timestamp":1773490407731,"prerendered":[]}
|
package/dist/interface/dashboard/public/_nuxt/builds/meta/0d6e812f-4393-4ccc-a3d3-5f84e10cef1f.json
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"id":"0d6e812f-4393-4ccc-a3d3-5f84e10cef1f","timestamp":1773490666544,"prerendered":[]}
|
package/dist/interface/dashboard/public/_nuxt/builds/meta/1056fc86-712c-4f08-82e7-b0195da2d885.json
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"id":"1056fc86-712c-4f08-82e7-b0195da2d885","timestamp":1773490635475,"prerendered":[]}
|
package/dist/interface/dashboard/public/_nuxt/builds/meta/287f4183-8718-470c-a30e-21776a882e1d.json
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"id":"287f4183-8718-470c-a30e-21776a882e1d","timestamp":1773490820817,"prerendered":[]}
|
package/dist/interface/dashboard/public/_nuxt/builds/meta/47653a05-51ca-47b0-8728-ea69a473e926.json
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"id":"47653a05-51ca-47b0-8728-ea69a473e926","timestamp":1773490592780,"prerendered":[]}
|
package/dist/interface/dashboard/public/_nuxt/builds/meta/4c1f6063-4cfa-4abb-8f92-2172e917f88b.json
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"id":"4c1f6063-4cfa-4abb-8f92-2172e917f88b","timestamp":1773490345234,"prerendered":[]}
|
package/dist/interface/dashboard/public/_nuxt/builds/meta/76aaca87-0e82-4fa7-a19a-155439e56038.json
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"id":"76aaca87-0e82-4fa7-a19a-155439e56038","timestamp":1773490735297,"prerendered":[]}
|
package/dist/interface/dashboard/public/_nuxt/builds/meta/8d10e489-6195-4ab5-9a19-ac5b6bfb3e21.json
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"id":"8d10e489-6195-4ab5-9a19-ac5b6bfb3e21","timestamp":1773490512318,"prerendered":[]}
|
package/dist/interface/dashboard/public/_nuxt/builds/meta/8ede37fe-3543-45b5-84a3-21350596eef7.json
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"id":"8ede37fe-3543-45b5-84a3-21350596eef7","timestamp":1773490464375,"prerendered":[]}
|
package/dist/interface/dashboard/public/_nuxt/builds/meta/b4071b37-04ea-4aad-89c1-665c0a33ff14.json
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"id":"b4071b37-04ea-4aad-89c1-665c0a33ff14","timestamp":1773502786903,"prerendered":[]}
|
package/dist/interface/dashboard/public/_nuxt/builds/meta/f020f3fc-f5ab-48bf-a28d-516004359cc2.json
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"id":"f020f3fc-f5ab-48bf-a28d-516004359cc2","timestamp":1773490549881,"prerendered":[]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
<!DOCTYPE html><html><head><meta charset="utf-8"><meta name="viewport" content="width=device-width, initial-scale=1"><link rel="stylesheet" href="/_nuxt/entry.Cjj6Q9Tz.css" crossorigin><link rel="modulepreload" as="script" crossorigin href="/_nuxt/FR49ZCl_.js"><script type="module" src="/_nuxt/FR49ZCl_.js" crossorigin></script><script>"use strict";(()=>{const t=window,e=document.documentElement,c=["dark","light"],n=getStorageValue("localStorage","nuxt-color-mode")||"system";let i=n==="system"?u():n;const r=e.getAttribute("data-color-mode-forced");r&&(i=r),l(i),t["__NUXT_COLOR_MODE__"]={preference:n,value:i,getColorScheme:u,addColorScheme:l,removeColorScheme:d};function l(o){const s=""+o+"",a="";e.classList?e.classList.add(s):e.className+=" "+s,a&&e.setAttribute("data-"+a,o)}function d(o){const s=""+o+"",a="";e.classList?e.classList.remove(s):e.className=e.className.replace(new RegExp(s,"g"),""),a&&e.removeAttribute("data-"+a)}function f(o){return t.matchMedia("(prefers-color-scheme"+o+")")}function u(){if(t.matchMedia&&f("").media!=="not all"){for(const o of c)if(f(":"+o).matches)return o}return"light"}})();function getStorageValue(t,e){switch(t){case"localStorage":return window.localStorage.getItem(e);case"sessionStorage":return window.sessionStorage.getItem(e);case"cookie":return getCookie(e);default:return null}}function getCookie(t){const c=("; "+window.document.cookie).split("; "+t+"=");if(c.length===2)return c.pop()?.split(";").shift()}</script></head><body><div id="__nuxt" class="isolate"></div><div id="teleports"></div><script>window.__NUXT__={};window.__NUXT__.config={public:{},app:{baseURL:"/",buildId:"
|
|
1
|
+
<!DOCTYPE html><html><head><meta charset="utf-8"><meta name="viewport" content="width=device-width, initial-scale=1"><link rel="stylesheet" href="/_nuxt/entry.Cjj6Q9Tz.css" crossorigin><link rel="modulepreload" as="script" crossorigin href="/_nuxt/FR49ZCl_.js"><script type="module" src="/_nuxt/FR49ZCl_.js" crossorigin></script><script>"use strict";(()=>{const t=window,e=document.documentElement,c=["dark","light"],n=getStorageValue("localStorage","nuxt-color-mode")||"system";let i=n==="system"?u():n;const r=e.getAttribute("data-color-mode-forced");r&&(i=r),l(i),t["__NUXT_COLOR_MODE__"]={preference:n,value:i,getColorScheme:u,addColorScheme:l,removeColorScheme:d};function l(o){const s=""+o+"",a="";e.classList?e.classList.add(s):e.className+=" "+s,a&&e.setAttribute("data-"+a,o)}function d(o){const s=""+o+"",a="";e.classList?e.classList.remove(s):e.className=e.className.replace(new RegExp(s,"g"),""),a&&e.removeAttribute("data-"+a)}function f(o){return t.matchMedia("(prefers-color-scheme"+o+")")}function u(){if(t.matchMedia&&f("").media!=="not all"){for(const o of c)if(f(":"+o).matches)return o}return"light"}})();function getStorageValue(t,e){switch(t){case"localStorage":return window.localStorage.getItem(e);case"sessionStorage":return window.sessionStorage.getItem(e);case"cookie":return getCookie(e);default:return null}}function getCookie(t){const c=("; "+window.document.cookie).split("; "+t+"=");if(c.length===2)return c.pop()?.split(";").shift()}</script></head><body><div id="__nuxt" class="isolate"></div><div id="teleports"></div><script>window.__NUXT__={};window.__NUXT__.config={public:{},app:{baseURL:"/",buildId:"b4071b37-04ea-4aad-89c1-665c0a33ff14",buildAssetsDir:"/_nuxt/",cdnURL:""}}</script><script type="application/json" data-nuxt-data="nuxt-app" data-ssr="false" id="__NUXT_DATA__">[{"prerenderedAt":1,"serverRendered":2},1773502792240,false]</script></body></html>
|
|
@@ -44,6 +44,9 @@ import { registerGetCandidateHistoryTool } from './tools/get-candidate-history.j
|
|
|
44
44
|
import { registerGetPendingActionsTool } from './tools/get-pending-actions.js';
|
|
45
45
|
import { registerGetPipelineCandidatesTool } from './tools/get-pipeline-candidates.js';
|
|
46
46
|
import { registerCheckDuplicateCandidateTool } from './tools/check-duplicate-candidate.js';
|
|
47
|
+
import { registerCompareCandidatesTool } from './tools/compare-candidates.js';
|
|
48
|
+
import { registerGetScreeningHistoryTool } from './tools/get-screening-history.js';
|
|
49
|
+
import { registerGenerateScreeningReportTool } from './tools/generate-screening-report.js';
|
|
47
50
|
export async function createServer() {
|
|
48
51
|
const config = loadConfig();
|
|
49
52
|
configureVectorDimension(config.embeddingModel);
|
|
@@ -72,7 +75,7 @@ export async function createServer() {
|
|
|
72
75
|
};
|
|
73
76
|
const server = new McpServer({
|
|
74
77
|
name: 'hirebase',
|
|
75
|
-
version: '1.
|
|
78
|
+
version: '1.2.0',
|
|
76
79
|
});
|
|
77
80
|
registerAddCvTool(server, deps);
|
|
78
81
|
registerUpdateCvTool(server, deps);
|
|
@@ -103,6 +106,9 @@ export async function createServer() {
|
|
|
103
106
|
registerGetPendingActionsTool(server, deps);
|
|
104
107
|
registerGetPipelineCandidatesTool(server, deps);
|
|
105
108
|
registerCheckDuplicateCandidateTool(server, deps);
|
|
109
|
+
registerCompareCandidatesTool(server, deps);
|
|
110
|
+
registerGetScreeningHistoryTool(server, deps);
|
|
111
|
+
registerGenerateScreeningReportTool(server, deps);
|
|
106
112
|
if (config.dashboardEnabled) {
|
|
107
113
|
const eventBus = new DashboardEventBus();
|
|
108
114
|
wrapAllRegisteredTools(server, eventBus, deps);
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server.js","sourceRoot":"","sources":["../../../src/interface/mcp/server.ts"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"server.js","sourceRoot":"","sources":["../../../src/interface/mcp/server.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
import { z } from 'zod/v4';
|
|
2
|
+
import { CompareCandidatesUseCase } from '../../../application/use-cases/compare-candidates.js';
|
|
3
|
+
export function registerCompareCandidatesTool(server, deps) {
|
|
4
|
+
const useCase = new CompareCandidatesUseCase(deps.candidateRepo, deps.jobPostingRepo);
|
|
5
|
+
server.tool('compare_candidates', 'Compare 2-5 candidates side by side. Optionally provide a job posting to include skill match scores.', {
|
|
6
|
+
candidate_ids: z.array(z.string()).describe('IDs of candidates to compare (2-5)'),
|
|
7
|
+
job_posting_id: z.string().optional().describe('Job posting ID for skill match scoring'),
|
|
8
|
+
}, async ({ candidate_ids, job_posting_id }) => {
|
|
9
|
+
try {
|
|
10
|
+
const result = await useCase.execute(candidate_ids, job_posting_id);
|
|
11
|
+
return {
|
|
12
|
+
content: [{ type: 'text', text: JSON.stringify(result, null, 2) }],
|
|
13
|
+
};
|
|
14
|
+
}
|
|
15
|
+
catch (error) {
|
|
16
|
+
return {
|
|
17
|
+
content: [{ type: 'text', text: JSON.stringify({ error: error.message }) }],
|
|
18
|
+
isError: true,
|
|
19
|
+
};
|
|
20
|
+
}
|
|
21
|
+
});
|
|
22
|
+
}
|
|
23
|
+
//# sourceMappingURL=compare-candidates.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"compare-candidates.js","sourceRoot":"","sources":["../../../../src/interface/mcp/tools/compare-candidates.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,CAAC,EAAE,MAAM,QAAQ,CAAC;AAC3B,OAAO,EAAE,wBAAwB,EAAE,MAAM,sDAAsD,CAAC;AAGhG,MAAM,UAAU,6BAA6B,CAAC,MAAiB,EAAE,IAAkB;IACjF,MAAM,OAAO,GAAG,IAAI,wBAAwB,CAAC,IAAI,CAAC,aAAa,EAAE,IAAI,CAAC,cAAc,CAAC,CAAC;IAEtF,MAAM,CAAC,IAAI,CACT,oBAAoB,EACpB,sGAAsG,EACtG;QACE,aAAa,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,QAAQ,CAAC,oCAAoC,CAAC;QACjF,cAAc,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,wCAAwC,CAAC;KACzF,EACD,KAAK,EAAE,EAAE,aAAa,EAAE,cAAc,EAAE,EAAE,EAAE;QAC1C,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,OAAO,CAAC,aAAa,EAAE,cAAc,CAAC,CAAC;YACpE,OAAO;gBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC;aAC5E,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO;gBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAG,KAAe,CAAC,OAAO,EAAE,CAAC,EAAE,CAAC;gBAC/F,OAAO,EAAE,IAAI;aACd,CAAC;QACJ,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
import { z } from 'zod/v4';
|
|
2
|
+
import { GenerateScreeningReportUseCase } from '../../../application/use-cases/generate-screening-report.js';
|
|
3
|
+
export function registerGenerateScreeningReportTool(server, deps) {
|
|
4
|
+
const useCase = new GenerateScreeningReportUseCase(deps.applicationRepo, deps.applicationEventRepo, deps.candidateRepo, deps.jobPostingRepo);
|
|
5
|
+
server.tool('generate_screening_report', 'Generate a screening summary report for a job posting — includes statistics, shortlisted, screened out, and pending candidates.', {
|
|
6
|
+
job_posting_id: z.string().describe('ID of the job posting'),
|
|
7
|
+
since: z.string().optional().describe('ISO date — only include screenings updated after this date'),
|
|
8
|
+
}, async ({ job_posting_id, since }) => {
|
|
9
|
+
try {
|
|
10
|
+
const result = await useCase.execute(job_posting_id, since);
|
|
11
|
+
return {
|
|
12
|
+
content: [{ type: 'text', text: JSON.stringify(result, null, 2) }],
|
|
13
|
+
};
|
|
14
|
+
}
|
|
15
|
+
catch (error) {
|
|
16
|
+
return {
|
|
17
|
+
content: [{ type: 'text', text: JSON.stringify({ error: error.message }) }],
|
|
18
|
+
isError: true,
|
|
19
|
+
};
|
|
20
|
+
}
|
|
21
|
+
});
|
|
22
|
+
}
|
|
23
|
+
//# sourceMappingURL=generate-screening-report.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"generate-screening-report.js","sourceRoot":"","sources":["../../../../src/interface/mcp/tools/generate-screening-report.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,CAAC,EAAE,MAAM,QAAQ,CAAC;AAC3B,OAAO,EAAE,8BAA8B,EAAE,MAAM,6DAA6D,CAAC;AAG7G,MAAM,UAAU,mCAAmC,CAAC,MAAiB,EAAE,IAAkB;IACvF,MAAM,OAAO,GAAG,IAAI,8BAA8B,CAChD,IAAI,CAAC,eAAe,EACpB,IAAI,CAAC,oBAAoB,EACzB,IAAI,CAAC,aAAa,EAClB,IAAI,CAAC,cAAc,CACpB,CAAC;IAEF,MAAM,CAAC,IAAI,CACT,2BAA2B,EAC3B,iIAAiI,EACjI;QACE,cAAc,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,uBAAuB,CAAC;QAC5D,KAAK,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,4DAA4D,CAAC;KACpG,EACD,KAAK,EAAE,EAAE,cAAc,EAAE,KAAK,EAAE,EAAE,EAAE;QAClC,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,OAAO,CAAC,cAAc,EAAE,KAAK,CAAC,CAAC;YAC5D,OAAO;gBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC;aAC5E,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO;gBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAG,KAAe,CAAC,OAAO,EAAE,CAAC,EAAE,CAAC;gBAC/F,OAAO,EAAE,IAAI;aACd,CAAC;QACJ,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
|
|
@@ -1,15 +1,34 @@
|
|
|
1
1
|
import { z } from 'zod/v4';
|
|
2
2
|
import { GetCvChunksUseCase } from '../../../application/use-cases/get-cv-chunks.js';
|
|
3
3
|
export function registerGetCvChunksTool(server, deps) {
|
|
4
|
-
const useCase = new GetCvChunksUseCase(deps.chunkRepo);
|
|
5
|
-
server.tool('get_cv_chunks', 'Get the text chunks (sections) of
|
|
6
|
-
candidate_id: z.string().describe('ID of
|
|
4
|
+
const useCase = new GetCvChunksUseCase(deps.chunkRepo, deps.candidateRepo);
|
|
5
|
+
server.tool('get_cv_chunks', 'Get the text chunks (sections) of one or more candidates. Use candidate_id for single, candidate_ids for batch (max 10).', {
|
|
6
|
+
candidate_id: z.string().optional().describe('ID of a single candidate'),
|
|
7
|
+
candidate_ids: z.array(z.string()).optional().describe('IDs of multiple candidates (max 10, batch mode)'),
|
|
7
8
|
section_type: z
|
|
8
9
|
.enum(['summary', 'experience', 'education', 'skills', 'projects', 'certifications', 'full'])
|
|
9
10
|
.optional()
|
|
10
11
|
.describe('Filter to a specific section type'),
|
|
11
|
-
}, async ({ candidate_id, section_type }) => {
|
|
12
|
+
}, async ({ candidate_id, candidate_ids, section_type }) => {
|
|
12
13
|
try {
|
|
14
|
+
if (candidate_id && candidate_ids) {
|
|
15
|
+
return {
|
|
16
|
+
content: [{ type: 'text', text: JSON.stringify({ error: 'Provide either candidate_id or candidate_ids, not both' }) }],
|
|
17
|
+
isError: true,
|
|
18
|
+
};
|
|
19
|
+
}
|
|
20
|
+
if (!candidate_id && !candidate_ids) {
|
|
21
|
+
return {
|
|
22
|
+
content: [{ type: 'text', text: JSON.stringify({ error: 'Provide either candidate_id or candidate_ids' }) }],
|
|
23
|
+
isError: true,
|
|
24
|
+
};
|
|
25
|
+
}
|
|
26
|
+
if (candidate_ids) {
|
|
27
|
+
const result = await useCase.executeBatch(candidate_ids, section_type);
|
|
28
|
+
return {
|
|
29
|
+
content: [{ type: 'text', text: JSON.stringify(result, null, 2) }],
|
|
30
|
+
};
|
|
31
|
+
}
|
|
13
32
|
const chunks = await useCase.execute(candidate_id, section_type);
|
|
14
33
|
return {
|
|
15
34
|
content: [{ type: 'text', text: JSON.stringify(chunks, null, 2) }],
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"get-cv-chunks.js","sourceRoot":"","sources":["../../../../src/interface/mcp/tools/get-cv-chunks.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,CAAC,EAAE,MAAM,QAAQ,CAAC;AAC3B,OAAO,EAAE,kBAAkB,EAAE,MAAM,iDAAiD,CAAC;AAIrF,MAAM,UAAU,uBAAuB,CAAC,MAAiB,EAAE,IAAkB;IAC3E,MAAM,OAAO,GAAG,IAAI,kBAAkB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;
|
|
1
|
+
{"version":3,"file":"get-cv-chunks.js","sourceRoot":"","sources":["../../../../src/interface/mcp/tools/get-cv-chunks.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,CAAC,EAAE,MAAM,QAAQ,CAAC;AAC3B,OAAO,EAAE,kBAAkB,EAAE,MAAM,iDAAiD,CAAC;AAIrF,MAAM,UAAU,uBAAuB,CAAC,MAAiB,EAAE,IAAkB;IAC3E,MAAM,OAAO,GAAG,IAAI,kBAAkB,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,aAAa,CAAC,CAAC;IAE3E,MAAM,CAAC,IAAI,CACT,eAAe,EACf,0HAA0H,EAC1H;QACE,YAAY,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,0BAA0B,CAAC;QACxE,aAAa,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,iDAAiD,CAAC;QACzG,YAAY,EAAE,CAAC;aACZ,IAAI,CAAC,CAAC,SAAS,EAAE,YAAY,EAAE,WAAW,EAAE,QAAQ,EAAE,UAAU,EAAE,gBAAgB,EAAE,MAAM,CAAC,CAAC;aAC5F,QAAQ,EAAE;aACV,QAAQ,CAAC,mCAAmC,CAAC;KACjD,EACD,KAAK,EAAE,EAAE,YAAY,EAAE,aAAa,EAAE,YAAY,EAAE,EAAE,EAAE;QACtD,IAAI,CAAC;YACH,IAAI,YAAY,IAAI,aAAa,EAAE,CAAC;gBAClC,OAAO;oBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,wDAAwD,EAAE,CAAC,EAAE,CAAC;oBAC/H,OAAO,EAAE,IAAI;iBACd,CAAC;YACJ,CAAC;YACD,IAAI,CAAC,YAAY,IAAI,CAAC,aAAa,EAAE,CAAC;gBACpC,OAAO;oBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,8CAA8C,EAAE,CAAC,EAAE,CAAC;oBACrH,OAAO,EAAE,IAAI;iBACd,CAAC;YACJ,CAAC;YAED,IAAI,aAAa,EAAE,CAAC;gBAClB,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,YAAY,CAAC,aAAa,EAAE,YAAuC,CAAC,CAAC;gBAClG,OAAO;oBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC;iBAC5E,CAAC;YACJ,CAAC;YAED,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,OAAO,CAAC,YAAa,EAAE,YAAuC,CAAC,CAAC;YAC7F,OAAO;gBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC;aAC5E,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO;gBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAG,KAAe,CAAC,OAAO,EAAE,CAAC,EAAE,CAAC;gBAC/F,OAAO,EAAE,IAAI;aACd,CAAC;QACJ,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import { z } from 'zod/v4';
|
|
2
|
+
import { GetScreeningHistoryUseCase } from '../../../application/use-cases/get-screening-history.js';
|
|
3
|
+
export function registerGetScreeningHistoryTool(server, deps) {
|
|
4
|
+
const useCase = new GetScreeningHistoryUseCase(deps.applicationRepo, deps.applicationEventRepo, deps.candidateRepo);
|
|
5
|
+
server.tool('get_screening_history', 'Get the screening history for a job posting — lists shortlisted, screened out, and in-pipeline candidates.', {
|
|
6
|
+
job_posting_id: z.string().describe('ID of the job posting'),
|
|
7
|
+
}, async ({ job_posting_id }) => {
|
|
8
|
+
try {
|
|
9
|
+
const result = await useCase.execute(job_posting_id);
|
|
10
|
+
return {
|
|
11
|
+
content: [{ type: 'text', text: JSON.stringify(result, null, 2) }],
|
|
12
|
+
};
|
|
13
|
+
}
|
|
14
|
+
catch (error) {
|
|
15
|
+
return {
|
|
16
|
+
content: [{ type: 'text', text: JSON.stringify({ error: error.message }) }],
|
|
17
|
+
isError: true,
|
|
18
|
+
};
|
|
19
|
+
}
|
|
20
|
+
});
|
|
21
|
+
}
|
|
22
|
+
//# sourceMappingURL=get-screening-history.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"get-screening-history.js","sourceRoot":"","sources":["../../../../src/interface/mcp/tools/get-screening-history.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,CAAC,EAAE,MAAM,QAAQ,CAAC;AAC3B,OAAO,EAAE,0BAA0B,EAAE,MAAM,yDAAyD,CAAC;AAGrG,MAAM,UAAU,+BAA+B,CAAC,MAAiB,EAAE,IAAkB;IACnF,MAAM,OAAO,GAAG,IAAI,0BAA0B,CAC5C,IAAI,CAAC,eAAe,EACpB,IAAI,CAAC,oBAAoB,EACzB,IAAI,CAAC,aAAa,CACnB,CAAC;IAEF,MAAM,CAAC,IAAI,CACT,uBAAuB,EACvB,4GAA4G,EAC5G;QACE,cAAc,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,uBAAuB,CAAC;KAC7D,EACD,KAAK,EAAE,EAAE,cAAc,EAAE,EAAE,EAAE;QAC3B,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YACrD,OAAO;gBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC;aAC5E,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO;gBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAG,KAAe,CAAC,OAAO,EAAE,CAAC,EAAE,CAAC;gBAC/F,OAAO,EAAE,IAAI;aACd,CAAC;QACJ,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
|
|
@@ -2,12 +2,25 @@ import { z } from 'zod/v4';
|
|
|
2
2
|
import { MatchCandidatesUseCase } from '../../../application/use-cases/match-candidates.js';
|
|
3
3
|
export function registerMatchCandidatesTool(server, deps) {
|
|
4
4
|
const useCase = new MatchCandidatesUseCase(deps.jobPostingRepo, deps.candidateRepo, deps.chunkRepo, deps.embeddingService, deps.applicationRepo);
|
|
5
|
-
server.tool('match_candidates', 'Find the best matching candidates for a job posting using semantic search, skill overlap, and loyalty scoring.
|
|
5
|
+
server.tool('match_candidates', 'Find the best matching candidates for a job posting using semantic search, skill overlap, and loyalty scoring. Supports filtering by location, required skills, and pipeline status exclusion.', {
|
|
6
6
|
job_posting_id: z.string().describe('ID of the job posting to match against'),
|
|
7
7
|
limit: z.number().optional().describe('Maximum number of candidates to return (default: 10)'),
|
|
8
|
-
|
|
8
|
+
exclude_pipeline: z.boolean().optional().describe('Exclude candidates already in the pipeline (default: false)'),
|
|
9
|
+
exclude_screened: z.boolean().optional().describe('Exclude shortlisted + screened_out candidates (default: false)'),
|
|
10
|
+
location: z.string().optional().describe('Filter by location (partial match, e.g. "Ankara")'),
|
|
11
|
+
required_skills: z.array(z.string()).optional().describe('Hard filter: candidate must have ALL these skills (case-insensitive exact match)'),
|
|
12
|
+
include_loyalty_detail: z.boolean().optional().describe('Include avgTenureMonths and shortStintCount in results (default: false)'),
|
|
13
|
+
}, async ({ job_posting_id, limit, exclude_pipeline, exclude_screened, location, required_skills, include_loyalty_detail }) => {
|
|
9
14
|
try {
|
|
10
|
-
const results = await useCase.execute(
|
|
15
|
+
const results = await useCase.execute({
|
|
16
|
+
jobPostingId: job_posting_id,
|
|
17
|
+
limit,
|
|
18
|
+
excludePipeline: exclude_pipeline,
|
|
19
|
+
excludeScreened: exclude_screened,
|
|
20
|
+
location,
|
|
21
|
+
requiredSkills: required_skills,
|
|
22
|
+
includeLoyaltyDetail: include_loyalty_detail,
|
|
23
|
+
});
|
|
11
24
|
return {
|
|
12
25
|
content: [{ type: 'text', text: JSON.stringify(results, null, 2) }],
|
|
13
26
|
};
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"match-candidates.js","sourceRoot":"","sources":["../../../../src/interface/mcp/tools/match-candidates.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,CAAC,EAAE,MAAM,QAAQ,CAAC;AAC3B,OAAO,EAAE,sBAAsB,EAAE,MAAM,oDAAoD,CAAC;AAG5F,MAAM,UAAU,2BAA2B,CAAC,MAAiB,EAAE,IAAkB;IAC/E,MAAM,OAAO,GAAG,IAAI,sBAAsB,CACxC,IAAI,CAAC,cAAc,EACnB,IAAI,CAAC,aAAa,EAClB,IAAI,CAAC,SAAS,EACd,IAAI,CAAC,gBAAgB,EACrB,IAAI,CAAC,eAAe,CACrB,CAAC;IAEF,MAAM,CAAC,IAAI,CACT,kBAAkB,EAClB,
|
|
1
|
+
{"version":3,"file":"match-candidates.js","sourceRoot":"","sources":["../../../../src/interface/mcp/tools/match-candidates.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,CAAC,EAAE,MAAM,QAAQ,CAAC;AAC3B,OAAO,EAAE,sBAAsB,EAAE,MAAM,oDAAoD,CAAC;AAG5F,MAAM,UAAU,2BAA2B,CAAC,MAAiB,EAAE,IAAkB;IAC/E,MAAM,OAAO,GAAG,IAAI,sBAAsB,CACxC,IAAI,CAAC,cAAc,EACnB,IAAI,CAAC,aAAa,EAClB,IAAI,CAAC,SAAS,EACd,IAAI,CAAC,gBAAgB,EACrB,IAAI,CAAC,eAAe,CACrB,CAAC;IAEF,MAAM,CAAC,IAAI,CACT,kBAAkB,EAClB,gMAAgM,EAChM;QACE,cAAc,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,wCAAwC,CAAC;QAC7E,KAAK,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,sDAAsD,CAAC;QAC7F,gBAAgB,EAAE,CAAC,CAAC,OAAO,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,6DAA6D,CAAC;QAChH,gBAAgB,EAAE,CAAC,CAAC,OAAO,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,gEAAgE,CAAC;QACnH,QAAQ,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,mDAAmD,CAAC;QAC7F,eAAe,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,kFAAkF,CAAC;QAC5I,sBAAsB,EAAE,CAAC,CAAC,OAAO,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,yEAAyE,CAAC;KACnI,EACD,KAAK,EAAE,EAAE,cAAc,EAAE,KAAK,EAAE,gBAAgB,EAAE,gBAAgB,EAAE,QAAQ,EAAE,eAAe,EAAE,sBAAsB,EAAE,EAAE,EAAE;QACzH,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,OAAO,CAAC;gBACpC,YAAY,EAAE,cAAc;gBAC5B,KAAK;gBACL,eAAe,EAAE,gBAAgB;gBACjC,eAAe,EAAE,gBAAgB;gBACjC,QAAQ;gBACR,cAAc,EAAE,eAAe;gBAC/B,oBAAoB,EAAE,sBAAsB;aAC7C,CAAC,CAAC;YACH,OAAO;gBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC;aAC7E,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO;gBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAG,KAAe,CAAC,OAAO,EAAE,CAAC,EAAE,CAAC;gBAC/F,OAAO,EAAE,IAAI;aACd,CAAC;QACJ,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
|
|
@@ -2,7 +2,7 @@ import { z } from 'zod/v4';
|
|
|
2
2
|
import { UpdatePipelineStatusUseCase } from '../../../application/use-cases/update-pipeline-status.js';
|
|
3
3
|
export function registerUpdatePipelineStatusTool(server, deps) {
|
|
4
4
|
const useCase = new UpdatePipelineStatusUseCase(deps.applicationRepo, deps.applicationEventRepo);
|
|
5
|
-
server.tool('update_pipeline_status', 'Update the status of a candidate in the recruitment pipeline. Statuses: new, contacted, unreachable, not_interested, interview_scheduled, no_show, interviewed, rejected, offer_sent, hired.', {
|
|
5
|
+
server.tool('update_pipeline_status', 'Update the status of a candidate in the recruitment pipeline. Statuses: new, contacted, unreachable, not_interested, interview_scheduled, no_show, interviewed, rejected, offer_sent, hired, shortlisted, screened_out.', {
|
|
6
6
|
application_id: z.string().describe('ID of the application'),
|
|
7
7
|
new_status: z
|
|
8
8
|
.enum([
|
|
@@ -16,6 +16,8 @@ export function registerUpdatePipelineStatusTool(server, deps) {
|
|
|
16
16
|
'rejected',
|
|
17
17
|
'offer_sent',
|
|
18
18
|
'hired',
|
|
19
|
+
'shortlisted',
|
|
20
|
+
'screened_out',
|
|
19
21
|
])
|
|
20
22
|
.describe('New status'),
|
|
21
23
|
note: z.string().optional().describe('Optional note about the status change'),
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"update-pipeline-status.js","sourceRoot":"","sources":["../../../../src/interface/mcp/tools/update-pipeline-status.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,CAAC,EAAE,MAAM,QAAQ,CAAC;AAC3B,OAAO,EAAE,2BAA2B,EAAE,MAAM,0DAA0D,CAAC;AAGvG,MAAM,UAAU,gCAAgC,CAAC,MAAiB,EAAE,IAAkB;IACpF,MAAM,OAAO,GAAG,IAAI,2BAA2B,CAAC,IAAI,CAAC,eAAe,EAAE,IAAI,CAAC,oBAAoB,CAAC,CAAC;IAEjG,MAAM,CAAC,IAAI,CACT,wBAAwB,EACxB,
|
|
1
|
+
{"version":3,"file":"update-pipeline-status.js","sourceRoot":"","sources":["../../../../src/interface/mcp/tools/update-pipeline-status.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,CAAC,EAAE,MAAM,QAAQ,CAAC;AAC3B,OAAO,EAAE,2BAA2B,EAAE,MAAM,0DAA0D,CAAC;AAGvG,MAAM,UAAU,gCAAgC,CAAC,MAAiB,EAAE,IAAkB;IACpF,MAAM,OAAO,GAAG,IAAI,2BAA2B,CAAC,IAAI,CAAC,eAAe,EAAE,IAAI,CAAC,oBAAoB,CAAC,CAAC;IAEjG,MAAM,CAAC,IAAI,CACT,wBAAwB,EACxB,yNAAyN,EACzN;QACE,cAAc,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,uBAAuB,CAAC;QAC5D,UAAU,EAAE,CAAC;aACV,IAAI,CAAC;YACJ,KAAK;YACL,WAAW;YACX,aAAa;YACb,gBAAgB;YAChB,qBAAqB;YACrB,SAAS;YACT,aAAa;YACb,UAAU;YACV,YAAY;YACZ,OAAO;YACP,aAAa;YACb,cAAc;SACf,CAAC;aACD,QAAQ,CAAC,YAAY,CAAC;QACzB,IAAI,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,uCAAuC,CAAC;KAC9E,EACD,KAAK,EAAE,EAAE,cAAc,EAAE,UAAU,EAAE,IAAI,EAAE,EAAE,EAAE;QAC7C,IAAI,CAAC;YACH,MAAM,WAAW,GAAG,MAAM,OAAO,CAAC,OAAO,CAAC;gBACxC,aAAa,EAAE,cAAc;gBAC7B,SAAS,EAAE,UAAU;gBACrB,IAAI;aACL,CAAC,CAAC;YACH,OAAO;gBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC;aACjF,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO;gBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAG,KAAe,CAAC,OAAO,EAAE,CAAC,EAAE,CAAC;gBAC/F,OAAO,EAAE,IAAI;aACd,CAAC;QACJ,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
|