mirror of
https://github.com/geekgeekrun/geekgeekrun.git
synced 2026-05-07 05:42:50 +08:00
add the expectJobRegExpStr field to make it can filter unrelated job
This commit is contained in:
@@ -5,5 +5,6 @@
|
||||
"degreeList": [],
|
||||
"scaleList": [],
|
||||
"industryList": []
|
||||
}
|
||||
},
|
||||
"expectJobRegExpStr": ""
|
||||
}
|
||||
@@ -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)
|
||||
|
||||
@@ -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,
|
||||
}
|
||||
@@ -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)
|
||||
])
|
||||
})
|
||||
|
||||
|
||||
@@ -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))
|
||||
|
||||
|
||||
@@ -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) {
|
||||
|
||||
Reference in New Issue
Block a user