Corin1998 commited on
Commit
2309f1e
·
verified ·
1 Parent(s): c52f5bf

Update modules/workflow.py

Browse files
Files changed (1) hide show
  1. modules/workflow.py +105 -46
modules/workflow.py CHANGED
@@ -1,64 +1,123 @@
 
1
  import os
2
- from modules.company_score import score_company
3
- from modules.rag_retriever import retrieve_contexts
4
- from modules.proposal_generator import make_proposal, suggest_next_actions
5
- from modules.exporters import export_docx, export_pptx
6
- from modules.emailer import build_tracking_url
7
- from modules.utils import truncate
8
-
9
- def _make_email(company_name: str, lead_email: str, proposal_md: str) -> dict:
10
- traking = build_tracking_url(
11
- identifier=f"{company_name}",
12
- payload={
13
- "company": company_name,
14
- "redirect": os.getenv("PUBLIC_BASE_URL","/")
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
15
  },
16
- )
17
- subject = f"{company_name}さま向けご提案(無料PoCのご案内)"
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
18
  body = (
19
- f"{company_name} ご担当者さま\n\n"
20
- "突然のご連絡失礼致します。貴社向けに要点を1枚にまとめた提案書を作成しました。\n\n"
21
- f"▼提案要旨\n{truncate(proposal_md, 800)}\n\n"
22
- f"詳細はこちら(クリック計測用リンク) :\n{traking}\n\n"
23
- f"30分ほどのオンラインミーティングの機会を頂戴できればと幸いです。\n"
24
- "ご検討のほど、よろしくお願いいたします。\n"
25
- "\n--\nAgent Studio"
26
  )
27
  return {"subject": subject, "body": body}
28
 
29
- def run_full_workflow(company_name: str,
30
- company_website: str,
31
- lead_email: str | None,
32
- objective: str,
33
- temperature: float=0.4) -> dict:
34
- # 1) スコア
35
- score = score_company(company_name, company_website)
36
 
37
- # 2) RAG 検索
38
- query = f"{company_name} {objective} {company_website}"
39
- top_contexts = retrieve_contexts(query, k=5)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
40
 
41
- # 3) 提案生成
42
- proposal_md = make_proposal(company_name, objective, top_contexts, temperature=temperature)
43
 
44
- # 4) エクスポート
45
- docx_path = export_docx(proposal_md, company_name)
46
- pptx_path = export_pptx(proposal_md, company_name)
 
47
 
48
- # 5) メール下書き
49
- email = _make_email(company_name, lead_email or "", proposal_md)
 
50
 
51
- # 6) 次アクション
52
- next_actions = suggest_next_actions(proposal_md, score["overall"])
 
 
 
 
 
 
 
 
 
53
 
54
  return {
55
  "score": score,
56
  "top_contexts": top_contexts,
57
  "proposal_markdown": proposal_md,
58
- "exports": {
59
- "docx_path": docx_path,
60
- "pptx_path": pptx_path,
61
- },
62
  "email": email,
63
- "next_actions": next_actions
 
64
  }
 
1
+ # modules/workflow.py
2
  import os
