|
|
import json, os, glob, pathlib |
|
|
from fastapi.middleware.cors import CORSMiddleware |
|
|
from fastapi import FastAPI, Request, Header, BackgroundTasks, HTTPException, status |
|
|
from google import genai |
|
|
from linebot import LineBotApi, WebhookHandler |
|
|
from linebot.exceptions import InvalidSignatureError |
|
|
from linebot.models import MessageEvent, TextMessage, TextSendMessage, ImageSendMessage, AudioMessage |
|
|
import PyPDF2 |
|
|
|
|
|
|
|
|
client = genai.Client(api_key=os.getenv("GOOGLE_API_KEY")) |
|
|
|
|
|
|
|
|
files = glob.glob('docs/*.pdf') |
|
|
pdf_content = '' |
|
|
for filename in files: |
|
|
pdf_file = open(filename, 'rb') |
|
|
pdf_reader = PyPDF2.PdfReader(pdf_file) |
|
|
for i in range(len(pdf_reader.pages)): |
|
|
page = pdf_reader.pages[i] |
|
|
pdf_content+= page.extract_text() |
|
|
|
|
|
|
|
|
system_instruction = pdf_content+"\n"+"您是一位問答助手。請僅限使用以上提供的內容來回答問題。如果您不知道答案,請說您不知道,不要捏造答案。" |
|
|
thinking_config = genai.types.ThinkingConfig(thinking_budget=0) |
|
|
generation_config = genai.types.GenerateContentConfig(max_output_tokens=3000, temperature=0.1, top_p=0.2, |
|
|
thinking_config=thinking_config, |
|
|
system_instruction=system_instruction) |
|
|
|
|
|
|
|
|
line_bot_api = LineBotApi(os.environ["CHANNEL_ACCESS_TOKEN"]) |
|
|
line_handler = WebhookHandler(os.environ["CHANNEL_SECRET"]) |
|
|
|
|
|
|
|
|
working_status = os.getenv("DEFALUT_TALKING", default = "true").lower() == "true" |
|
|
|
|
|
|
|
|
app = FastAPI() |
|
|
|
|
|
|
|
|
app.add_middleware( |
|
|
CORSMiddleware, |
|
|
allow_origins=["*"], |
|
|
allow_credentials=True, |
|
|
allow_methods=["*"], |
|
|
allow_headers=["*"], |
|
|
) |
|
|
|
|
|
|
|
|
@app.get("/") |
|
|
def root(): |
|
|
return {"title": "Line Bot"} |
|
|
|
|
|
|
|
|
@app.post("/webhook") |
|
|
async def webhook( |
|
|
request: Request, |
|
|
background_tasks: BackgroundTasks, |
|
|
x_line_signature=Header(None), |
|
|
): |
|
|
|
|
|
body = await request.body() |
|
|
try: |
|
|
|
|
|
background_tasks.add_task( |
|
|
line_handler.handle, body.decode("utf-8"), x_line_signature |
|
|
) |
|
|
except InvalidSignatureError: |
|
|
|
|
|
raise HTTPException(status_code=400, detail="Invalid signature") |
|
|
return "ok" |
|
|
|
|
|
|
|
|
@line_handler.add(MessageEvent, message=TextMessage) |
|
|
def handle_message(event): |
|
|
global working_status |
|
|
|
|
|
|
|
|
if event.type != "message" or event.message.type != "text": |
|
|
|
|
|
line_bot_api.reply_message( |
|
|
event.reply_token, |
|
|
TextSendMessage(text="Event type error:[No message or the message does not contain text]") |
|
|
) |
|
|
|
|
|
|
|
|
elif event.message.text == "再見": |
|
|
|
|
|
line_bot_api.reply_message( |
|
|
event.reply_token, |
|
|
TextSendMessage(text="Bye!") |
|
|
) |
|
|
return |
|
|
|
|
|
|
|
|
elif working_status: |
|
|
try: |
|
|
|
|
|
prompt = event.message.text |
|
|
|
|
|
completion = client.models.generate_content( |
|
|
model="gemini-2.5-flash", |
|
|
contents=prompt, |
|
|
config=generation_config).text |
|
|
|
|
|
if (completion != None): |
|
|
|
|
|
out = completion |
|
|
else: |
|
|
|
|
|
out = "Gemini沒答案!請換個說法!" |
|
|
except: |
|
|
|
|
|
out = "Gemini執行出錯!請換個說法!" |
|
|
|
|
|
|
|
|
line_bot_api.reply_message( |
|
|
event.reply_token, |
|
|
TextSendMessage(text=out)) |
|
|
|
|
|
if __name__ == "__main__": |
|
|
|
|
|
uvicorn.run("main:app", host="0.0.0.0", port=7860, reload=True) |