randydev commited on
Commit
aefdd02
Β·
verified Β·
1 Parent(s): 2aca136

Upload join_request.py

Browse files
Files changed (1) hide show
  1. chatbot/plugins/join_request.py +229 -62
chatbot/plugins/join_request.py CHANGED
@@ -31,31 +31,85 @@ logger = logging.getLogger(__name__)
31
 
32
  user_list = []
33
  captcha_texts = {}
 
34
 
35
- def thanks_hacker_by_randydev():
36
- url = "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcR0u8UqJ2JDhfmPOCb_zAHjUQG2NYMjTwLbkq_sQhCQOxX8hn66YbaGFvLL&s=10"
37
- response = requests.get(url)
38
- image_hacker = "hacker.png"
39
- with open(image_hacker, "wb") as f:
40
- f.write(response.content)
41
- return image_hacker
42
 
43
- async def remove_captcha_after_timeout(client, user_id, delay=300):
44
- await asyncio.sleep(delay)
45
- if user_id in captcha_texts:
46
- del captcha_texts[user_id]
47
- await client.send_message(user_id, "⏰ Your CAPTCHA verification has expired. Please try again.")
48
- logger.info(f"CAPTCHA for user {user_id} has expired and been removed.")
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
49
 
50
- def generate_captcha_multiple_choice():
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
51
  letters = string.ascii_uppercase + string.digits
52
  captcha_text = ''.join(random.choice(letters) for _ in range(5))
 
53
  choices = [captcha_text]
54
- for _ in range(2):
55
  wrong_choice = ''.join(random.choice(letters) for _ in range(5))
56
- while wrong_choice in choices:
57
- wrong_choice = ''.join(random.choice(letters) for _ in range(5))
58
- choices.append(wrong_choice)
59
  random.shuffle(choices)
60
 
61
  width, height = 200, 80
@@ -68,7 +122,7 @@ def generate_captcha_multiple_choice():
68
  y = random.randint(0, height)
69
  noise_color = (random.randint(150, 200), random.randint(150, 200), random.randint(150, 200))
70
  d.point((x, y), fill=noise_color)
71
-
72
  try:
73
  font = ImageFont.truetype("arial.ttf", 45)
74
  except IOError:
@@ -77,25 +131,99 @@ def generate_captcha_multiple_choice():
77
  text_width, text_height = d.textsize(captcha_text, font=font)
78
  text_x = (width - text_width) / 2
79
  text_y = (height - text_height) / 2
80
-
81
  text_image = Image.new('RGBA', (text_width, text_height), (255, 255, 255, 0))
82
  text_draw = ImageDraw.Draw(text_image)
83
  text_draw.text((0, 0), captcha_text, font=font, fill=(0, 0, 0))
84
  rotated_text = text_image.rotate(random.randint(-25, 25), expand=1)
85
  img.paste(rotated_text, (int(text_x), int(text_y)), rotated_text)
86
 
 
 
 
 
 
 
 
 
87
  for _ in range(5):
88
  start = (random.randint(0, width), random.randint(0, height))
89
  end = (random.randint(0, width), random.randint(0, height))
90
  line_color = (random.randint(0, 150), random.randint(0, 150), random.randint(0, 150))
91
  d.line([start, end], fill=line_color, width=2)
92
-
93
  img = img.filter(ImageFilter.BLUR)
94
- img_path = f"captcha_{captcha_text}.png"
95
  img.save(img_path)
96
- return captcha_text, img_path, choices
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
97
 
98
- @Client.on_chat_join_request()
99
  async def join_request(client: Client, event: ChatJoinRequest):
100
  member = await client.get_chat_member(event.chat.id, "me")
101
  if member.status != ChatMemberStatus.ADMINISTRATOR:
@@ -108,19 +236,19 @@ async def join_request(client: Client, event: ChatJoinRequest):
108
  except ChatAdminRequired:
109
  await client.send_message(event.chat.id, text="I need to be an administrator to perform this action.")
110
  return
111
- captcha_text, img_path, choices = generate_captcha_multiple_choice()
112
- captcha_texts[event.from_user.id] = captcha_text
 
113
  captcha_texts["chat_id"] = event.chat.id
114
  captcha_texts["chat_link"] = chat_link
