add the expectJobRegExpStr field to make it can filter unrelated job

This commit is contained in:
geekgeekrun
2024-12-25 02:47:59 +08:00
parent 5184d275c3
commit 4e28fdd56e
6 changed files with 100 additions and 19 deletions

View File

@@ -5,5 +5,6 @@
"degreeList": [],
"scaleList": [],
"industryList": []
}
},
"expectJobRegExpStr": ""
}

View File

@@ -78,6 +78,7 @@ const bossLocalStorage = readStorageFile('boss-local-storage.json')
const targetCompanyList = readConfigFile('target-company-list.json').filter(it => !!it.trim());
const anyCombineRecommendJobFilter = readConfigFile('boss.json').anyCombineRecommendJobFilter
const expectJobRegExpStr = readConfigFile('boss.json').expectJobRegExpStr
const localStoragePageUrl = `https://www.zhipin.com/desktop/`
const recommendJobPageUrl = `https://www.zhipin.com/web/geek/job-recommend`
@@ -96,8 +97,9 @@ let page
const blockBossNotNewChat = new Set()
const blockBossNotActive = new Set()
const blockJobNotSuit = new Set()
async function markJobAsNotSuitInRecommendPage () {
async function markJobAsNotSuitInRecommendPage (reasonCode) {
/**
* @type {{chosenReasonInUi?: { code: number, text: string}}}
*/
@@ -127,15 +129,29 @@ async function markJobAsNotSuitInRecommendPage () {
})()
let isOptionChosen = false
if (chooseReasonDialogProxy) {
const bossNotActiveOptionProxy = await chooseReasonDialogProxy.$(`.zp-type-item[title="BOSS活跃度低"]`)
if (bossNotActiveOptionProxy) {
await bossNotActiveOptionProxy.click()
isOptionChosen = true
} else {
const recruitStoppedOptionProxy = await chooseReasonDialogProxy.$(`.zp-type-item[title="职位停招/招满"]`)
if (recruitStoppedOptionProxy) {
await recruitStoppedOptionProxy.click()
isOptionChosen = true
switch (reasonCode) {
case MarkAsNotSuitReason.BOSS_INACTIVE: {
const bossNotActiveOptionProxy = await chooseReasonDialogProxy.$(`.zp-type-item[title="BOSS活跃度低"]`)
if (bossNotActiveOptionProxy) {
await bossNotActiveOptionProxy.click()
isOptionChosen = true
} else {
const recruitStoppedOptionProxy = await chooseReasonDialogProxy.$(`.zp-type-item[title="职位停招/招满"]`)
if (recruitStoppedOptionProxy) {
await recruitStoppedOptionProxy.click()
isOptionChosen = true
}
}
break
}
case MarkAsNotSuitReason.JOB_NOT_SUIT:
default: {
const jobNotSuitOptionProxy = await chooseReasonDialogProxy.$(`.zp-type-item[title="面试过/入职过"]`)
if (jobNotSuitOptionProxy) {
await jobNotSuitOptionProxy.click()
isOptionChosen = true
}
break
}
}
@@ -175,6 +191,24 @@ async function markJobAsNotSuitInRecommendPage () {
return result
}
export function testIfJobTitleOrDescriptionSuit (jobInfo, regExpStr) {
if (!regExpStr) {
return true
}
try {
const regExp = new RegExp(regExpStr, 'i')
if (
!regExp.test(jobInfo.jobName)
&& !regExp.test(jobInfo.positionName)
&& !regExp.test(jobInfo.postDescription)
) {
return false
}
} catch {
}
return true
}
async function setFilterCondition (selectedFilters) {
const {
salaryList = [],
@@ -434,9 +468,14 @@ async function toRecommendPage (hooks) {
continueFind: while (targetJobIndex < 0 && !hasReachLastPage) {
// when disable company allow list, we will believe that the first one in the list is your expect job.
let tempTargetJobIndexToCheckDetail = enableCompanyAllowList ? jobListData.findIndex(
it => !blockBossNotNewChat.has(it.encryptBossId) && !blockBossNotActive.has(it.encryptBossId) && [...expectCompanySet].find(name => it.brandName.includes(name))
it => !blockBossNotNewChat.has(it.encryptBossId)
&& !blockBossNotActive.has(it.encryptBossId)
&& [...expectCompanySet].find(
name => it.brandName.includes(name)
)
&& !blockJobNotSuit.has(it.encryptJobId)
) : jobListData.findIndex(
it => !blockBossNotNewChat.has(it.encryptBossId) && !blockBossNotActive.has(it.encryptBossId)
it => !blockBossNotNewChat.has(it.encryptBossId) && !blockBossNotActive.has(it.encryptBossId) && !blockJobNotSuit.has(it.encryptJobId)
)
while (tempTargetJobIndexToCheckDetail < 0 && !hasReachLastPage) {
// fetch new
@@ -472,7 +511,11 @@ async function toRecommendPage (hooks) {
document.querySelector('.job-recommend-main')?.__vue__?.jobList
`
)
tempTargetJobIndexToCheckDetail = jobListData.findIndex(it => !blockBossNotNewChat.has(it.encryptBossId) && !blockBossNotActive.has(it.encryptBossId) && [...expectCompanySet].find(name => it.brandName.includes(name)))
tempTargetJobIndexToCheckDetail = jobListData.findIndex(it =>
!blockBossNotNewChat.has(it.encryptBossId) &&
!blockBossNotActive.has(it.encryptBossId) &&
[...expectCompanySet].find(name => it.brandName.includes(name))) &&
!blockJobNotSuit.has(it.encryptJobId)
}
if (tempTargetJobIndexToCheckDetail < 0 && hasReachLastPage) {
@@ -533,7 +576,7 @@ async function toRecommendPage (hooks) {
blockBossNotActive.add(targetJobData.jobInfo.encryptUserId)
// click prevent recommend button
try {
const { chosenReasonInUi } = await markJobAsNotSuitInRecommendPage()
const { chosenReasonInUi } = await markJobAsNotSuitInRecommendPage(MarkAsNotSuitReason.BOSS_INACTIVE)
await hooks.jobMarkedAsNotSuit.promise(
targetJobData,
{
@@ -549,6 +592,28 @@ async function toRecommendPage (hooks) {
}
continue continueFind
}
if (
!testIfJobTitleOrDescriptionSuit(targetJobData.jobInfo, expectJobRegExpStr)
) {
blockJobNotSuit.add(targetJobData.jobInfo.encryptId)
try {
const { chosenReasonInUi } = await markJobAsNotSuitInRecommendPage(MarkAsNotSuitReason.JOB_NOT_SUIT)
await hooks.jobMarkedAsNotSuit.promise(
targetJobData,
{
markFrom: ChatStartupFrom.AutoFromRecommendList,
markReason: MarkAsNotSuitReason.JOB_NOT_SUIT,
extInfo: {
bossActiveTimeDesc: targetJobData.bossInfo.activeTimeDesc,
chosenReasonInUi
}
}
)
} catch {
}
debugger
continue continueFind
}
const startChatButtonInnerHTML = await page.evaluate('document.querySelector(".job-detail-box .op-btn.op-btn-chat")?.innerHTML.trim()')
if (startChatButtonInnerHTML !== '立即沟通') {
blockBossNotNewChat.add(targetJobData.jobInfo.encryptUserId)

View File

@@ -1,5 +1,6 @@
export enum MarkAsNotSuitReason {
UNKNOWN = 0,
BOSS_INACTIVE = 1,
USER_MANUAL_OPERATION_WITH_UNKNOWN_REASON = 2
USER_MANUAL_OPERATION_WITH_UNKNOWN_REASON = 2,
JOB_NOT_SUIT = 3,
}

View File

@@ -60,11 +60,12 @@ export default function initIpc() {
const bossConfig = readConfigFile('boss.json')
bossConfig.anyCombineRecommendJobFilter = payload.anyCombineRecommendJobFilter
bossConfig.expectJobRegExpStr = payload.expectJobRegExpStr
return await Promise.all([
writeConfigFile('dingtalk.json', dingtalkConfig),
writeConfigFile('target-company-list.json', payload.expectCompanies.split(',')),
writeConfigFile('boss.json', bossConfig),
writeConfigFile('boss.json', bossConfig)
])
})

View File

@@ -12,6 +12,12 @@
>
<el-input v-model="formContent.dingtalkRobotAccessToken" />
</el-form-item>
<el-form-item
label="期望职位正则(按照职位名称+职位描述筛选职位,非目标职位直接标注不合适;为空时将不筛选)"
prop="expectJobRegExpStr"
>
<el-input v-model="formContent.expectJobRegExpStr" />
</el-form-item>
<el-form-item
label="期望公司(以逗号分隔,置空即遍历推荐列表,依次开聊)"
prop="expectCompanies"
@@ -67,7 +73,8 @@ const router = useRouter()
const formContent = ref({
dingtalkRobotAccessToken: '',
expectCompanies: '',
anyCombineRecommendJobFilter: {}
anyCombineRecommendJobFilter: {},
expectJobRegExpStr: ''
})
const currentAnyCombineRecommendJobFilterCombinationCount = computed(() => {
@@ -86,12 +93,14 @@ electron.ipcRenderer.invoke('fetch-config-file-content').then((res) => {
scaleList: [],
industryList: []
}
formContent.value.expectJobRegExpStr = res.config['boss.json']?.expectJobRegExpStr ?? ''
})
const formRules = {}
const formRef = ref<InstanceType<typeof ElForm>>()
const handleSubmit = async () => {
formContent.value.expectJobRegExpStr = (formContent.value.expectJobRegExpStr || '').trim()
await formRef.value!.validate()
await electron.ipcRenderer.invoke('save-config-file-from-ui', JSON.stringify(formContent.value))

View File

@@ -33,6 +33,9 @@
<strong>{{ markReasonTopicMap[row.markReason] }}</strong>
<pre class="m-0 of-auto">{{ formatMarkReason(row) }}</pre>
</template>
<template v-else-if="row.markReason === MarkAsNotSuitReason.JOB_NOT_SUIT">
<strong>{{ markReasonTopicMap[row.markReason] }}</strong>
</template>
</template>
</ElTableColumn>
<ElTableColumn prop="experienceName" label="工作经验" />
@@ -168,7 +171,8 @@ function handleViewJobSnapshotButtonClick(record: VMarkAsNotSuitLog) {
const markReasonTopicMap = {
[MarkAsNotSuitReason.BOSS_INACTIVE]: 'Boss不活跃',
[MarkAsNotSuitReason.USER_MANUAL_OPERATION_WITH_UNKNOWN_REASON]: '手动标记不合适'
[MarkAsNotSuitReason.USER_MANUAL_OPERATION_WITH_UNKNOWN_REASON]: '手动标记不合适',
[MarkAsNotSuitReason.JOB_NOT_SUIT]: '职位不合适'
}
function formatMarkReason(row: VMarkAsNotSuitLog) {