텔레그램 봇과 명령과 응답을 양방향으로 주고 받는 방법을 구현해보겠습니다. (Python 3 사용)
- 텔레그램 봇을 구현하는 방법에는 두가지가 있는데, 그 중 한가지는 웹훅을 사용하는 방식으로 고정된 주소를 가진 사용자 서버를 등록시켜놓고 이를 통해 서비스하는 방법인데 아무래도 개인 사용자가 하기에는 부담이 따르겠죠.
- 다른 한가지는 Long-Polling을 사용하는 방식인데, 서버를 열어놓지 않고 새로운 메시지가 있는지 여부를 약간 긴 주기로 계속적으로 확인하는 방법으로, 고정된 주소가 없는 개인 PC로도 서비스할 수 있는 장점이 있습니다. 여기서는 Long-Polling을 사용한 방식을 정리해보겠습니다.
다음은 Long-Polling에 의한 요청 처리가 구현되어, 기본적인 명령과 응답을 주고받는 기능이 동작하는 예제입니다.
- 입력된 요청 문자열을 텔레그램 봇 창에 메세지로 보내는 내용이 구현되어있습니다.
- 앱 상에서 요청 입력은 채널 포스팅에서 사용되었던 채널창이 아닌 텔레그램 봇 자체창에서 입력되어야 합니다
소스 중 <텔레그램 봇 토큰>은 아래 글을 참고하여 적당한 값으로 바꿔줍니다.
telegramagent.py :
# -*- coding: utf-8 -*- import http.client, urllib import json import datetime class ChatInfo: def __init__(self): self.chatId = None self.firstName = None self.lastName= None self.userName = None self.text = None self.date = None class TelegramAgent: API_URL = "api.telegram.org" CONTENT_TYPE = "Content-type" APPLICATION_JSON = "application/json" BOT_PATH = "/bot%s" SEND_MESSAGE_PATH = "/sendMessage" SEND_MESSAGE_DATA = '{"chat_id": %s, "text": "%s"}' GET_ME_PATH = "/getMe" GET_UPDATES_PATH = "/getUpdates" GET_UPDATES_DATA = '{"offset": "%d"}' SEND_PHOTO_PATH = "/sendPhoto" SEND_PHOTO_DATA = '{"chat_id": %s, "photo": "%s"}' def __init__(self): self.bContinue = True self.response = None self.callback = None self.bDumpData = False def setToken(self, token): self.token = token def setCallback(self, callback): self.callback = callback def setContinue(self, bContinue): self.bContinue = bContinue def setDumpData(self, bDumpData): self.bDumpData = bDumpData def convertDate(self, date, format): return datetime.datetime.fromtimestamp(date).strftime(format) def packChatId(self, chatId): if len(chatId) > 0 and chatId[:1] == "@": return "\"%s\"" % chatId return chatId def postRequest(self, url, path, headers, paramsRaw): try: params = paramsRaw conn = http.client.HTTPSConnection(url) conn.request("POST", path, params.encode("UTF-8"), headers) self.response = conn.getresponse() self.data = self.response.read().decode("UTF-8") if self.bDumpData: print(self.data) conn.close() except (ConnectionError, TimeoutError) as e: print(f'Catched : {e}') def sendCommon(self, actionPath, data): path = self.BOT_PATH % self.token + actionPath self.postRequest(self.API_URL, path, {self.CONTENT_TYPE : self.APPLICATION_JSON}, data); def sendNoData(self, actionPath): self.sendCommon(actionPath, "") def sendMessage(self, chatId, message): chatId = self.packChatId(chatId) data = self.SEND_MESSAGE_DATA % (chatId, message) self.sendCommon(self.SEND_MESSAGE_PATH, data) def sendPhoto(self, chatId, photoUrl): data = self.SEND_PHOTO_DATA % (chatId, photoUrl) self.sendCommon(self.SEND_PHOTO_PATH, data) def getMe(self): self.sendNoData(self.GET_ME_PATH) def getUpdates(self, offset): data = self.GET_UPDATES_DATA % offset self.sendCommon(self.GET_UPDATES_PATH, data) def processResult(self, result): chatInfo = ChatInfo() try: message = result['message'] if message == None: print("message is null") else: chat = message['chat'] if chat == None: print("chat is null") else: chatInfo.chatId = "%d" % chat['id'] chatInfo.firstName = chat['first_name'] chatInfo.lastName = chat['last_name'] chatInfo.userName = None if 'username' in chat: # username 키 값 존재 체크하도록. 설정 안하면 없을 수 있음 chatInfo.userName = chat['username'] chatInfo.text = message['text'] chatInfo.date = message['date'] if self.callback != None: self.callback(chatInfo) except KeyError as e: return def loop(self): self.bContinue = True updateId = 0 while (self.bContinue): self.getUpdates(updateId) if self.response.status == 200: respDict = json.loads(self.data) results = respDict['result'] resultsNum = len(results) if resultsNum > 0: lastResult = results[resultsNum - 1] updateId = lastResult['update_id'] + 1 self.processResult(lastResult)
main.py :
# -*- coding: utf-8 -*- from telegramagent import * def myCallback(chatInfo): print("chat id = %s" % chatInfo.chatId) print("username = %s" % chatInfo.userName) print("date = %s" % agent.convertDate(chatInfo.date, '%Y-%m-%d %H:%M:%S')) print("text = %s" % chatInfo.text) if chatInfo.text == "quit": agent.setContinue(False) else: agent.sendMessage(chatInfo.chatId, "입력된 요청 : %s" % chatInfo.text) agent = TelegramAgent() agent.setDumpData(True) agent.setToken("<텔레그램 봇 토큰>") agent.setCallback(myCallback) agent.loop()
telegramagent.py 주요 행 설명
- 7 : 요청을 보낸 사용자 정보가 담기는 클래스입니다.
- 77 : chatId가 채널 ID일 경우 따옴표로 감싸줍니다.
- 133 : Long-Polling 시, update id를 0으로 설정할 경우, 기존에 가지고 있는 상태 데이터를 모두 가져오고, 두번째 호출부터는 마지막 update id + 1로 설정하면 기존의 데이터는 무시하고 새로운 데이터만 가져오게 됩니다.
main.py 주요 행 설명
- 11 : quit 명령을 받으면 서버를 종료합니다.
- 13 : Callback 함수에서 입력된 요청 문자열을 텔레그램 봇 창에 메세지로 보내줍니다.
- 16 : 텔레그램 서버에서 받은 응답 정보를 출력합니다. 실사용시는 False로 합니다.
- 18 : 요청을 처리할 Callback 함수를 등록하여 사용합니다.
정찰 위성 이미지
참고 사이트
- 더 많은 텔레그램 봇 라이브러리들 : https://core.telegram.org/bots/samples
<테스트 환경>
- OS : Windows 10
- Python 버전 : 3.6
'1. 연구 모듈 > 텔레그램' 카테고리의 다른 글
텔레그램 봇 생성 및 봇 토큰 얻기 (0) | 2018.03.15 |
---|---|
텔레그램 봇으로 채널 포스팅 - 9. Python 사용 예제 (4) (0) | 2018.02.22 |
텔레그램 봇으로 채널 포스팅 - 8. Python 사용 예제 (3) (0) | 2018.02.20 |
텔레그램 봇으로 채널 포스팅 - 7. Clojure 사용 예제 (2) (0) | 2017.09.14 |
텔레그램 봇으로 채널 포스팅 - 6. Python 사용 예제 (2) (1) | 2017.03.21 |