-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathparse_methods.py
102 lines (87 loc) · 3.79 KB
/
parse_methods.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
"""
Parse methods from codeforces.com
Copyright (C) 2021 Vadim Vergasov
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
"""
import requests
from lxml import html
from codeforces_api.api_requests import CodeforcesApi
class CodeforcesParser:
session = None
problem_tags = dict()
def __init__(self):
"""
Initializing class. All we will need is a session to optimize performance.
"""
self.session = requests.Session()
def get_solution(self, contest_id, submit_id):
"""
Returns source code for specified submit.
contest_id is the id of contest.
submit_id is the id of submission.
"""
solutionPage = self.session.get(
"https://codeforces.com/contest/"
+ str(contest_id)
+ "/submission/"
+ str(submit_id),
cookies={
"RCPC": "333e1cab3b9ab19c225e45305669bef8",
"expires": "Thu, 31-Dec-37 23:55:55 GMT",
"path": "/",
},
)
if int(solutionPage.status_code) != 200:
raise Exception("Returned not OK code " + str(solutionPage))
tree2 = html.fromstring(solutionPage.text)
code = tree2.xpath("//pre[@id='program-source-text']/text()")
if len(code) == 0:
raise ValueError("Incorrect contest_id or submit_id " + str(code))
return code[0].replace("\r", "")
def get_tags(self, contest_id, index, include_rating=False):
"""
Get tags in lexicographical order of the given problem.
contest_id is the number of the contest.
index is the number of the problem, better to be a capital letter. Also could be an integer or lowercase letter.
include_rating is bool which indicates include or not task rating.
"""
# If we don't have tags we should get them.
if self.problem_tags == dict():
cf_api = CodeforcesApi()
for problem in cf_api.problemset_problems()["problems"]:
if str(problem.contest_id) not in self.problem_tags.keys():
self.problem_tags[str(problem.contest_id)] = dict()
self.problem_tags[str(problem.contest_id)][
str(problem.index)
] = problem.tags
try:
self.problem_tags[str(problem.contest_id)][
str(problem.index)
].append("*" + str(problem.rating))
except KeyError:
pass
self.problem_tags[str(problem.contest_id)][str(problem.index)].sort()
if isinstance(index, int):
index = chr(ord("A") + index)
elif isinstance(index, str):
if index.isnumeric():
index = chr(ord("A") + int(index))
index = index.capitalize()
try:
if include_rating:
return self.problem_tags[str(contest_id)][index]
return self.problem_tags[str(contest_id)][index][1:]
except KeyError:
# If problem included in more than one division.
if include_rating:
return self.problem_tags[str(int(contest_id) - 1)][index]
return self.problem_tags[str(int(contest_id) - 1)][index][1:]