forked from LichessBot-Coders/Lichess-Coded-Bot
-
Notifications
You must be signed in to change notification settings - Fork 12
/
lichess.py
105 lines (86 loc) · 3.9 KB
/
lichess.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
import requests
from urllib.parse import urljoin
from requests.exceptions import ConnectionError, HTTPError, ReadTimeout
from urllib3.exceptions import ProtocolError
from http.client import RemoteDisconnected
import backoff
ENDPOINTS = {
"profile": "/api/account",
"playing": "/api/account/playing",
"stream": "/api/bot/game/stream/{}",
"stream_event": "/api/stream/event",
"game": "/api/bot/game/{}",
"move": "/api/bot/game/{}/move/{}",
"chat": "/api/bot/game/{}/chat",
"abort": "/api/bot/game/{}/abort",
"accept": "/api/challenge/{}/accept",
"decline": "/api/challenge/{}/decline",
"upgrade": "/api/bot/account/upgrade",
"resign": "/api/bot/game/{}/resign"
}
# docs: https://lichess.org/api
class Lichess:
def __init__(self, token, url, version):
self.version = version
self.header = {
"Authorization": "Bearer {}".format(token)
}
self.baseUrl = url
self.session = requests.Session()
self.session.headers.update(self.header)
self.set_user_agent("?")
def is_final(exception):
return isinstance(exception, HTTPError) and exception.response.status_code < 500
@backoff.on_exception(backoff.constant,
(RemoteDisconnected, ConnectionError, ProtocolError, HTTPError, ReadTimeout),
max_time=60,
interval=0.1,
giveup=is_final)
def api_get(self, path):
url = urljoin(self.baseUrl, path)
response = self.session.get(url, timeout=2)
response.raise_for_status()
return response.json()
@backoff.on_exception(backoff.constant,
(RemoteDisconnected, ConnectionError, ProtocolError, HTTPError, ReadTimeout),
max_time=60,
interval=0.1,
giveup=is_final)
def api_post(self, path, data=None, headers=None):
url = urljoin(self.baseUrl, path)
response = self.session.post(url, data=data, headers=headers, timeout=2)
response.raise_for_status()
return response.json()
def get_game(self, game_id):
return self.api_get(ENDPOINTS["game"].format(game_id))
def upgrade_to_bot_account(self):
return self.api_post(ENDPOINTS["upgrade"])
def make_move(self, game_id, move):
return self.api_post(ENDPOINTS["move"].format(game_id, move))
def chat(self, game_id, room, text):
payload = {'room': room, 'text': text}
return self.api_post(ENDPOINTS["chat"].format(game_id), data=payload)
def abort(self, game_id):
return self.api_post(ENDPOINTS["abort"].format(game_id))
def get_event_stream(self):
url = urljoin(self.baseUrl, ENDPOINTS["stream_event"])
return requests.get(url, headers=self.header, stream=True)
def get_game_stream(self, game_id):
url = urljoin(self.baseUrl, ENDPOINTS["stream"].format(game_id))
return requests.get(url, headers=self.header, stream=True)
def accept_challenge(self, challenge_id):
return self.api_post(ENDPOINTS["accept"].format(challenge_id))
def decline_challenge(self, challenge_id, reason="generic"):
return self.api_post(ENDPOINTS["decline"].format(challenge_id), data=f"reason={reason}", headers={"Content-Type": "application/x-www-form-urlencoded"})
def get_profile(self):
profile = self.api_get(ENDPOINTS["profile"])
self.set_user_agent(profile["username"])
return profile
def get_ongoing_games(self):
ongoing_games = self.api_get(ENDPOINTS["playing"])["nowPlaying"]
return ongoing_games
def resign(self, game_id):
self.api_post(ENDPOINTS["resign"].format(game_id))
def set_user_agent(self, username):
self.header.update({"User-Agent": "lichess-bot/{} user:{}".format(self.version, username)})
self.session.headers.update(self.header)