1
+ import { spawn } from 'node:child_process' ;
2
+
3
+ import fetchCookie from 'fetch-cookie' ;
1
4
import cron from 'node-cron' ;
2
- import fetchCookie from 'fetch-cookie'
3
5
import zulipInit from 'zulip-js' ;
4
6
5
7
6
- import { questionOfTheDay } from './queries.js' ;
7
8
import grind75_problems from './Grind75.json' assert { type : 'json ' } ;
9
+ import { questionOfTheDay } from './queries.js' ;
8
10
9
11
let zulipClient ;
10
12
if ( process . env . ZULIP_USERNAME && process . env . ZULIP_API_KEY && process . env . ZULIP_REALM ) {
@@ -39,21 +41,31 @@ class LeetCodeBot {
39
41
console . log ( `Getting leetcode problem for ${ humanReadableDateString } ` ) ;
40
42
41
43
try {
42
- const response = await fetch_with_cookies ( 'https://leetcode.com/graphql' , {
44
+ const response = await fetch ( 'https://leetcode.com/graphql' , {
43
45
method : 'POST' ,
44
46
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' ,
47
50
} ,
48
51
body : `{"query":"${ questionOfTheDay } ","operationName":"questionOfToday"}`
49
52
} ) ;
50
53
51
54
let leetcode_data ;
52
55
if ( response . ok ) {
53
56
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 ( ) ;
54
63
} else {
55
64
console . error ( 'There was a problem fetching data from the Leetcode API.' ) ;
65
+ console . error ( response . status , response . statusText ) ;
66
+ console . groupCollapsed ( 'Response body:' )
56
67
console . error ( await response . text ( ) ) ;
68
+ console . groupEnd ( ) ;
57
69
}
58
70
59
71
// 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 {
95
107
}
96
108
} catch ( error ) {
97
109
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 ) ;
106
111
console . groupEnd ( ) ;
107
112
}
108
113
} , {
@@ -123,7 +128,7 @@ class LeetCodeBot {
123
128
leetcode_message = `1. (${ problems . leetcode_daily . difficulty } ) [${ problems . leetcode_daily . title } ](${ baseLeetcodeURL } ${ problems . leetcode_daily . link } )`
124
129
} else {
125
130
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/).`
127
132
}
128
133
129
134
let message = `${ date }
@@ -197,7 +202,7 @@ ${Object.entries(problems.grind75.problems).reduce((acc, [difficulty, problem])
197
202
{
198
203
type : "link" ,
199
204
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' ,
201
206
}
202
207
]
203
208
}
@@ -288,4 +293,27 @@ ${Object.entries(problems.grind75.problems).reduce((acc, [difficulty, problem])
288
293
*/
289
294
const random = ( max ) => Math . floor ( Math . random ( ) * ( max + 1 ) ) ;
290
295
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
+
291
319
LeetCodeBot . run ( ) ;
0 commit comments