3
+ from typing import Dict, Any, List
4
+
5
+ # --- retriever は相対インポートを優先、失敗時は絶対インポートにフォールバック ---
6
+ try:
7
+ from .rag_retriever import retrieve_contexts
8
+ except Exception:
9
+ from modules.rag_retriever import retrieve_contexts # type: ignore
10
+
11
+ # exporters / utils も相対で
12
+ try:
13
+ from .exporters import export_docx, export_pptx
14
+ from .utils import ensure_dirs, export_dir, make_tracking_token
15
+ except Exception:
16
+ # 万一相対が失敗する環境でも動くように
17
+ from modules.exporters import export_docx, export_pptx # type: ignore
18
+ from modules.utils import ensure_dirs, export_dir, make_tracking_token # type: ignore
19
+
20
+
21
+ def _score_company(company_name: str, website: str, contexts: List[str]) -> Dict[str, Any]:
22
+ """
23
+ デモ用の簡易スコア(実運用では独自ロジック/モデルと差し替え可)
24
+ """
25
+ score = 50
26
+ if website:
27
+ score += 10
28
+ if contexts:
29
+ score += min(40, len(contexts) * 5)
30
+ return {
31
+ "company": company_name,
32
+ "website": website,
33
+ "score": min(score, 95),
34
+ "factors": {
35
+ "has_website": bool(website),
36
+ "ctx_hits": len(contexts),
37
  },
38
+ }
39
+
40
+
41
+ def _draft_proposal_md(company_name: str, objective: str, contexts: List[str]) -> str:
42
+ bullets = "\n".join([f"- {c[:200]}..." for c in contexts[:5]])
43
+ obj = objective or "商談化のための初回提案"
44
+ return f"""# 提案ドラフト({company_name} 向け)
45
+
46
+ ## 目的
47
+ {obj}
48
+
49
+ ## 背景インサイト(上位抜粋)
50
+ {bullets if bullets else '- (まだインデックスが空です)'}
51
+
52
+ ## 提案骨子
53
+ 1. 現状の課題仮説を確認
54
+ 2. PoC スコープのすり合わせ
55
+ 3. 成果指標(KPI)とスケジュール
56
+ 4. 次のアクション
57
+ """
58
+
59
+
60
+ def _draft_email(company_name: str) -> Dict[str, str]:
61
+ subject = f"{company_name}様向けご提案(ドラフト)"
62
  body = (
63
+ f"{company_name} ご担当者様\n\n"
64
+ "お世話になっております。簡易の提案ドラフトを共有いたします。\n"
65
+ "ご確認の上、ご意見いただけますと幸いです。\n\n"
66
+ "何卒よろしくお願いいたします。"
 
 
 
67
  )
68
  return {"subject": subject, "body": body}
69
 
 
 
 
 
 
 
 
70
 
71
+ def run_full_workflow(
72
+ company_name: str,
73
+ company_website: str,
74
+ lead_email: str,
75
+ objective: str,
76
+ temperature: float = 0.4,
77
+ ) -> Dict[str, Any]:
78
+ """
79
+ app.py から呼ばれるメイン処理。
80
+ - RAG で上位コンテキスト取得
81
+ - 簡易スコアリング
82
+ - 提案ドラフト生成
83
+ - DOCX/PPTX にエクスポート
84
+ - メール文面と次アクションの素案
85
+ """
86
+ ensure_dirs()
87
+
88
+ query = f"{company_name} {objective or ''}".strip()
89
+ top_contexts = retrieve_contexts(query, top_k=5)
90
 
91
+ score = _score_company(company_name, company_website, top_contexts)
92
+ proposal_md = _draft_proposal_md(company_name, objective, top_contexts)
93
 
94
+ # エクスポート先パス
95
+ out_dir = export_dir()
96
+ docx_path = os.path.join(out_dir, f"{company_name}_proposal.docx")
97
+ pptx_path = os.path.join(out_dir, f"{company_name}_proposal.pptx")
98
 
99
+ # 出力
100
+ export_docx(proposal_md, docx_path)
101
+ export_pptx(proposal_md, pptx_path)
102
 
103
+ email = _draft_email(company_name)
104
+
105
+ # 次アクション(ダミー)
106
+ next_actions = (
107
+ "- 先方の課題確認ミーティング候補日程を提案する\n"
108
+ "- PoC スコープの要件一覧を作成して共有\n"
109
+ "- 競合比較の要点を 3 点に要約\n"
110
+ )
111
+
112
+ # 計測リンクの例(app.py の /t/{token} で受ける想定)
113
+ tracking_token = make_tracking_token({"company": company_name, "redirect": "/"})
114
 
115
  return {
116
  "score": score,
117
  "top_contexts": top_contexts,
118
  "proposal_markdown": proposal_md,
119
+ "exports": {"docx_path": docx_path, "pptx_path": pptx_path},
 
 
 
120
  "email": email,
121
+ "next_actions": next_actions,
122
+ "tracking_token": tracking_token,
123
  }