Spaces:
Sleeping
Sleeping
File size: 6,545 Bytes
2da8e5c 071beff ce425c0 22f516b 89c2715 ce425c0 c4a77fb ce425c0 2da8e5c 22f516b ce425c0 2da8e5c ce425c0 0acfd1d ce425c0 0acfd1d ce425c0 0acfd1d ce425c0 0acfd1d 2da8e5c ce425c0 2da8e5c ce425c0 0acfd1d 22f516b ce425c0 2da8e5c ce425c0 22f516b ce425c0 139ddee ce425c0 2da8e5c 139ddee ce425c0 139ddee ce425c0 97ce441 ce425c0 139ddee ce425c0 139ddee ce425c0 97ce441 ce425c0 2da8e5c ce425c0 2da8e5c 89c2715 0acfd1d 2da8e5c 22f516b 89c2715 ce425c0 89c2715 2da8e5c 0acfd1d 2da8e5c 0acfd1d ce425c0 2da8e5c 89c2715 22f516b 2da8e5c 22f516b 0acfd1d 89c2715 ce425c0 2da8e5c |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 |
# command_handler.py (已整合 PWS 查詢功能)
import pandas as pd
from linebot.v3.messaging import TextMessage, ImageMessage
# 匯入所有服務函式
from cwa_service import fetch_cwa_alarm_list, fetch_significant_earthquakes, fetch_latest_significant_earthquake
from usgs_service import fetch_global_last24h_text, fetch_taiwan_df_this_year
from plotting_service import create_and_save_map
from ai_service import generate_ai_text
from pws_service import fetch_latest_pws_info # 匯入新的 PWS 服務
from config import CURRENT_YEAR, MCP_SERVER_URL
def get_help_message() -> TextMessage:
"""產生並回傳包含所有指令的說明文字。"""
text = (
"📖 指令列表 (輸入數字即可)\n\n"
"【地震資訊】\n"
"• 1 - 最新一筆顯著地震 (含圖)\n"
"• 2 - 全球近24小時顯著地震(USGS)\n"
"• 3 - 今年台灣顯著地震列表(USGS)\n"
"• 4 - CWA 地震目錄查詢 (外部連結)\n"
"• 5 - CWA 最新地震預警\n"
"• 6 - CWA 最近7天顯著有感地震\n\n"
"【AI 與工具】\n"
"• 7 <問題> - 與 AI 助理對話\n\n"
"【基本指令】\n"
"• 8 - 關於此機器人\n"
"• 9 - 顯示此說明\n"
"【天氣與警報】\n"
"• 10 - 最新 PWS 發布情形\n\n"
)
return TextMessage(text=text)
def get_info_message() -> TextMessage:
"""產生並回傳關於此機器人的資訊。"""
text = (
"🤖 關於我\n\n"
"我是一個多功能助理機器人,提供地震查詢與 AI 對話功能。\n\n"
"• 版本: 5.0 (Gemini Edition) 搭配搜尋地震目錄的MCP功能(可用資料 : 1973-01-01 至 2025-07-06)\n"
"• 資料來源: CWA, USGS, Google Gemini\n"
"• 開發者: dayichen"
)
return TextMessage(text=text)
def get_taiwan_earthquake_list() -> TextMessage:
"""從 USGS 取得今年台灣的地震列表並格式化回覆。"""
result = fetch_taiwan_df_this_year()
if isinstance(result, pd.DataFrame):
count = len(result)
lines = [f"🇹🇼 今年 ({CURRENT_YEAR} 年) 台灣區域顯著地震 (M≥5.0),共 {count} 筆:", "-" * 20]
for _, row in result.head(15).iterrows():
t = row["time_utc"].strftime("%Y-%m-%d %H:%M")
lines.append(
f"規模: {row['magnitude']:.1f} | 日期時間: {t} (UTC)\n"
f"地點: {row['place']}\n"
f"報告連結: {row.get('url', '無')}"
)
if count > 15:
lines.append(f"... (還有 {count-15} 筆資料)")
reply_text = "\n\n".join(lines)
else:
reply_text = result
return TextMessage(text=reply_text)
def get_latest_earthquake_reply() -> list:
"""取得 CWA 最新一筆地震,並組合文字與圖片訊息。"""
try:
latest_eq = fetch_latest_significant_earthquake()
if not latest_eq:
return [TextMessage(text="✅ 近期無顯著有感地震報告。")]
mag_str = f"{latest_eq['Magnitude']:.1f}" if latest_eq.get('Magnitude') is not None else "—"
depth_str = f"{latest_eq['Depth']:.0f}" if latest_eq.get('Depth') is not None else "—"
text_message_content = (
f"🚨 CWA 最新顯著有感地震\n"
f"----------------------------------\n"
f"時間: {latest_eq.get('TimeStr', '—')}\n"
f"地點: {latest_eq.get('Location', '—')}\n"
f"規模: M{mag_str} | 深度: {depth_str} km\n"
f"報告: {latest_eq.get('URL', '無')}"
)
reply_messages = [TextMessage(text=text_message_content)]
if latest_eq.get("ImageURL"):
image_url = latest_eq["ImageURL"]
reply_messages.append(
ImageMessage(original_content_url=image_url, preview_image_url=image_url)
)
return reply_messages
except Exception as e:
return [TextMessage(text=f"❌ 查詢最新地震失敗:{e}")]
def process_message(user_message_raw: str, request_base_url: str) -> list:
"""
處理使用者輸入的主函式。
解析訊息、對應指令並呼叫相應的服務函式。
"""
user_message = (user_message_raw or "").strip()
# 指令與快捷鍵的映射字典
cmd_map = {
'1': '/latest', '2': '/global', '3': '/taiwan',
'4': '/map', '5': '/alert', '6': '/significant',
'7': '/ai', '8': '/info', '9': '/help',
'10': '/pws', # 新增 PWS 指令
'地震': '/global', 'quake': '/global', '幫助': '/help',
'台灣地震': '/taiwan', '臺灣地震': '/taiwan',
'台灣地震畫圖': '/map', '臺灣地震畫圖': '/map',
'地震預警': '/alert',
}
command = ""
arg = ""
# 解析使用者輸入以分離指令和參數
parts = user_message.split(' ', 1)
cmd_key = parts[0].lower()
if cmd_key in cmd_map:
command = cmd_map[cmd_key]
if len(parts) > 1:
arg = parts[1].strip()
# 允許使用者仍然可以使用 /command 格式
elif user_message.startswith('/') and cmd_key in cmd_map.values():
command = cmd_key
if len(parts) > 1:
arg = parts[1].strip()
# 根據解析出的指令執行對應的動作
if command:
if command == '/help': return [get_help_message()]
if command == '/info': return [get_info_message()]
if command == '/latest': return get_latest_earthquake_reply()
if command == '/global': return [TextMessage(text=fetch_global_last24h_text())]
if command == '/taiwan': return [get_taiwan_earthquake_list()]
if command == '/map': return [TextMessage(text=f"🗺️ 外部地震查詢服務\n\n請點擊以下連結:\n{MCP_SERVER_URL}")]
if command == '/alert': return [TextMessage(text=fetch_cwa_alarm_list(limit=5))]
if command == '/significant': return [TextMessage(text=fetch_significant_earthquakes(limit=5))]
if command == '/pws': return [TextMessage(text=fetch_latest_pws_info())] # 新增 PWS 處理邏輯
if command == '/ai':
prompt = arg
if not prompt: return [TextMessage(text="請輸入問題,例如:7 台灣最高的山是哪座?")]
return [TextMessage(text=generate_ai_text(prompt))]
# 如果輸入的不是已知指令,則交由 AI 處理
return [TextMessage(text=generate_ai_text(user_message))] |