岗位自动投递agent 对话功能
This commit is contained in:
@@ -0,0 +1 @@
|
||||
"""求职助手 Agent 对话 AI 模块"""
|
||||
@@ -0,0 +1,52 @@
|
||||
"""求职助手 Agent 对话 AI 引擎
|
||||
|
||||
构造 prompt + 调 LLM + 解析返回。
|
||||
依赖:LLM 枚举、job_agent/prompts、parse_llm_json
|
||||
"""
|
||||
|
||||
from app.ai.job_agent.prompts import SYSTEM_PROMPT
|
||||
from app.ai.models import LLM
|
||||
from app.core.logger import log
|
||||
from app.tool.json_helper import parse_llm_json
|
||||
|
||||
|
||||
async def agent_chat(resume_text: str, message: str, history: list[dict],
|
||||
job_categories: list[str], regions: list[str],
|
||||
industries: list[str]) -> dict:
|
||||
"""求职助手对话
|
||||
1. 构造 system prompt 2. 拼 messages 3. 调 LLM 4. 解析返回
|
||||
"""
|
||||
# 1. 构造 system prompt
|
||||
system_content = SYSTEM_PROMPT.format(
|
||||
resume_text=resume_text,
|
||||
job_categories="、".join(job_categories) if job_categories else "未设置",
|
||||
regions="、".join(regions) if regions else "未设置",
|
||||
industries="、".join(industries) if industries else "未设置",
|
||||
)
|
||||
|
||||
# 2. 拼 messages
|
||||
messages = [("system", system_content)]
|
||||
for msg in history:
|
||||
messages.append((msg["role"], msg["content"]))
|
||||
messages.append(("human", message))
|
||||
|
||||
# 3. 调 LLM
|
||||
try:
|
||||
llm = LLM.JIAYU_CLAUDE_SONNET_4_5.create(temperature=0.7)
|
||||
result = await llm.ainvoke(messages)
|
||||
raw = result.content
|
||||
except Exception as e:
|
||||
log.error(f"求职助手AI调用失败: {e}")
|
||||
return {"message": "抱歉,我暂时无法回复,请稍后再试。", "tool": None, "toolParams": None}
|
||||
|
||||
# 4. 解析返回
|
||||
try:
|
||||
parsed = parse_llm_json(raw)
|
||||
return {
|
||||
"message": parsed.get("message", ""),
|
||||
"tool": parsed.get("tool"),
|
||||
"toolParams": parsed.get("toolParams"),
|
||||
}
|
||||
except Exception as e:
|
||||
log.warning(f"求职助手AI返回解析失败, raw={raw}, error={e}")
|
||||
return {"message": raw.strip() if raw else "抱歉,我暂时无法回复。", "tool": None, "toolParams": None}
|
||||
@@ -0,0 +1,34 @@
|
||||
"""求职助手 Agent 对话 Prompt 模板"""
|
||||
|
||||
SYSTEM_PROMPT = """你是 OfferPie 求职助手,帮助用户找到合适的工作。
|
||||
|
||||
【用户简历】
|
||||
{resume_text}
|
||||
|
||||
【求职意向】
|
||||
意向岗位:{job_categories}
|
||||
意向城市:{regions}
|
||||
意向行业:{industries}
|
||||
|
||||
【你的能力】
|
||||
1. 回答求职相关问题(面试技巧、简历建议、行业分析等),回复不超过200字
|
||||
2. 当用户想看岗位推荐时,提取用户的偏好描述,调用岗位推荐工具
|
||||
3. 当用户想修改求职偏好/设置时,调用偏好设置工具
|
||||
|
||||
【输出格式】
|
||||
严格返回 JSON,不要其他内容:
|
||||
{{"message":"回复内容","tool":null,"toolParams":null}}
|
||||
|
||||
tool 可选值:
|
||||
- null:普通对话,不触发工具
|
||||
- "recommend":岗位推荐,toolParams 必须包含 {{"preference":"用户偏好描述"}}
|
||||
- "editPreference":调整偏好,toolParams 为 null
|
||||
|
||||
【规则】
|
||||
1. 只聊求职相关话题,其他话题礼貌拒绝
|
||||
2. 回复简洁,不超过200字
|
||||
3. 用户表达想看岗位、推荐岗位、帮我找工作等意图时,从对话中提取偏好描述,返回 recommend
|
||||
4. 用户表达想改设置、调整偏好、修改意向等意图时,返回 editPreference
|
||||
5. 偏好描述要准确概括用户的岗位偏好,如"更偏技术方向的产品岗"、"大厂优先"
|
||||
6. 如果用户没有明确偏好,preference 填"无特殊偏好"
|
||||
"""
|
||||
@@ -0,0 +1,24 @@
|
||||
"""求职助手 Agent 对话接口"""
|
||||
|
||||
from fastapi import APIRouter
|
||||
|
||||
from app.core.context import RequestContext
|
||||
from app.core.database import get_db
|
||||
from app.schemas.job_agent_chat import JobAgentChatParam
|
||||
from app.services.job_agent_chat_service import JobAgentChatService
|
||||
|
||||
router = APIRouter(prefix="/job-agent", tags=["求职助手agent"])
|
||||
|
||||
|
||||
@router.post("/chat", summary="求职助手对话")
|
||||
async def chat(param: JobAgentChatParam):
|
||||
"""求职助手对话,根据用户简历和意向提供求职建议、触发岗位推荐或偏好调整"""
|
||||
user_id = RequestContext.user_id.get()
|
||||
async for session in get_db():
|
||||
service = JobAgentChatService(session)
|
||||
result = await service.chat(
|
||||
user_id, param.resume_id, param.message,
|
||||
[msg.model_dump() for msg in param.history],
|
||||
param.job_categories, param.regions, param.industries,
|
||||
)
|
||||
return result
|
||||
@@ -34,11 +34,13 @@ from app.api.health import router as health_router
|
||||
from app.api.resume import router as resume_router
|
||||
from app.api.resume_diagnose import router as resume_diagnose_router
|
||||
from app.api.skill_gap import router as skill_gap_router
|
||||
from app.api.job_agent_chat import router as job_agent_chat_router
|
||||
|
||||
app.include_router(health_router)
|
||||
app.include_router(resume_router)
|
||||
app.include_router(resume_diagnose_router)
|
||||
app.include_router(skill_gap_router)
|
||||
app.include_router(job_agent_chat_router)
|
||||
# ==============================
|
||||
|
||||
if __name__ == "__main__":
|
||||
|
||||
@@ -0,0 +1,37 @@
|
||||
"""求职助手 Agent 对话 Schema
|
||||
|
||||
请求参数 Param、响应 Dto。
|
||||
字段命名使用 camelCase alias,与前端 JSON 对齐。
|
||||
"""
|
||||
|
||||
from typing import Literal
|
||||
|
||||
from pydantic import BaseModel, Field
|
||||
|
||||
ChatRole = Literal["user", "assistant"]
|
||||
|
||||
|
||||
class ChatMessage(BaseModel):
|
||||
role: ChatRole
|
||||
content: str
|
||||
|
||||
|
||||
class JobAgentChatParam(BaseModel):
|
||||
message: str = Field(..., description="用户输入的消息")
|
||||
resume_id: int = Field(..., alias="resumeId", description="简历ID")
|
||||
history: list[ChatMessage] = Field(default_factory=list, description="对话历史")
|
||||
job_categories: list[str] = Field(default_factory=list, alias="jobCategories", description="意向岗位类型名称")
|
||||
regions: list[str] = Field(default_factory=list, alias="regions", description="意向城市名称")
|
||||
industries: list[str] = Field(default_factory=list, alias="industries", description="意向行业名称")
|
||||
|
||||
|
||||
class ToolParams(BaseModel):
|
||||
preference: str = Field(default="", description="用户岗位偏好描述")
|
||||
|
||||
|
||||
class JobAgentChatDto(BaseModel):
|
||||
message: str = Field(..., description="AI回复文本,不超过200字")
|
||||
tool: str | None = Field(default=None, description="前端需执行的工具:recommend / editPreference / null")
|
||||
tool_params: ToolParams | None = Field(default=None, alias="toolParams", description="工具参数")
|
||||
|
||||
model_config = {"populate_by_name": True}
|
||||
@@ -0,0 +1,85 @@
|
||||
"""求职助手 Agent 对话 Service
|
||||
|
||||
主要功能:查询简历数据,调用 AI 模块完成对话。
|
||||
依赖:UserResume + 5张子表 ORM、job_agent.chat AI 模块
|
||||
使用表:bg_user_resume、bg_user_resume_education/work/internship/project/competition(查询简历)
|
||||
"""
|
||||
|
||||
from sqlalchemy import select
|
||||
from sqlalchemy.ext.asyncio import AsyncSession
|
||||
|
||||
from app.ai.job_agent.chat import agent_chat
|
||||
from app.models.user_resume import UserResume
|
||||
from app.models.user_resume_education import UserResumeEducation
|
||||
from app.models.user_resume_work import UserResumeWork
|
||||
from app.models.user_resume_internship import UserResumeInternship
|
||||
from app.models.user_resume_project import UserResumeProject
|
||||
from app.models.user_resume_competition import UserResumeCompetition
|
||||
|
||||
|
||||
class JobAgentChatService:
|
||||
|
||||
def __init__(self, session: AsyncSession):
|
||||
self.session = session
|
||||
|
||||
async def chat(self, user_id: int, resume_id: int, message: str,
|
||||
history: list[dict], job_categories: list[str],
|
||||
regions: list[str], industries: list[str]) -> dict:
|
||||
"""求职助手对话:查简历 → 序列化 → 调 AI 模块"""
|
||||
resume = await self._get_resume(resume_id, user_id)
|
||||
edu, work, intern, proj, comp = await self._load_sub_tables(resume_id)
|
||||
resume_text = self._build_resume_text(resume, edu, work, intern, proj, comp)
|
||||
return await agent_chat(resume_text, message, history, job_categories, regions, industries)
|
||||
|
||||
async def _get_resume(self, resume_id: int, user_id: int) -> UserResume:
|
||||
"""查指定简历"""
|
||||
result = await self.session.execute(select(UserResume).where(UserResume.id == resume_id, UserResume.user_id == user_id))
|
||||
resume = result.scalar_one_or_none()
|
||||
if not resume:
|
||||
raise ValueError("简历不存在")
|
||||
return resume
|
||||
|
||||
async def _load_sub_tables(self, resume_id: int):
|
||||
"""查询简历5张子表"""
|
||||
edu = (await self.session.execute(select(UserResumeEducation).where(UserResumeEducation.resume_id == resume_id))).scalars().all()
|
||||
work = (await self.session.execute(select(UserResumeWork).where(UserResumeWork.resume_id == resume_id))).scalars().all()
|
||||
intern = (await self.session.execute(select(UserResumeInternship).where(UserResumeInternship.resume_id == resume_id))).scalars().all()
|
||||
proj = (await self.session.execute(select(UserResumeProject).where(UserResumeProject.resume_id == resume_id))).scalars().all()
|
||||
comp = (await self.session.execute(select(UserResumeCompetition).where(UserResumeCompetition.resume_id == resume_id))).scalars().all()
|
||||
return edu, work, intern, proj, comp
|
||||
|
||||
@staticmethod
|
||||
def _build_resume_text(resume: UserResume, edu_list, work_list, intern_list, proj_list, comp_list) -> str:
|
||||
"""将简历数据序列化为文本供 AI 使用"""
|
||||
parts = []
|
||||
if resume.name:
|
||||
parts.append(f"姓名:{resume.name}")
|
||||
if resume.target_position:
|
||||
parts.append(f"目标岗位:{resume.target_position}")
|
||||
if resume.skills:
|
||||
parts.append(f"技能:{'、'.join(resume.skills)}")
|
||||
if resume.certificates:
|
||||
parts.append(f"证书:{'、'.join(resume.certificates)}")
|
||||
if resume.summary:
|
||||
parts.append(f"个人概述:{resume.summary}")
|
||||
if edu_list:
|
||||
parts.append("教育经历:")
|
||||
for r in edu_list:
|
||||
parts.append(f" - {r.school or ''} {r.major or ''} {r.degree or ''}")
|
||||
if work_list:
|
||||
parts.append("工作经历:")
|
||||
for r in work_list:
|
||||
parts.append(f" - {r.company_name or ''} {r.position or ''}")
|
||||
if intern_list:
|
||||
parts.append("实习经历:")
|
||||
for r in intern_list:
|
||||
parts.append(f" - {r.company_name or ''} {r.position or ''}")
|
||||
if proj_list:
|
||||
parts.append("项目经历:")
|
||||
for r in proj_list:
|
||||
parts.append(f" - {r.project_name or ''} {r.role or ''}")
|
||||
if comp_list:
|
||||
parts.append("竞赛经历:")
|
||||
for r in comp_list:
|
||||
parts.append(f" - {r.competition_name or ''} {r.award or ''}")
|
||||
return "\n".join(parts) if parts else "暂无简历信息"
|
||||
Reference in New Issue
Block a user