REFACTOR: Use connection pooling

Signed-off-by: baalajimaestro <me@baalajimaestro.me>
This commit is contained in:
baalajimaestro 2022-11-12 13:35:24 +05:30
parent f0954bdcbc
commit 20f397135b
Signed by: baalajimaestro
GPG key ID: F93C394FE9BBAFD5

View file

@ -12,14 +12,13 @@ import std/[asyncdispatch,
options, options,
strutils, strutils,
random, random,
with,
os, os,
tables, tables,
times, times,
sequtils, sequtils,
json] json]
import norm/[model, sqlite] import norm/[model, sqlite, pool]
# Logging Level # Logging Level
var L = newConsoleLogger(levelThreshold=lvlError, fmtStr="$levelname, [$time] ") var L = newConsoleLogger(levelThreshold=lvlError, fmtStr="$levelname, [$time] ")
@ -49,6 +48,7 @@ var GroupMedia = initTable[int, string]()
let AdminID = getEnv("ADMIN_ID").split(",") let AdminID = getEnv("ADMIN_ID").split(",")
let dbConn* = getDb() let dbConn* = getDb()
var connPool = newPool[DbConn](15)
# Functions to assist adding/deleting entries from tables # Functions to assist adding/deleting entries from tables
func NewCensoredData*(ftype = ""; fhash = ""; fileid = ""; time = 0.0; caption = ""): func NewCensoredData*(ftype = ""; fhash = ""; fileid = ""; time = 0.0; caption = ""):
@ -58,8 +58,9 @@ func NewBannedUsers*(userid = int64(0), bantime = 0.0, bantype = ""):
BannedUsers = BannedUsers(userid: userid, bantime: bantime, bantype: bantype) BannedUsers = BannedUsers(userid: userid, bantime: bantime, bantype: bantype)
# Create tables if they dont exist # Create tables if they dont exist
dbConn.createTables(NewCensoredData()) withDb(connPool):
dbConn.createTables(NewBannedUsers()) db.createTables(NewCensoredData())
db.createTables(NewBannedUsers())
# Ratelimits a user if there is more than 30 timestamps within 60 secs. # Ratelimits a user if there is more than 30 timestamps within 60 secs.
# Resets their ratelimit counter if 60s has passed since first timestamp # Resets their ratelimit counter if 60s has passed since first timestamp
@ -70,24 +71,26 @@ proc ManageRateLimit(): void=
elif len(RateLimiter[i]) >= 30: elif len(RateLimiter[i]) >= 30:
var BannedUser = NewBannedUsers(i, epochTime(), "auto") var BannedUser = NewBannedUsers(i, epochTime(), "auto")
RateLimiter.del(i) RateLimiter.del(i)
with dbConn: withDb(connPool):
insert BannedUser db.insert(BannedUser)
if dbConn.exists(BannedUsers, "bantype = ? and ? - bantime >= 1800", "auto", epochTime()): withDb(connPool):
var TempData = @[NewBannedUsers()] if db.exists(BannedUsers, "bantype = ? and ? - bantime >= 1800", "auto", epochTime()):
dbConn.select(TempData, "bantype = ? and ? - bantime >= 1800", "auto", epochTime()) var TempData = @[NewBannedUsers()]
for i in TempData: db.select(TempData, "bantype = ? and ? - bantime >= 1800", "auto", epochTime())
var e = i for i in TempData:
dbConn.delete(e) var e = i
db.delete(e)
# Cleanup old data to save space # Cleanup old data to save space
# Deletes old data after 6 months of entry # Deletes old data after 6 months of entry
proc OldDataCleanup(): void= proc OldDataCleanup(): void=
if dbConn.exists(CensoredData, "? - time >= 15780000", epochTime()): withDb(connPool):
var TempData = @[NewCensoredData()] if db.exists(CensoredData, "? - time >= 15780000", epochTime()):
dbConn.select(TempData, "? - time >= 15780000", epochTime()) var TempData = @[NewCensoredData()]
for i in TempData: db.select(TempData, "? - time >= 15780000", epochTime())
var e = i for i in TempData:
dbConn.delete(e) var e = i
db.delete(e)
# Generates a 6-character magic that will be used to identify the file # Generates a 6-character magic that will be used to identify the file
proc generate_hash(): string= proc generate_hash(): string=
@ -101,86 +104,89 @@ proc generate_hash(): string=
proc startHandler(b: Telebot, c: Command): Future[bool] {.gcsafe, async.} = proc startHandler(b: Telebot, c: Command): Future[bool] {.gcsafe, async.} =
let param = c.params let param = c.params
ManageRateLimit() ManageRateLimit()
if not dbConn.exists(BannedUsers, "userid = ?", c.message.chat.id): withDb(connPool):
# Update rate-limit counter if not db.exists(BannedUsers, "userid = ?", c.message.chat.id):
if not AdminID.contains($c.message.chat.id): # Update rate-limit counter
if RateLimiter.contains(c.message.chat.id): if not AdminID.contains($c.message.chat.id):
RateLimiter[c.message.chat.id].insert(epochTime()) if RateLimiter.contains(c.message.chat.id):
else: RateLimiter[c.message.chat.id].insert(epochTime())
RateLimiter[c.message.chat.id] = @[epochTime()]
if param == "":
discard await b.sendMessage(c.message.chat.id,
"Hey, To create a censored post, you can share any album, video, photo, gif, sticker, etc. " &
"The response of the bot can be used to share the media with other chats.")
else:
if not dbConn.exists(CensoredData, "fhash = ?", param):
discard await b.sendMessage(c.message.chat.id,
"Media does not exist on database, ask the sender to censor this again!")
else:
var TempData = @[NewCensoredData()]
dbConn.select(TempData, "fhash = ?", param)
if len(TempData) > 1:
var inputmedia = newSeq[InputMediaPhoto]()
for i in TempData:
if i.caption != "":
inputmedia.insert(InputMediaPhoto(kind: i.ftype, media: i.fileid, caption: some(i.caption)))
else:
inputmedia.insert(InputMediaPhoto(kind: i.ftype, media: i.fileid))
discard await b.sendMediaGroup($c.message.chat.id, media=inputmedia)
else: else:
if TempData[0].ftype == "photo": RateLimiter[c.message.chat.id] = @[epochTime()]
discard await b.sendPhoto(chatId=c.message.chat.id, photo=TempData[0].fileid, caption=TempData[0].caption) if param == "":
elif TempData[0].ftype == "document": discard await b.sendMessage(c.message.chat.id,
discard await b.sendDocument(c.message.chat.id, TempData[0].fileid, caption=TempData[0].caption) "Hey, To create a censored post, you can share any album, video, photo, gif, sticker, etc. " &
elif TempData[0].ftype == "video": "The response of the bot can be used to share the media with other chats.")
discard await b.sendVideo(c.message.chat.id, TempData[0].fileid, caption=TempData[0].caption) else:
elif TempData[0].ftype == "videoNote": if not db.exists(CensoredData, "fhash = ?", param):
discard await b.sendVideoNote(c.message.chat.id, TempData[0].fileid) discard await b.sendMessage(c.message.chat.id,
elif TempData[0].ftype == "animation": "Media does not exist on database, ask the sender to censor this again!")
discard await b.sendAnimation(c.message.chat.id, TempData[0].fileid, caption=TempData[0].caption) else:
elif TempData[0].ftype == "sticker": var TempData = @[NewCensoredData()]
discard await b.sendSticker(c.message.chat.id, TempData[0].fileid) db.select(TempData, "fhash = ?", param)
if len(TempData) > 1:
var inputmedia = newSeq[InputMediaPhoto]()
for i in TempData:
if i.caption != "":
inputmedia.insert(InputMediaPhoto(kind: i.ftype, media: i.fileid, caption: some(i.caption)))
else:
inputmedia.insert(InputMediaPhoto(kind: i.ftype, media: i.fileid))
discard await b.sendMediaGroup($c.message.chat.id, media=inputmedia)
else:
if TempData[0].ftype == "photo":
discard await b.sendPhoto(chatId=c.message.chat.id, photo=TempData[0].fileid, caption=TempData[0].caption)
elif TempData[0].ftype == "document":
discard await b.sendDocument(c.message.chat.id, TempData[0].fileid, caption=TempData[0].caption)
elif TempData[0].ftype == "video":
discard await b.sendVideo(c.message.chat.id, TempData[0].fileid, caption=TempData[0].caption)
elif TempData[0].ftype == "videoNote":
discard await b.sendVideoNote(c.message.chat.id, TempData[0].fileid)
elif TempData[0].ftype == "animation":
discard await b.sendAnimation(c.message.chat.id, TempData[0].fileid, caption=TempData[0].caption)
elif TempData[0].ftype == "sticker":
discard await b.sendSticker(c.message.chat.id, TempData[0].fileid)
# Give them source url # Give them source url
proc sourceHandler(b: Telebot, c: Command): Future[bool] {.gcsafe, async.} = proc sourceHandler(b: Telebot, c: Command): Future[bool] {.gcsafe, async.} =
ManageRateLimit() ManageRateLimit()
if not dbConn.exists(BannedUsers, "userid = ?", c.message.chat.id): withDb(connPool):
# Update rate-limit counter if not db.exists(BannedUsers, "userid = ?", c.message.chat.id):
if not AdminID.contains($c.message.chat.id): # Update rate-limit counter
if RateLimiter.contains(c.message.chat.id): if not AdminID.contains($c.message.chat.id):
RateLimiter[c.message.chat.id].insert(epochTime()) if RateLimiter.contains(c.message.chat.id):
else: RateLimiter[c.message.chat.id].insert(epochTime())
RateLimiter[c.message.chat.id] = @[epochTime()] else:
discard await b.sendMessage(c.message.chat.id, RateLimiter[c.message.chat.id] = @[epochTime()]
"Hey, this bot is open-source and licensed under AGPL-v3! " & discard await b.sendMessage(c.message.chat.id,
"\n\nYou are welcome to selfhost your own instance of this bot" & "Hey, this bot is open-source and licensed under AGPL-v3! " &
"\n\nThe source code is [here](https://git.baalajimaestro.me/baalajimaestro/nim-censor-bot)", "\n\nYou are welcome to selfhost your own instance of this bot" &
parseMode="Markdown", "\n\nThe source code is [here](https://git.baalajimaestro.me/baalajimaestro/nim-censor-bot)",
disableWebPagePreview = true) parseMode="Markdown",
disableWebPagePreview = true)
# UnBan Handler # UnBan Handler
proc unbanHandler(b: Telebot, c: Command): Future[bool] {.gcsafe, async.} = proc unbanHandler(b: Telebot, c: Command): Future[bool] {.gcsafe, async.} =
if $c.message.chat.id in AdminID: if $c.message.chat.id in AdminID:
let user = c.params let user = c.params
if dbConn.exists(BannedUsers, "userid = ?", int64(parseInt(user))): withDb(connPool):
var TempData = @[NewBannedUsers()] if db.exists(BannedUsers, "userid = ?", int64(parseInt(user))):
dbConn.select(TempData, "userid = ?", int64(parseInt(user))) var TempData = @[NewBannedUsers()]
for i in TempData: db.select(TempData, "userid = ?", int64(parseInt(user)))
var e = i for i in TempData:
dbConn.delete(e) var e = i
discard await b.sendMessage(c.message.chat.id, "Unbanned!") db.delete(e)
discard await b.sendMessage(c.message.chat.id, "Unbanned!")
# ban Handler # ban Handler
proc banHandler(b: Telebot, c: Command): Future[bool] {.gcsafe, async.} = proc banHandler(b: Telebot, c: Command): Future[bool] {.gcsafe, async.} =
if $c.message.chat.id in AdminID: if $c.message.chat.id in AdminID:
let user = c.params let user = c.params
var BannedUser = NewBannedUsers(int64(parseInt(user)), epochTime(), "permanent") var BannedUser = NewBannedUsers(int64(parseInt(user)), epochTime(), "permanent")
with dbConn: withDb(connPool):
insert BannedUser db.insert(BannedUser)
discard await b.sendMessage(c.message.chat.id, "Banned!") discard await b.sendMessage(c.message.chat.id, "Banned!")
# Inline share handler # Inline share handler
proc inlineHandler(b: Telebot, u: InlineQuery): Future[bool]{.async.} = proc inlineHandler(b: Telebot, u: InlineQuery): Future[bool]{.gcsafe, async.} =
var TempData = @[NewCensoredData()] var TempData = @[NewCensoredData()]
var ftype = "" var ftype = ""
var res: InlineQueryResultArticle var res: InlineQueryResultArticle
@ -188,25 +194,26 @@ proc inlineHandler(b: Telebot, u: InlineQuery): Future[bool]{.async.} =
res.kind = "article" res.kind = "article"
res.id = "1" res.id = "1"
if u.query != "": if u.query != "":
if not dbConn.exists(CensoredData, "fhash = ?",u.query): withDb(connPool):
res.title = "Media Not Found" if not db.exists(CensoredData, "fhash = ?",u.query):
res.inputMessageContent = InputTextMessageContent( res.title = "Media Not Found"
"Media does not exist on database, ask the sender to censor this again!").some res.inputMessageContent = InputTextMessageContent(
else: "Media does not exist on database, ask the sender to censor this again!").some
dbConn.select(TempData, "fhash = ?", u.query) else:
if len(TempData) > 1: db.select(TempData, "fhash = ?", u.query)
ftype = "Album" if len(TempData) > 1:
elif len(TempData) == 1: ftype = "Album"
ftype = TempData[0].ftype elif len(TempData) == 1:
res.title = "NSFW " & capitalizeAscii(ftype) ftype = TempData[0].ftype
res.inputMessageContent = InputMessageContent(kind: TextMessage, res.title = "NSFW " & capitalizeAscii(ftype)
messageText:"*NSFW " & res.inputMessageContent = InputMessageContent(kind: TextMessage,
capitalizeAscii(ftype) & messageText:"*NSFW " &
"*\n\n[Tap to View](tg://resolve?domain=" & capitalizeAscii(ftype) &
b.username & "&start=" & "*\n\n[Tap to View](tg://resolve?domain=" &
u.query & b.username & "&start=" &
")", u.query &
parseMode: some("Markdown")).some ")",
parseMode: some("Markdown")).some
else: else:
res.title = "Waiting for File Hash" res.title = "Waiting for File Hash"
res.inputMessageContent = InputTextMessageContent( res.inputMessageContent = InputTextMessageContent(
@ -220,82 +227,83 @@ proc updateHandler(b: Telebot, u: Update): Future[bool] {.async, gcsafe.} =
let response = u.message.get let response = u.message.get
# Refresh rate-limits # Refresh rate-limits
ManageRateLimit() ManageRateLimit()
# Dont bother about rate-limited/banned users withDb(connPool):
if not dbConn.exists(BannedUsers, "userid = ?", response.chat.id): # Dont bother about rate-limited/banned users
# Update rate-limit counter if not db.exists(BannedUsers, "userid = ?", response.chat.id):
if not AdminID.contains($response.chat.id): # Update rate-limit counter
if RateLimiter.contains(response.chat.id): if not AdminID.contains($response.chat.id):
RateLimiter[response.chat.id].insert(epochTime()) if RateLimiter.contains(response.chat.id):
else: RateLimiter[response.chat.id].insert(epochTime())
RateLimiter[response.chat.id] = @[epochTime()] else:
if not response.text.isSome: RateLimiter[response.chat.id] = @[epochTime()]
var fileid = "" if not response.text.isSome:
var ftype = "" var fileid = ""
var fcaption = "" var ftype = ""
if response.caption.isSome: var fcaption = ""
fcaption = response.caption.get if response.caption.isSome:
if response.document.isSome: fcaption = response.caption.get
fileid = response.document.get.fileId if response.document.isSome:
ftype = "document" fileid = response.document.get.fileId
elif response.video.isSome: ftype = "document"
fileid = response.video.get.fileId elif response.video.isSome:
ftype = "video" fileid = response.video.get.fileId
elif response.videoNote.isSome: ftype = "video"
fileid = response.videoNote.get.fileId elif response.videoNote.isSome:
ftype = "videoNote" fileid = response.videoNote.get.fileId
elif response.animation.isSome: ftype = "videoNote"
fileid = response.animation.get.fileId elif response.animation.isSome:
ftype = "animation" fileid = response.animation.get.fileId
elif response.photo.isSome: ftype = "animation"
fileid = response.photo.get[0].fileId elif response.photo.isSome:
ftype = "photo" fileid = response.photo.get[0].fileId
elif response.sticker.isSome: ftype = "photo"
fileid = response.sticker.get.fileId elif response.sticker.isSome:
ftype = "sticker" fileid = response.sticker.get.fileId
# Workaround for groupmedia using hashtables ftype = "sticker"
# Telegram is sending multiple updates, instead of 1, so just workaround it # Workaround for groupmedia using hashtables
if response.mediaGroupId.isSome: # Telegram is sending multiple updates, instead of 1, so just workaround it
if parseInt(response.mediaGroupId.get) notin GroupMedia.keys().toSeq(): if response.mediaGroupId.isSome:
if parseInt(response.mediaGroupId.get) notin GroupMedia.keys().toSeq():
let filehash = generate_hash()
GroupMedia[parseInt(response.mediaGroupId.get)] = filehash
var CensoredRow = NewCensoredData(ftype, filehash, fileid, epochTime(), fcaption)
withDb(connPool):
db.insert(CensoredRow)
var replybutton = InlineKeyboardButton(text: "Share", switchInlineQuery: some(filehash))
let replymark = newInlineKeyboardMarkup(@[replybutton])
discard await b.sendMessage(response.chat.id,
"*NSFW " &
"Album" &
"*\n\n[Tap to View](tg://resolve?domain=" &
b.username &
"&start=" &
filehash &
")",
replyMarkup = replymark,
parseMode="Markdown")
else:
let filehash = GroupMedia[parseInt(response.mediaGroupId.get)]
var CensoredRow = NewCensoredData(ftype, filehash, fileid, epochTime(), fcaption)
withDb(connPool):
db.insert(CensoredRow)
else:
let filehash = generate_hash() let filehash = generate_hash()
GroupMedia[parseInt(response.mediaGroupId.get)] = filehash
var CensoredRow = NewCensoredData(ftype, filehash, fileid, epochTime(), fcaption) var CensoredRow = NewCensoredData(ftype, filehash, fileid, epochTime(), fcaption)
with dbConn: withDb(connPool):
insert CensoredRow db.insert(CensoredRow)
var replybutton = InlineKeyboardButton(text: "Share", switchInlineQuery: some(filehash)) var replybutton = InlineKeyboardButton(text: "Share", switchInlineQuery: some(filehash))
let replymark = newInlineKeyboardMarkup(@[replybutton]) let replymark = newInlineKeyboardMarkup(@[replybutton])
discard await b.sendMessage(response.chat.id, discard await b.sendMessage(response.chat.id,
"*NSFW " & "*NSFW " &
"Album" & capitalizeAscii(ftype) &
"*\n\n[Tap to View](tg://resolve?domain=" & "*\n\n[Tap to View](tg://resolve?domain=" &
b.username & b.username &
"&start=" & "&start=" &
filehash & filehash &
")", ")",
replyMarkup = replymark, replyMarkup = replymark,
parseMode="Markdown") parseMode="Markdown")
else:
let filehash = GroupMedia[parseInt(response.mediaGroupId.get)]
var CensoredRow = NewCensoredData(ftype, filehash, fileid, epochTime(), fcaption)
with dbConn:
insert CensoredRow
else:
let filehash = generate_hash()
var CensoredRow = NewCensoredData(ftype, filehash, fileid, epochTime(), fcaption)
with dbConn:
insert CensoredRow
var replybutton = InlineKeyboardButton(text: "Share", switchInlineQuery: some(filehash))
let replymark = newInlineKeyboardMarkup(@[replybutton])
discard await b.sendMessage(response.chat.id,
"*NSFW " &
capitalizeAscii(ftype) &
"*\n\n[Tap to View](tg://resolve?domain=" &
b.username &
"&start=" &
filehash &
")",
replyMarkup = replymark,
parseMode="Markdown")
OldDataCleanup() OldDataCleanup()
when isMainModule: when isMainModule: