-
Notifications
You must be signed in to change notification settings - Fork 0
/
app.py
111 lines (86 loc) · 3.46 KB
/
app.py
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
import os
import sys
from flask import Flask, jsonify, request, abort, send_file
from dotenv import load_dotenv
from linebot import LineBotApi, WebhookParser
from linebot.exceptions import InvalidSignatureError
from linebot.models import MessageEvent, TextMessage, TextSendMessage, UnfollowEvent
from fsm import TocMachine, create_machine
from utils import send_text_message
load_dotenv()
machine_for_graph = create_machine()
machines = {}
app = Flask(__name__, static_url_path="")
# get channel_secret and channel_access_token from your environment variable
channel_secret = os.getenv("LINE_CHANNEL_SECRET", None)
channel_access_token = os.getenv("LINE_CHANNEL_ACCESS_TOKEN", None)
if channel_secret is None:
print("Specify LINE_CHANNEL_SECRET as environment variable.")
sys.exit(1)
if channel_access_token is None:
print("Specify LINE_CHANNEL_ACCESS_TOKEN as environment variable.")
sys.exit(1)
line_bot_api = LineBotApi(channel_access_token)
parser = WebhookParser(channel_secret)
# test for callback
@app.route("/callback", methods=["POST"])
def callback():
signature = request.headers["X-Line-Signature"]
# get request body as text
body = request.get_data(as_text=True)
app.logger.info("Request body: " + body)
# parse webhook body
try:
events = parser.parse(body, signature)
except InvalidSignatureError:
abort(400)
# if event is MessageEvent and message is TextMessage, then echo text
for event in events:
if not isinstance(event, MessageEvent):
continue
if not isinstance(event.message, TextMessage):
continue
line_bot_api.reply_message(
event.reply_token, TextSendMessage(text=event.message.text)
)
return "OK"
@app.route("/webhook", methods=["POST"])
def webhook_handler():
signature = request.headers["X-Line-Signature"]
# get request body as text
body = request.get_data(as_text=True)
app.logger.info(f"Request body: {body}")
# parse webhook body
try:
events = parser.parse(body, signature)
except InvalidSignatureError:
abort(400)
# if event is MessageEvent and message is TextMessage, then echo text
for event in events:
if isinstance(event, UnfollowEvent):
if event.source.user_id in machines:
del machines[event.source.user_id]
continue
if not isinstance(event, MessageEvent):
continue
if not isinstance(event.message, TextMessage):
continue
if not isinstance(event.message.text, str):
continue
if event.source.user_id not in machines:
machines[event.source.user_id] = create_machine()
print(f"\nFSM STATE: {machines[event.source.user_id].state}")
# print(f"REQUEST BODY: \n{body}")
response = machines[event.source.user_id].advance(event)
print(f"\nFSM STATE: {machines[event.source.user_id].state}")
if response == False:
send_text_message(event.reply_token,
"無效的回應,請再次嘗試、選擇!\n若您認為操作正常,請嘗試重新輸入「開始」。")
return "OK"
@app.route("/show-fsm", methods=["GET"])
def show_fsm():
machine_for_graph.get_graph().draw("fsm.png", prog="dot", format="png")
return send_file("fsm.png", mimetype="image/png")
if __name__ == "__main__":
port = os.environ.get("PORT", 8000)
app.run(host="0.0.0.0", port=port, debug=True)