Skip to content

Commit

Permalink
Create getDescriptionsLineIdx and refactor getFirstBulletPointLineIdx -
Browse files Browse the repository at this point in the history
  • Loading branch information
xitanggg committed Jul 6, 2023
1 parent d29b899 commit 68ac640
Show file tree
Hide file tree
Showing 7 changed files with 65 additions and 59 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ import {
import { getTextWithHighestFeatureScore } from "lib/parse-resume-from-pdf/extract-resume-from-sections/lib/feature-scoring-system";
import {
getBulletPointsFromLines,
getFirstBulletPointLineIdx,
getDescriptionsLineIdx,
} from "lib/parse-resume-from-pdf/extract-resume-from-sections/lib/bullet-points";

/**
Expand All @@ -37,7 +37,7 @@ const hasDegree = (item: TextItem) =>
const matchGPA = (item: TextItem) => item.text.match(/[0-4]\.\d{1,2}/);
const matchGrade = (item: TextItem) => {
const grade = parseFloat(item.text);
if (Number.isFinite(grade)) {
if (Number.isFinite(grade) && grade <= 110) {
return [String(grade)] as RegExpMatchArray;
}
return null;
Expand Down Expand Up @@ -87,15 +87,10 @@ export const extractEducation = (sections: ResumeSectionToLines) => {
);

let descriptions: string[] = [];
const firstBulletPointLineIdx = getFirstBulletPointLineIdx(
subsectionLines,
[":"]
);
if (firstBulletPointLineIdx !== undefined) {
const subsectionBulletPointLines = subsectionLines.slice(
firstBulletPointLineIdx
);
descriptions = getBulletPointsFromLines(subsectionBulletPointLines);
const descriptionsLineIdx = getDescriptionsLineIdx(subsectionLines);
if (descriptionsLineIdx !== undefined) {
const descriptionsLines = subsectionLines.slice(descriptionsLineIdx);
descriptions = getBulletPointsFromLines(descriptionsLines);
}

educations.push({ school, degree, gpa, date, descriptions });
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ import { divideSectionIntoSubsections } from "lib/parse-resume-from-pdf/extract-
import { getTextWithHighestFeatureScore } from "lib/parse-resume-from-pdf/extract-resume-from-sections/lib/feature-scoring-system";
import {
getBulletPointsFromLines,
getFirstBulletPointLineIdx,
getDescriptionsLineIdx,
} from "lib/parse-resume-from-pdf/extract-resume-from-sections/lib/bullet-points";

export const extractProject = (sections: ResumeSectionToLines) => {
Expand All @@ -23,11 +23,10 @@ export const extractProject = (sections: ResumeSectionToLines) => {
const subsections = divideSectionIntoSubsections(lines);

for (const subsectionLines of subsections) {
const firstBulletPointLineIdx =
getFirstBulletPointLineIdx(subsectionLines) ?? 1;
const descriptionsLineIdx = getDescriptionsLineIdx(subsectionLines) ?? 1;

const subsectionInfoTextItems = subsectionLines
.slice(0, firstBulletPointLineIdx)
.slice(0, descriptionsLineIdx)
.flat();
const [date, dateScores] = getTextWithHighestFeatureScore(
subsectionInfoTextItems,
Expand All @@ -43,10 +42,8 @@ export const extractProject = (sections: ResumeSectionToLines) => {
false
);

const subsectionBulletPointLines = subsectionLines.slice(
firstBulletPointLineIdx
);
const descriptions = getBulletPointsFromLines(subsectionBulletPointLines);
const descriptionsLines = subsectionLines.slice(descriptionsLineIdx);
const descriptions = getBulletPointsFromLines(descriptionsLines);

projects.push({ project, date, descriptions });
projectsScores.push({
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,20 +5,18 @@ import { getSectionLinesByKeywords } from "lib/parse-resume-from-pdf/extract-res
import { initialFeaturedSkills } from "lib/redux/resumeSlice";
import {
getBulletPointsFromLines,
getFirstBulletPointLineIdx,
getDescriptionsLineIdx,
} from "lib/parse-resume-from-pdf/extract-resume-from-sections/lib/bullet-points";

export const extractSkills = (sections: ResumeSectionToLines) => {
const lines = getSectionLinesByKeywords(sections, ["skill"]);
const descriptions = getBulletPointsFromLines(lines);
const descriptionsLineIdx = getDescriptionsLineIdx(lines) ?? 0;
const descriptionsLines = lines.slice(descriptionsLineIdx);
const descriptions = getBulletPointsFromLines(descriptionsLines);

const featuredSkills = deepClone(initialFeaturedSkills) as FeaturedSkill[];
const firstBulletPointLineIndex = getFirstBulletPointLineIdx(lines);
if (
firstBulletPointLineIndex !== undefined &&
firstBulletPointLineIndex !== 0
) {
const featuredSkillsLines = lines.slice(0, firstBulletPointLineIndex);
if (descriptionsLineIdx !== 0) {
const featuredSkillsLines = lines.slice(0, descriptionsLineIdx);
const featuredSkillsTextItems = featuredSkillsLines
.flat()
.filter((item) => item.text.trim())
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ import { divideSectionIntoSubsections } from "lib/parse-resume-from-pdf/extract-
import { getTextWithHighestFeatureScore } from "lib/parse-resume-from-pdf/extract-resume-from-sections/lib/feature-scoring-system";
import {
getBulletPointsFromLines,
getFirstBulletPointLineIdx,
getDescriptionsLineIdx,
} from "lib/parse-resume-from-pdf/extract-resume-from-sections/lib/bullet-points";

// prettier-ignore
Expand Down Expand Up @@ -44,11 +44,10 @@ export const extractWorkExperience = (sections: ResumeSectionToLines) => {
const subsections = divideSectionIntoSubsections(lines);

for (const subsectionLines of subsections) {
const firstBulletPointLineIdx =
getFirstBulletPointLineIdx(subsectionLines) ?? 2;
const descriptionsLineIdx = getDescriptionsLineIdx(subsectionLines) ?? 2;

const subsectionInfoTextItems = subsectionLines
.slice(0, firstBulletPointLineIdx)
.slice(0, descriptionsLineIdx)
.flat();
const [date, dateScores] = getTextWithHighestFeatureScore(
subsectionInfoTextItems,
Expand All @@ -69,10 +68,9 @@ export const extractWorkExperience = (sections: ResumeSectionToLines) => {
false
);

const subsectionBulletPointLines = subsectionLines.slice(
firstBulletPointLineIdx
);
const descriptions = getBulletPointsFromLines(subsectionBulletPointLines);
const subsectionDescriptionsLines =
subsectionLines.slice(descriptionsLineIdx);
const descriptions = getBulletPointsFromLines(subsectionDescriptionsLines);

workExperiences.push({ company, jobTitle, date, descriptions });
workExperiencesScores.push({
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import type { Lines } from "lib/parse-resume-from-pdf/types";
import type { Lines, TextItem } from "lib/parse-resume-from-pdf/types";

/**
* List of bullet points
Expand Down Expand Up @@ -66,24 +66,6 @@ export const getBulletPointsFromLines = (lines: Lines): string[] => {
.filter((text) => !!text);
};

export const getFirstBulletPointLineIdx = (
lines: Lines,
additionalChars: string[] = []
): number | undefined => {
for (let i = 0; i < lines.length; i++) {
for (let item of lines[i]) {
if (
[...BULLET_POINTS, ...additionalChars].some((bullet) =>
item.text.includes(bullet)
)
) {
return i;
}
}
}
return undefined;
};

const getMostCommonBulletPoint = (str: string): string => {
const bulletToCount: { [bullet: string]: number } = BULLET_POINTS.reduce(
(acc: { [bullet: string]: number }, cur) => {
Expand All @@ -104,3 +86,38 @@ const getMostCommonBulletPoint = (str: string): string => {
}
return bulletWithMostCount;
};

const getFirstBulletPointLineIdx = (lines: Lines): number | undefined => {
for (let i = 0; i < lines.length; i++) {
for (let item of lines[i]) {
if (BULLET_POINTS.some((bullet) => item.text.includes(bullet))) {
return i;
}
}
}
return undefined;
};

// Only consider words that don't contain numbers
const isWord = (str: string) => /^[^0-9]+$/.test(str);
const hasAtLeast8Words = (item: TextItem) =>
item.text.split(/\s/).filter(isWord).length >= 8;

export const getDescriptionsLineIdx = (lines: Lines): number | undefined => {
// The main heuristic to determine descriptions is to check if has bullet point
let idx = getFirstBulletPointLineIdx(lines);

// Fallback heuristic if the main heuristic doesn't apply (e.g. LinkedIn resume) to
// check if the line has at least 8 words
if (idx === undefined) {
for (let i = 0; i < lines.length; i++) {
const line = lines[i];
if (line.length === 1 && hasAtLeast8Words(line[0])) {
idx = i;
break;
}
}
}

return idx;
};
Original file line number Diff line number Diff line change
Expand Up @@ -8,8 +8,8 @@ export const hasNumber = (item: TextItem) => /[0-9]/.test(item.text);
export const hasComma = (item: TextItem) => item.text.includes(",");
export const getHasText = (text: string) => (item: TextItem) =>
item.text.includes(text);
export const hasOnlyLettersAndSpaces = (item: TextItem) =>
/^[A-Za-z\s]+$/.test(item.text);
export const hasOnlyLettersSpacesAmpersands = (item: TextItem) =>
/^[A-Za-z\s&]+$/.test(item.text);
export const hasLetterAndIsAllUpperCase = (item: TextItem) =>
hasLetter(item) && item.text.toUpperCase() === item.text;

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ import type {
} from "lib/parse-resume-from-pdf/types";
import {
hasLetterAndIsAllUpperCase,
hasOnlyLettersAndSpaces,
hasOnlyLettersSpacesAmpersands,
isBold,
} from "lib/parse-resume-from-pdf/extract-resume-from-sections/lib/common-features";

Expand Down Expand Up @@ -81,12 +81,13 @@ const isSectionTitle = (line: Line, lineNumber: number) => {
// The following is a fallback heuristic to detect section title if it includes a keyword match
// (This heuristics is not well tested and may not work well)
const text = textItem.text.trim();
const textHasAtMost2Words = text.split(" ").length <= 2;
const textHasAtMost2Words =
text.split(" ").filter((s) => s !== "&").length <= 2;
const startsWithCapitalLetter = /[A-Z]/.test(text.slice(0, 1));

if (
textHasAtMost2Words &&
hasOnlyLettersAndSpaces(textItem) &&
hasOnlyLettersSpacesAmpersands(textItem) &&
startsWithCapitalLetter &&
SECTION_TITLE_KEYWORDS.some((keyword) =>
text.toLowerCase().includes(keyword)
Expand Down

0 comments on commit 68ac640

Please sign in to comment.