REFACTOR: Use connection pooling
Signed-off-by: baalajimaestro <me@baalajimaestro.me>
This commit is contained in:
parent
f0954bdcbc
commit
20f397135b
1 changed files with 170 additions and 162 deletions
|
@ -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:
|
||||||
|
|
Loading…
Reference in a new issue