Skip to content

Commit 96ff999

Browse files
committed
feat: add python script to get leetcode qotd in case nodejs gets blocked by cloudflare bot detection
1 parent b9d4463 commit 96ff999

File tree

3 files changed

+93
-15
lines changed

3 files changed

+93
-15
lines changed

bot.js

+43-15
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,12 @@
1+
import { spawn } from 'node:child_process';
2+
3+
import fetchCookie from 'fetch-cookie';
14
import cron from 'node-cron';
2-
import fetchCookie from 'fetch-cookie'
35
import zulipInit from 'zulip-js';
46

57

6-
import { questionOfTheDay } from './queries.js';
78
import grind75_problems from './Grind75.json' assert { type: 'json' };
9+
import { questionOfTheDay } from './queries.js';
810

911
let zulipClient;
1012
if (process.env.ZULIP_USERNAME && process.env.ZULIP_API_KEY && process.env.ZULIP_REALM) {
@@ -39,21 +41,31 @@ class LeetCodeBot {
3941
console.log(`Getting leetcode problem for ${humanReadableDateString}`);
4042

4143
try {
42-
const response = await fetch_with_cookies('https://leetcode.com/graphql', {
44+
const response = await fetch('https://leetcode.com/graphql', {
4345
method: 'POST',
4446
headers: {
45-
referer: 'https://leetcode.com/problemset/',
46-
'Content-Type': 'application/json',
47+
authority: 'leetcode.com',
48+
'Content-Type': 'application/json',
49+
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36',
4750
},
4851
body: `{"query":"${questionOfTheDay}","operationName":"questionOfToday"}`
4952
});
5053

5154
let leetcode_data;
5255
if (response.ok) {
5356
leetcode_data = (await response.json()).activeDailyCodingChallengeQuestion;
57+
} else if (response.headers.get('cf-mitigated')) {
58+
console.error('Cloudflare blocked the request to the Leetcode API.');
59+
console.error(response.status, response.statusText);
60+
console.group('Getting data from python script...');
61+
leetcode_data = (await run_python()).data.activeDailyCodingChallengeQuestion;
62+
console.groupEnd();
5463
} else {
5564
console.error('There was a problem fetching data from the Leetcode API.');
65+
console.error(response.status, response.statusText);
66+
console.groupCollapsed('Response body:')
5667
console.error(await response.text());
68+
console.groupEnd();
5769
}
5870

5971
// Choose problems from grind75 by selecting a random topic and then random questions for each difficulty from that topic
@@ -95,14 +107,7 @@ class LeetCodeBot {
95107
}
96108
} catch (error) {
97109
console.group('Error getting problems:');
98-
if (error.response?.headers) {
99-
console.error(error.response.status);
100-
// console.group('Response Headers');
101-
// console.error(error.response.headers);
102-
console.groupEnd();
103-
} else {
104-
console.error(error);
105-
}
110+
console.error(error);
106111
console.groupEnd();
107112
}
108113
}, {
@@ -123,7 +128,7 @@ class LeetCodeBot {
123128
leetcode_message = `1. (${problems.leetcode_daily.difficulty}) [${problems.leetcode_daily.title}](${baseLeetcodeURL}${problems.leetcode_daily.link})`
124129
} else {
125130
leetcode_message = `> There was a problem accessing the leetcode API.
126-
> Find the daily problem on the calenadar [here](https://leetcode.com/problemset/).`
131+
> Find the daily problem on the calendar [here](https://leetcode.com/problemset/).`
127132
}
128133

129134
let message = `${date}
@@ -197,7 +202,7 @@ ${Object.entries(problems.grind75.problems).reduce((acc, [difficulty, problem])
197202
{
198203
type: "link",
199204
url: `${baseLeetcodeURL}${problems.leetcode_daily ? problems.leetcode_daily.link : '/problemset'}`,
200-
text: problems.leetcode_daily ? problems.leetcode_daily.title : 'Find the daily problem on the calenadar here',
205+
text: problems.leetcode_daily ? problems.leetcode_daily.title : 'Find the daily problem on the calendar here',
201206
}
202207
]
203208
}
@@ -288,4 +293,27 @@ ${Object.entries(problems.grind75.problems).reduce((acc, [difficulty, problem])
288293
*/
289294
const random = (max) => Math.floor(Math.random() * (max + 1));
290295

296+
/**
297+
* Returns a JSON object containing the result of the get_daily_question.py script
298+
* @returns { Promise<object> }
299+
*/
300+
async function run_python() {
301+
return new Promise((resolve, reject) => {
302+
let jsonFromPython;
303+
304+
const python = spawn('python', ['./get_daily_question.py']);
305+
306+
python.stdout.on('data', function (data) {
307+
jsonFromPython = JSON.parse(data.toString());
308+
});
309+
310+
python.on('close', (code) => {
311+
console.log(`Python script finished with code ${code}`);
312+
console.log('Data from python script:');
313+
console.log(jsonFromPython);
314+
resolve(jsonFromPython);
315+
});
316+
});
317+
}
318+
291319
LeetCodeBot.run();

get_daily_question.py

+49
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
import requests
2+
import json
3+
4+
payload = {
5+
"query": """
6+
query questionOfToday {
7+
activeDailyCodingChallengeQuestion {
8+
date
9+
userStatus
10+
link
11+
question {
12+
acRate
13+
difficulty
14+
freqBar
15+
frontendQuestionId: questionFrontendId
16+
isFavor
17+
paidOnly: isPaidOnly
18+
status
19+
title
20+
titleSlug
21+
hasVideoSolution
22+
hasSolution
23+
topicTags {
24+
name
25+
id
26+
slug
27+
}
28+
}
29+
}
30+
}
31+
""",
32+
"variables": {},
33+
"operationName": "questionOfToday",
34+
}
35+
36+
headers = {
37+
"authority": "leetcode.com",
38+
"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36",
39+
}
40+
41+
req = requests.Request("POST", "https://leetcode.com/graphql/", headers=headers, json=payload)
42+
r = req.prepare()
43+
s = requests.Session()
44+
response = s.send(r)
45+
46+
if response.status_code == 200:
47+
print(json.dumps(response.json()))
48+
else:
49+
print("Failed to fetch data. Status code:", response.status_code, response.text)

requirements.txt

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
requests==2.31.0

0 commit comments

Comments
 (0)