สวัสดีครับ อันนี้เป็นบทความเก่าที่ผมเอามารีรัน + ย่อให้สั้นลงครับ อาจจะมีบางส่วนตกหล่นไปบ้างแต่คิดว่าไม่น่าจะเยอะ เอาเป็นว่าถ้าขาดส่วนไหนไปสามารถเข้าไปดูได้ที่บทความเก่าๆที่ผมเคยเขียนไว้ได้ครับ
สร้าง AI Chatbot ด้วย IntentParser (พูดอังกฤษ) : Part I
สร้าง AI Chatbot ด้วย IntentParser (พูดอังกฤษ) : Part II
สร้าง AI Chatbot ด้วย IntentParser (พูดอังกฤษ) : Part III
สรุปย่อเผื่อขี้เกียจอ่าน
- ในบทความนี้ใช้ module intent parser สามารถศึกษาได้ที่ https://github.com/nonkung51/IntentParser
- อันนี้ที่อยู่ไฟล์แชทบอท https://github.com/nonkung51/IntentParser/blob/master/examples/Chatbot_examples.py
มาเริ่มวางแผนกันก่อนครับ เราต้องการให้ chatbot ของเรา(ภาษาอังกฤษนะครับ)
- พูด Hello กลับมาตอนเราพูด Hello ไป
- เหมือนกันกับ Good bye.
- ให้ถามเวลาได้ด้วย
เป็นแผนที่ดูดีมากๆ!
แล้วเราจะใช้ module อะไรบ้าง?
- datetime สำหรับบอกเวลา
- random ไว้บอกคำตอบ
- แล้วก็ IntentParser
ดูดี!
แต่อะไรคือ IntentParser? มันคือโมดูลที่ผมเขียนขึ้นมาเองครับ เข้าไปดูได้ที่ลิ้งค์ข้างล่างนี่โลด
https://github.com/nonkung51/IntentParser
สิ่งที่มันทำได้คือแปลภาษาคนไปเป็น format ที่คอมพิวเตอร์อ่านได้ ในที่นี้คือ dict ในภาษา Python
ตัวอย่าง. What’s the weather like in California?
=> {'type': 'WeatherIntent', 'confidence': 0.8571428571428571, 'args': [('location', 'California')}
ลองไปดูได้ครับ!
เรามาเริ่มเขียนแชทบอทกันดีกว่า
เริ่มจากอิมพอร์ตสิ่งที่เราจะใช้import intentparser
import datetime
import random
ตั้งค่า intentparser สำหรับสิ่งที่เราอยากรู้allIntent = []HelloIntent = ip.intentParser({
'description' : {
"type" : 'HelloIntent',
"args" : [],
"keyword" : [
(ip.REQUIRE, "hello_keyword"),
]},
'hello_keyword' : ['hello', 'hi'],
})
HelloIntent.teachWords(["Hello, How are you?", "Hi, Mr.John", "Hello, nice to meet you!"])
allIntent.append(HelloIntent)ByeIntent = ip.intentParser({
'description' : {
"type" : 'ByeIntent',
"args" : [],
"keyword" : [
(ip.REQUIRE, "bye_keyword"),
]},
'bye_keyword' : ['bye'],
})
ByeIntent.teachWords(["Good bye.", "Bye, Josh.", "Good bye, Be safe."])
allIntent.append(ByeIntent)TimeIntent = ip.intentParser({
'description' : {
"type" : 'TimeIntent',
"args" : [(ip.OPTIONAL, "scopes")],
"keyword" : [
(ip.REQUIRE, "time_keyword"),
(ip.OPTIONAL, "scopes")
]},
'time_keyword' : ['is', 'are', 'what', 'how', 'clock', 'how\'s'],
'scopes' : [
"day",
"time"
]
})
TimeIntent.teachWords(["What time is it?", "How's the clock", "What is this day"])
allIntent.append(TimeIntent)
จะเห็นว่าหลังจากประกาศ intent มาแล้วผมจะ append มันไปที่ allIntent ครับ
ที่ผมทำอย่างนั้นเพราะว่าประโยคที่เราจะรับไม่ได้รับจากแค่ intent เดียว
เพราะว่าแชทบอทของเราตอบคำถามได้ 3 อย่างคือ
- สวัสดี
- ลาก่อน
- แล้วก็บอกเวลา
ไปที่คำถามคำถามเดียว นั่นคือเหตุผลที่เราต้องเลือกคำตอบที่ดีที่สุด จากทุกคำตอบที่เราตอบได้ ซึ่งเราจะเปรียบเทียบมันด้วยค่า confident ครับ ถ้าเราเอามันมาใส่ใน list เดียวกันมันก็จะสามารถเปรียบเทียบได้ง่ายขึ้น
แค่ใช้ max() เท่านั้น
เราเขียนโค้ดลูปสำหรับรับคำถามwhile True:
จากนั้นก็รับคำถามมา แล้วก็ประกาศ list สำหรับตัวแปรชั่วคราวtext = input('User said : ').lower()
temp = []
ต่อไปก็ลูป intent ทุกตัวที่เรามีfor i in allIntent:
_temp = i.getResult(text)
try:
temp.append((_temp['confidence'], _temp['type']))
except Exception as e:
pass
ถ้า intent ของเราตอบคำถามได้ก็จะนำไปเก็บในตัวแปรชั่วคราวที่เราสร้างไว้try:
candidate = max(temp)
if candidate[1] == 'HelloIntent':
print(random.choice(['Chatbot said : Hello!',
'Chatbot said : How are you?',
'Chatbot said : What\'s up!']))
จากนั้นก็ใช้ max() ในการหาคำตอบที่ดีที่สุด แล้วค่อยไปเช็คว่าคำตอบที่ดีที่สุดมันคืออะไร แล้วค่อยตอบด้วยครับตอบนั้นๆ จะเห็นว่าเราใช้ random มาเพื่อไม่ให้คำตอบซ้ำเหมือนเดิมทุกๆครั้งด้วยครับelif candidate[1] == 'ByeIntent':
print(random.choice(['Chatbot said : Good bye!',
'Chatbot said : Good Luck!',
'Chatbot said : See ya!']))
break
อันนี้ก็เหมือนกับข้างบนแต่ต่างกันที่ถ้าพูด bye แล้วโปรแกรมจะหยุดลูปและหยุดการทำงานelif candidate[1] == 'TimeIntent':
typeOfTime = TimeIntent.getResult(text)['args']
typeOfTime = [item for item in typeOfTime if item[0] == 'scopes'][0][1]
now = datetime.datetime.now()
if 'day' in typeOfTime:
print("Chatbot said : Today is {}/{}/{}".format(now.day, now.month, now.year))
else:
print("Chatbot said : It's {}:{}".format(now.hour, now.minute if now.minute > 9 else "0" + str(now.minute)))
del typeOfTime, now
อันนี้เราอยากรู้เวลา แต่ Intent เวลาสามารถคืนค่าประเภทมาได้ด้วยว่าถามวันที่หรือถามเวลาครับ เราเลยเช็คมาว่าถามอะไรแล้วก็ตอบไปตามนั้นelse:
print('error')
del text, temp, _temp, candidate
except Exception as e:
print('Error')
ถ้าเราหาคำตอบที่ดีที่สุดไม่ได้ก็จะตอบว่า error และถ้ามันมีปัญหาอะไรซักอย่างก็จะบอกว่า error เหมือนกัน
เสร็จแล้วครับ มาลองเลยดีกว่า
สำหรับวันนี้ก็มีแค่เท่านี้ครับ บทความนี้อาจจะไม่ค่อยสละสลวยเท่าไรเพราะผมเอาบทความเก่าๆมาย่อยครับ เอาเป็นว่าถ้าผิดพลาดตรงไหนก็ขอโทษมา ณ ที่นี้ด้วยนะครับ
ป.ล. ถ้าใครชอบรบกวนกดปุ่ม clap หรือ follow ให้หน่อยนะครับขอบคุณมากครับ