115
- keyboard = InlineKeyboardMarkup(
116
- [
117
- [InlineKeyboardButton(choice, callback_data=f"verify_{event.from_user.id}_{choice}")]
118
- for choice in choices
119
- ] + [
120
- [InlineKeyboardButton("πŸ”„ Refresh CAPTCHA", callback_data="refresh_captcha")],
121
- [InlineKeyboardButton("❌ Cancel", callback_data="cancel_captcha")]
122
- ]
123
- )
124
  if event.chat.type == ChatType.SUPERGROUP:
125
  try:
126
  await client.send_message(
@@ -164,7 +292,7 @@ async def close_final(client: Client, cb: CallbackQuery):
164
  def create_button_join_group(chat_link):
165
  return InlineKeyboardMarkup(
166
  [
167
- [InlineKeyboardButton("πŸ‘οΈ Join chat", url=chat_link)],
168
  [InlineKeyboardButton("πŸ”˜ Close", callback_data="close")],
169
  ]
170
  )
@@ -180,45 +308,57 @@ def create_button_userinfo(user_id, username):
180
  @Client.on_callback_query(filters.regex("^refresh_captcha$"))
181
  async def refresh_captcha_callback(client: Client, cb: CallbackQuery):
182
  user_id = cb.from_user.id
 
183
  if user_id in captcha_texts:
184
  del captcha_texts[user_id]
185
- captcha_text, img_path, choices = generate_captcha_multiple_choice()
186
- captcha_texts[user_id] = captcha_text
187
- keyboard = InlineKeyboardMarkup(
188
- [
189
- [InlineKeyboardButton("πŸ”„ Refresh CAPTCHA", callback_data="refresh_captcha")],
190
- [InlineKeyboardButton("βœ… Verify", callback_data="verify_captcha")],
191
- [InlineKeyboardButton("❌ Cancel", callback_data="cancel_captcha")]
192
- ]
193
- )
194
- await cb.edit_message_media(
195
- media=InputMediaPhoto(
196
- img_path,
197
- caption=f"❗️ **Verify that you are human!**\n\n❔ Please select the correct CAPTCHA text shown in the image below.",
198
- ),
199
- reply_markup=keyboard
200
- )
 
 
 
 
 
201
  os.remove(img_path)
202
- await cb.answer("CAPTCHA refreshed!", show_alert=False)
203
 
204
  @Client.on_callback_query(filters.regex("^verify_"))
205
- async def verify_captcha_multiple_choice_callback(client: Client, cb: CallbackQuery):
206
  data = cb.data.split("_")
207
  if len(data) != 3:
208
- await cb.answer("Invalid data format.", show_alert=True)
209
  return
210
  _, user_id_str, user_choice = data
211
  try:
212
  user_id = int(user_id_str)
213
  except ValueError:
214
- await cb.answer("Invalid user ID.", show_alert=True)
 
 
 
 
215
  return
216
  if user_id not in captcha_texts:
217
- await cb.answer("❗️ Please start the CAPTCHA verification first.", show_alert=True)
 
218
  return
219
- correct_captcha = captcha_texts.get(user_id)
220
- if user_choice == correct_captcha:
221
- hacker_image = thanks_hacker_by_randydev()
 
222
  await cb.edit_message_media(
223
  media=InputMediaPhoto(
224
  hacker_image,
@@ -226,7 +366,7 @@ async def verify_captcha_multiple_choice_callback(client: Client, cb: CallbackQu
226
  ),
227
  reply_markup=create_button_join_group(captcha_texts.get("chat_link"))
228
  )
229
- logger.info(f"User {user_id} successfully verified CAPTCHA.")
230
  await client.approve_chat_join_request(
231
  chat_id=captcha_texts.get("chat_id"),
232
  user_id=user_id
@@ -235,14 +375,41 @@ async def verify_captcha_multiple_choice_callback(client: Client, cb: CallbackQu
235
  del captcha_texts["chat_id"]
236
  del captcha_texts["chat_link"]
237
  else:
238
- await cb.edit_message_text("❌ Incorrect CAPTCHA. Please try again")
 
 
 
 
 
239
  await client.decline_chat_join_request(
240
  chat_id=captcha_texts.get("chat_id"),
241
  user_id=user_id
242
  )
 
243
  del captcha_texts[user_id]
244
  del captcha_texts["chat_id"]
245
  del captcha_texts["chat_link"]
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
246
 
247
  @Client.on_callback_query(filters.regex("^verify_captcha$"))
248
  async def verify_captcha_callback(client: Client, cb: CallbackQuery):
 
31
 
32
  user_list = []
33
  captcha_texts = {}
34
+ captcha_modes = {}
35
 
36
+ def generate_math_captcha():
37
+ operations = ['+', '-', '*']
38
+ operation = random.choice(operations)
 
 
 
 
39
 
40
+ num1 = random.randint(1, 20)
41
+ num2 = random.randint(1, 20)
42
+
43
+ if operation == '+':
44
+ correct_answer = num1 + num2
45
+ elif operation == '-':
46
+ correct_answer = num1 - num2
47
+ else:
48
+ correct_answer = num1 * num2
49
+
50
+ captcha_text = f"{num1} {operation} {num2} = ?"
51
+
52
+ choices = [str(correct_answer)]
53
+ while len(choices) < 3:
54
+ wrong_answer = correct_answer + random.choice([-5, -3, -2, 2, 3, 5])
55
+ if str(wrong_answer) not in choices:
56
+ choices.append(str(wrong_answer))
57
+ random.shuffle(choices)
58
+
59
+ width, height = 250, 100
60
+ background_color = (random.randint(200, 255), random.randint(200, 255), random.randint(200, 255)) # Warna pastel
61
+ img = Image.new('RGB', (width, height), color=background_color)
62
+ d = ImageDraw.Draw(img)
63
+
64
+ for _ in range(1000):
65
+ x = random.randint(0, width)
66
+ y = random.randint(0, height)
67
+ noise_color = (random.randint(150, 200), random.randint(150, 200), random.randint(150, 200))
68
+ d.point((x, y), fill=noise_color)
69
+
70
+ try:
71
+ font = ImageFont.truetype("arial.ttf", 40)
72
+ except IOError:
73
+ font = ImageFont.load_default()
74
 
75
+ text_width, text_height = d.textsize(captcha_text, font=font)
76
+ text_x = (width - text_width) / 2
77
+ text_y = (height - text_height) / 2 - 20
78
+
79
+ text_image = Image.new('RGBA', (text_width, text_height), (255, 255, 255, 0))
80
+ text_draw = ImageDraw.Draw(text_image)
81
+ text_draw.text((0, 0), captcha_text, font=font, fill=(0, 0, 0))
82
+ rotated_text = text_image.rotate(random.randint(-25, 25), expand=1)
83
+ img.paste(rotated_text, (int(text_x), int(text_y)), rotated_text)
84
+
85
+ choice_font = ImageFont.truetype("arial.ttf", 30) if font else ImageFont.load_default()
86
+ for idx, choice in enumerate(choices):
87
+ choice_text = f"{idx + 1}. {choice}"
88
+ choice_width, choice_height = d.textsize(choice_text, font=choice_font)
89
+ choice_x = 50 + idx * 80
90
+ choice_y = height - choice_height - 20
91
+ d.text((choice_x, choice_y), choice_text, font=choice_font, fill=(0, 0, 0))
92
+
93
+ for _ in range(5):
94
+ start = (random.randint(0, width), random.randint(0, height))
95
+ end = (random.randint(0, width), random.randint(0, height))
96
+ line_color = (random.randint(0, 150), random.randint(0, 150), random.randint(0, 150))
97
+ d.line([start, end], fill=line_color, width=2)
98
+
99
+ img = img.filter(ImageFilter.BLUR)
100
+ img_path = f"captcha_math_{captcha_text}.png"
101
+ img.save(img_path)
102
+ return captcha_text, img_path, choices, correct_answer
103
+
104
+ def generate_text_captcha():
105
  letters = string.ascii_uppercase + string.digits
106
  captcha_text = ''.join(random.choice(letters) for _ in range(5))
107
+
108
  choices = [captcha_text]
109
+ while len(choices) < 3:
110
  wrong_choice = ''.join(random.choice(letters) for _ in range(5))
111
+ if wrong_choice not in choices:
112
+ choices.append(wrong_choice)
 
113
  random.shuffle(choices)
114
 
115
  width, height = 200, 80
 
122
  y = random.randint(0, height)
123
  noise_color = (random.randint(150, 200), random.randint(150, 200), random.randint(150, 200))
124
  d.point((x, y), fill=noise_color)
125
+
126
  try:
127
  font = ImageFont.truetype("arial.ttf", 45)
128
  except IOError:
 
131
  text_width, text_height = d.textsize(captcha_text, font=font)
132
  text_x = (width - text_width) / 2
133
  text_y = (height - text_height) / 2
134
+
135
  text_image = Image.new('RGBA', (text_width, text_height), (255, 255, 255, 0))
136
  text_draw = ImageDraw.Draw(text_image)
137
  text_draw.text((0, 0), captcha_text, font=font, fill=(0, 0, 0))
138
  rotated_text = text_image.rotate(random.randint(-25, 25), expand=1)
139
  img.paste(rotated_text, (int(text_x), int(text_y)), rotated_text)
140
 
141
+ choice_font = ImageFont.truetype("arial.ttf", 30) if font else ImageFont.load_default()
142
+ for idx, choice in enumerate(choices):
143
+ choice_text = f"{idx + 1}. {choice}"
144
+ choice_width, choice_height = d.textsize(choice_text, font=choice_font)
145
+ choice_x = 50 + idx * 80
146
+ choice_y = height - choice_height - 20
147
+ d.text((choice_x, choice_y), choice_text, font=choice_font, fill=(0, 0, 0))
148
+
149
  for _ in range(5):
150
  start = (random.randint(0, width), random.randint(0, height))
151
  end = (random.randint(0, width), random.randint(0, height))
152
  line_color = (random.randint(0, 150), random.randint(0, 150), random.randint(0, 150))
153
  d.line([start, end], fill=line_color, width=2)
154
+
155
  img = img.filter(ImageFilter.BLUR)
156
+ img_path = f"captcha_text_{captcha_text}.png"
157
  img.save(img_path)
158
+ return captcha_text, img_path, choices, captcha_text
159
+
160
+ def generate_captcha(user_id, mode='math'):
161
+ if mode == 'math':
162
+ return generate_math_captcha()
163
+ else:
164
+ return generate_text_captcha()
165
+
166
+ def thanks_hacker_by_randydev():
167
+ url = "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcR0u8UqJ2JDhfmPOCb_zAHjUQG2NYMjTwLbkq_sQhCQOxX8hn66YbaGFvLL&s=10"
168
+ response = requests.get(url)
169
+ image_hacker = "hacker.png"
170
+ with open(image_hacker, "wb") as f:
171
+ f.write(response.content)
172
+ return image_hacker
173
+
174
+ def failed_hacker_by_randydev():
175
+ url = "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcTM0vb4s59H9F1-_7FyELiLU04e8bCHy7o6KQV2mG3DFRLnzP547KjckKG2&s=10"
176
+ response = requests.get(url)
177
+ image_hacker = "failed_hacker.png"
178
+ with open(image_hacker, "wb") as f:
179
+ f.write(response.content)
180
+ return image_hacker
181
+
182
+ async def remove_captcha_after_timeout(client, user_id, delay=300):
183
+ await asyncio.sleep(delay)
184
+ if user_id in captcha_texts:
185
+ del captcha_texts[user_id]
186
+ await client.send_message(user_id, "⏰ Your CAPTCHA verification has expired. Please try again.")
187
+ logger.info(f"CAPTCHA for user {user_id} has expired and been removed.")
188
+
189
+ @Client.on_message(filters.command("settingmode") & filters.private)
190
+ async def setting_mode(client: Client, message: Message):
191
+ user_id = message.from_user.id
192
+ keyboard = InlineKeyboardMarkup(
193
+ [
194
+ [InlineKeyboardButton("πŸ”’ Mathematics", callback_data="mode_math")],
195
+ [InlineKeyboardButton("πŸ”€ Text", callback_data="mode_text")],
196
+ [InlineKeyboardButton("❌ Cancel", callback_data="cancel_mode")]
197
+ ]
198
+ )
199
+ await message.reply_text(
200
+ "πŸ“ Select the CAPTCHA mode you want:",
201
+ reply_markup=keyboard
202
+ )
203
+
204
+ @Client.on_callback_query(filters.regex("^mode_"))
205
+ async def mode_callback(client: Client, cb: CallbackQuery):
206
+ user_id = cb.from_user.id
207
+ data = cb.data.split("_")
208
+ if len(data) != 2:
209
+ await cb.answer("Invalid selection.", show_alert=True)
210
+ return
211
+ mode = data[1]
212
+ if mode not in ['math', 'text']:
213
+ await cb.answer("Invalid mode.", show_alert=True)
214
+ return
215
+ captcha_modes[user_id] = mode
216
+ await cb.edit_message_text(f"CAPTCHA mode has been set to **{'Mathematics' if mode == 'math' else 'Text'}**.")
217
+ await cb.answer("The mode has been changed.", show_alert=False)
218
+ logger.info(f"User {user_id} set CAPTCHA mode to {mode}.")
219
+
220
+ @Client.on_callback_query(filters.regex("^cancel_mode$"))
221
+ async def cancel_mode_callback(client: Client, cb: CallbackQuery):
222
+ await cb.edit_message_text("CAPTCHA mode setting has been canceled.")
223
+ await cb.answer("Cancellation successful.", show_alert=False)
224
+ logger.info(f"User {cb.from_user.id} canceled CAPTCHA mode setting.")
225
 
226
+ @Client.on_chat_join_request(filters.chat("KillerXSupport"))
227
  async def join_request(client: Client, event: ChatJoinRequest):
228
  member = await client.get_chat_member(event.chat.id, "me")
229
  if member.status != ChatMemberStatus.ADMINISTRATOR:
 
236
  except ChatAdminRequired:
237
  await client.send_message(event.chat.id, text="I need to be an administrator to perform this action.")
238
  return
239
+ mode = captcha_modes.get(event.from_user.id, "math")
240
+ captcha_text, img_path, choices, correct_answer = generate_captcha(event.from_user.id, mode)
241
+ captcha_texts[event.from_user.id] = (captcha_text, correct_answer)
242
  captcha_texts["chat_id"] = event.chat.id
243
  captcha_texts["chat_link"] = chat_link
244
+ buttons = []
245
+ for choice in choices:
246
+ buttons.append([InlineKeyboardButton(choice, callback_data=f"verify_{user_id}_{choice}")])
247
+ buttons.append([
248
+ InlineKeyboardButton("πŸ”„ Refresh CAPTCHA", callback_data="refresh_captcha"),
249
+ InlineKeyboardButton("❌ Cancel", callback_data="cancel_captcha")
250
+ ])
251
+ keyboard = InlineKeyboardMarkup(buttons)
 
252
  if event.chat.type == ChatType.SUPERGROUP:
253
  try:
254
  await client.send_message(
 
292
  def create_button_join_group(chat_link):
293
  return InlineKeyboardMarkup(
294
  [
295
+ [InlineKeyboardButton("πŸ‘οΈ Join chat", url=chat_link")],
296
  [InlineKeyboardButton("πŸ”˜ Close", callback_data="close")],
297
  ]
298
  )
 
308
  @Client.on_callback_query(filters.regex("^refresh_captcha$"))
309
  async def refresh_captcha_callback(client: Client, cb: CallbackQuery):
310
  user_id = cb.from_user.id
311
+ mode = captcha_modes.get(user_id, 'math')
312
  if user_id in captcha_texts:
313
  del captcha_texts[user_id]
314
+ logger.info(f"Old CAPTCHA for user {user_id} has been removed.")
315
+ captcha_text, img_path, choices, correct_answer = generate_captcha(user_id, mode)
316
+ captcha_texts[user_id] = (captcha_text, correct_answer)
317
+ logger.info(f"Generated new {mode} CAPTCHA for user {user_id}: {captcha_text}")
318
+ buttons = []
319
+ for choice in choices:
320
+ buttons.append([InlineKeyboardButton(choice, callback_data=f"verify_{user_id}_{choice}")])
321
+ buttons.append([
322
+ InlineKeyboardButton("πŸ”„ Refresh CAPTCHA", callback_data="refresh_captcha"),
323
+ InlineKeyboardButton("❌ Cancel", callback_data="cancel_captcha")
324
+ ])
325
+ keyboard = InlineKeyboardMarkup(buttons)
326
+ try:
327
+ await cb.edit_message_media(
328
+ media=InputMediaPhoto(open(img_path, "rb")),
329
+ reply_markup=keyboard
330
+ )
331
+ logger.info(f"Updated CAPTCHA image for user {user_id}.")
332
+ except Exception as e:
333
+ await cb.answer("❌ Failed to update CAPTCHA.", show_alert=True)
334
+ logger.error(f"Error refreshing CAPTCHA for user {user_id}: {e}")
335
  os.remove(img_path)
336
+ await cb.answer("πŸ”„ CAPTCHA has been updated!", show_alert=False)
337
 
338
  @Client.on_callback_query(filters.regex("^verify_"))
339
+ async def verify_captcha_callback(client: Client, cb: CallbackQuery):
340
  data = cb.data.split("_")
341
  if len(data) != 3:
342
+ await cb.answer("❌ Format data tidak valid.", show_alert=True)
343
  return
344
  _, user_id_str, user_choice = data
345
  try:
346
  user_id = int(user_id_str)
347
  except ValueError:
348
+ await cb.answer("❌ Invalid user ID.", show_alert=True)
349
+ return
350
+ if cb.from_user.id != user_id:
351
+ await cb.answer("❌ You have no right to do this.", show_alert=True)
352
+ logger.warning(f"User {cb.from_user.id} mencoba memverifikasi CAPTCHA milik user {user_id}.")
353
  return
354
  if user_id not in captcha_texts:
355
+ await cb.answer("❗️ No active CAPTCHA verification.", show_alert=True)
356
+ logger.warning(f"User {user_id} mencoba memverifikasi CAPTCHA tanpa aktif.")
357
  return
358
+ captcha_text, correct_answer = captcha_texts.get(user_id)
359
+ failed_image = failed_hacker_by_randydev()
360
+ hacker_image = thanks_hacker_by_randydev()
361
+ if str(user_choice) == str(correct_answer):
362
  await cb.edit_message_media(
363
  media=InputMediaPhoto(
364
  hacker_image,
 
366
  ),
367
  reply_markup=create_button_join_group(captcha_texts.get("chat_link"))
368
  )
369
+ logger.info(f"User {user_id} berhasil memverifikasi CAPTCHA.")
370
  await client.approve_chat_join_request(
371
  chat_id=captcha_texts.get("chat_id"),
372
  user_id=user_id
 
375
  del captcha_texts["chat_id"]
376
  del captcha_texts["chat_link"]
377
  else:
378
+ await cb.edit_message_media(
379
+ media=InputMediaPhoto(
380
+ failed_image,
381
+ caption="❌ **CAPTCHA verification failed.**\n\nPlease try again."
382
+ )
383
+ )
384
  await client.decline_chat_join_request(
385
  chat_id=captcha_texts.get("chat_id"),
386
  user_id=user_id
387
  )
388
+ logger.info(f"User {user_id} gagal memverifikasi CAPTCHA.")
389
  del captcha_texts[user_id]
390
  del captcha_texts["chat_id"]
391
  del captcha_texts["chat_link"]
392
+ await cb.answer()
393
+
394
+ @Client.on_callback_query(filters.regex("^cancel_mode$"))
395
+ async def cancel_mode_callback(client: Client, cb: CallbackQuery):
396
+ await cb.edit_message_text("❌ CAPTCHA mode setting has been canceled.")
397
+ await cb.answer("Pembatalan berhasil.", show_alert=False)
398
+ logger.info(f"User {cb.from_user.id} canceled CAPTCHA mode setting.")
399
+
400
+ @Client.on_callback_query(filters.regex("^cancel_captcha$"))
401
+ async def cancel_captcha_callback(client: Client, cb: CallbackQuery):
402
+ user_id = cb.from_user.id
403
+ if user_id in captcha_texts:
404
+ del captcha_texts[user_id]
405
+ logger.info(f"User {user_id} has canceled CAPTCHA verification.")
406
+ await cb.edit_message_caption(
407
+ caption="❌ **CAPTCHA verification has been canceled.**\n\nIf you want to try again.",
408
+ )
409
+ await cb.answer("CAPTCHA verification canceled.", show_alert=False)
410
+ else:
411
+ await cb.answer("❗️ No active CAPTCHA verification found.", show_alert=True)
412
+ logger.warning(f"User {user_id} attempted to cancel CAPTCHA without active verification.")
413
 
414
  @Client.on_callback_query(filters.regex("^verify_captcha$"))
415
  async def verify_captcha_callback(client: Client, cb: CallbackQuery):