mirror of
				https://github.com/InsanusMokrassar/BooruGrabberTelegramBot.git
				synced 2025-10-26 09:40:15 +00:00 
			
		
		
		
	almost completed grabber
This commit is contained in:
		| @@ -1,10 +1,11 @@ | |||||||
| import dev.inmo.krontab.utils.asFlow | import dev.inmo.krontab.utils.asFlow | ||||||
| import dev.inmo.micro_utils.coroutines.runCatchingSafely | import dev.inmo.micro_utils.coroutines.* | ||||||
| import dev.inmo.micro_utils.coroutines.subscribeSafelyWithoutExceptions |  | ||||||
| import dev.inmo.micro_utils.pagination.utils.doForAllWithNextPaging | import dev.inmo.micro_utils.pagination.utils.doForAllWithNextPaging | ||||||
|  | import dev.inmo.micro_utils.repos.add | ||||||
| import dev.inmo.micro_utils.repos.cache.cache.FullKVCache | import dev.inmo.micro_utils.repos.cache.cache.FullKVCache | ||||||
| import dev.inmo.micro_utils.repos.cache.cached | import dev.inmo.micro_utils.repos.cache.cached | ||||||
| import dev.inmo.micro_utils.repos.exposed.keyvalue.ExposedKeyValueRepo | import dev.inmo.micro_utils.repos.exposed.keyvalue.ExposedKeyValueRepo | ||||||
|  | import dev.inmo.micro_utils.repos.exposed.onetomany.ExposedKeyValuesRepo | ||||||
| import dev.inmo.micro_utils.repos.mappers.withMapper | import dev.inmo.micro_utils.repos.mappers.withMapper | ||||||
| import dev.inmo.micro_utils.repos.unset | import dev.inmo.micro_utils.repos.unset | ||||||
| import dev.inmo.tgbotapi.bot.ktor.telegramBot | import dev.inmo.tgbotapi.bot.ktor.telegramBot | ||||||
| @@ -15,18 +16,20 @@ import dev.inmo.tgbotapi.extensions.api.send.media.* | |||||||
| import dev.inmo.tgbotapi.extensions.api.send.reply | import dev.inmo.tgbotapi.extensions.api.send.reply | ||||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.buildBehaviourWithLongPolling | import dev.inmo.tgbotapi.extensions.behaviour_builder.buildBehaviourWithLongPolling | ||||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling.onCommand | import dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling.onCommand | ||||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling.onCommandWithArgs |  | ||||||
| import dev.inmo.tgbotapi.requests.abstracts.FileUrl | import dev.inmo.tgbotapi.requests.abstracts.FileUrl | ||||||
| import dev.inmo.tgbotapi.types.* | import dev.inmo.tgbotapi.types.* | ||||||
| import dev.inmo.tgbotapi.types.chat.ChannelChat | import dev.inmo.tgbotapi.types.chat.ChannelChat | ||||||
| import dev.inmo.tgbotapi.types.chat.PrivateChat | import dev.inmo.tgbotapi.types.chat.PrivateChat | ||||||
| import dev.inmo.tgbotapi.types.media.TelegramMediaPhoto | import dev.inmo.tgbotapi.types.media.TelegramMediaPhoto | ||||||
|  | import dev.inmo.tgbotapi.types.message.content.PhotoContent | ||||||
| import java.io.File | import java.io.File | ||||||
| import kotlinx.coroutines.* | import kotlinx.coroutines.* | ||||||
| import kotlinx.coroutines.sync.Mutex | import kotlinx.coroutines.sync.Mutex | ||||||
| import kotlinx.coroutines.sync.withLock | import kotlinx.coroutines.sync.withLock | ||||||
| import kotlinx.serialization.json.Json | import kotlinx.serialization.json.Json | ||||||
| import models.Config | import models.Config | ||||||
|  | import net.kodehawa.lib.imageboards.ImageBoard | ||||||
|  | import net.kodehawa.lib.imageboards.entities.BoardImage | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * This method by default expects one argument in [args] field: telegram bot configuration |  * This method by default expects one argument in [args] field: telegram bot configuration | ||||||
| @@ -39,8 +42,10 @@ suspend fun main(args: Array<String>) { | |||||||
|     // that is your bot |     // that is your bot | ||||||
|     val bot = telegramBot(config.token) |     val bot = telegramBot(config.token) | ||||||
|  |  | ||||||
|  |     ImageBoard.setUserAgent("WhoAmI?") | ||||||
|  |  | ||||||
|     // that is kotlin coroutine scope which will be used in requests and parallel works under the hood |     // that is kotlin coroutine scope which will be used in requests and parallel works under the hood | ||||||
|     val scope = CoroutineScope(Dispatchers.Default) |     val scope = CoroutineScope(Dispatchers.Default + ContextSafelyExceptionHandler { it.printStackTrace() }) | ||||||
|  |  | ||||||
|     val repo = ExposedKeyValueRepo( |     val repo = ExposedKeyValueRepo( | ||||||
|         config.database.database, |         config.database.database, | ||||||
| @@ -54,6 +59,18 @@ suspend fun main(args: Array<String>) { | |||||||
|         { json.decodeFromString(ChatSettings.serializer(), this) }, |         { json.decodeFromString(ChatSettings.serializer(), this) }, | ||||||
|     ).cached(FullKVCache(), scope = scope) |     ).cached(FullKVCache(), scope = scope) | ||||||
|  |  | ||||||
|  |     val chatsUrlsSeen = ExposedKeyValuesRepo( | ||||||
|  |         config.database.database, | ||||||
|  |         { long("chat_id") }, | ||||||
|  |         { text("url") }, | ||||||
|  |         "chatsUrlsSeen" | ||||||
|  |     ).withMapper( | ||||||
|  |         { chatId }, | ||||||
|  |         { this }, | ||||||
|  |         { ChatId(this) }, | ||||||
|  |         { this }, | ||||||
|  |     ).cached(FullKVCache(), scope = scope) | ||||||
|  |  | ||||||
|     val chatsChangingMutex = Mutex() |     val chatsChangingMutex = Mutex() | ||||||
|     val chatsSendingJobs = mutableMapOf<ChatId, Job>() |     val chatsSendingJobs = mutableMapOf<ChatId, Job>() | ||||||
|  |  | ||||||
| @@ -68,26 +85,44 @@ suspend fun main(args: Array<String>) { | |||||||
|                 chatsSendingJobs[chatId] ?.cancel() |                 chatsSendingJobs[chatId] ?.cancel() | ||||||
|                 settings ?.let { |                 settings ?.let { | ||||||
|                     chatsSendingJobs[chatId] = settings.scheduler.asFlow().subscribeSafelyWithoutExceptions(scope) { |                     chatsSendingJobs[chatId] = settings.scheduler.asFlow().subscribeSafelyWithoutExceptions(scope) { | ||||||
|                         val result = settings.makeRequest() |                         val result = mutableListOf<BoardImage>() | ||||||
|  |                         let { | ||||||
|  |                             var i = 0 | ||||||
|  |                             while (result.size < settings.count) { | ||||||
|  |                                 val images = settings.makeRequest(i).takeIf { it.isNotEmpty() } ?: break | ||||||
|  |                                 result.addAll( | ||||||
|  |                                     images.filterNot { | ||||||
|  |                                         chatsUrlsSeen.contains(chatId, it.url) | ||||||
|  |                                     } | ||||||
|  |                                 ) | ||||||
|  |                                 i++ | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|                         when { |                         when { | ||||||
|                             result.isEmpty() -> return@subscribeSafelyWithoutExceptions |                             result.isEmpty() -> return@subscribeSafelyWithoutExceptions | ||||||
|                             result.size == 1 -> sendPhoto( |                             result.size == 1 -> sendPhoto( | ||||||
|                                 chatId, |                                 chatId, | ||||||
|                                 FileUrl(result.first().url) |                                 FileUrl(result.first().url) | ||||||
|                             ) |                             ).also { | ||||||
|  |                                 result.forEach { chatsUrlsSeen.add(chatId, it.url) } | ||||||
|  |                             } | ||||||
|                             settings.gallery -> result.chunked(mediaCountInMediaGroup.last + 1).forEach { |                             settings.gallery -> result.chunked(mediaCountInMediaGroup.last + 1).forEach { | ||||||
|                                 sendVisualMediaGroup( |                                 sendVisualMediaGroup( | ||||||
|                                     chatId, |                                     chatId, | ||||||
|                                     it.map { |                                     it.map { | ||||||
|                                         TelegramMediaPhoto(FileUrl(it.url)) |                                         TelegramMediaPhoto(FileUrl(it.url)) | ||||||
|                                     } |                                     } | ||||||
|                                 ) |                                 ).also { _ -> | ||||||
|  |                                     it.forEach { chatsUrlsSeen.add(chatId, it.url) } | ||||||
|  |                                 } | ||||||
|                             } |                             } | ||||||
|                             else -> result.forEach { |                             else -> result.forEach { | ||||||
|                                 sendPhoto( |                                 sendPhoto( | ||||||
|                                     chatId, |                                     chatId, | ||||||
|                                     FileUrl(it.url) |                                     FileUrl(it.url) | ||||||
|                                 ) |                                 ).also { _ -> | ||||||
|  |                                     chatsUrlsSeen.add(chatId, it.url) | ||||||
|  |                                 } | ||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
| @@ -98,10 +133,12 @@ suspend fun main(args: Array<String>) { | |||||||
|         doForAllWithNextPaging { |         doForAllWithNextPaging { | ||||||
|             repo.keys(it).also { |             repo.keys(it).also { | ||||||
|                 it.results.forEach { |                 it.results.forEach { | ||||||
|  |                     runCatchingSafely { | ||||||
|                         refreshChatJob(it, null) |                         refreshChatJob(it, null) | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|         repo.onNewValue.subscribeSafelyWithoutExceptions(this) { |         repo.onNewValue.subscribeSafelyWithoutExceptions(this) { | ||||||
|             refreshChatJob(it.first, it.second) |             refreshChatJob(it.first, it.second) | ||||||
| @@ -113,19 +150,20 @@ suspend fun main(args: Array<String>) { | |||||||
|         onCommand(Regex("(help|start)"), requireOnlyCommandInMessage = true) { |         onCommand(Regex("(help|start)"), requireOnlyCommandInMessage = true) { | ||||||
|             reply(it, EnableArgsParser(it.chat.id, repo, scope).getFormattedHelp().takeIf { it.isNotBlank() } ?: return@onCommand) |             reply(it, EnableArgsParser(it.chat.id, repo, scope).getFormattedHelp().takeIf { it.isNotBlank() } ?: return@onCommand) | ||||||
|         } |         } | ||||||
|         onCommandWithArgs("enable") { message, strings -> |         onCommand("enable", requireOnlyCommandInMessage = false) { | ||||||
|             val parser = EnableArgsParser(message.chat.id, repo, this) |             val args = it.content.textSources.drop(1).joinToString("") { it.source }.split(" ") | ||||||
|  |             val parser = EnableArgsParser(it.chat.id, repo, this) | ||||||
|             runCatchingSafely { |             runCatchingSafely { | ||||||
|                 parser.parse(strings) |                 parser.parse(args) | ||||||
|             }.onFailure { e -> |             }.onFailure { e -> | ||||||
|                 e.printStackTrace() |                 e.printStackTrace() | ||||||
|                 if (message.chat is PrivateChat) { |                 if (it.chat is PrivateChat) { | ||||||
|                     reply(message, parser.getFormattedHelp()) |                     reply(it, parser.getFormattedHelp()) | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             runCatchingSafely { |             runCatchingSafely { | ||||||
|                 if (message.chat is ChannelChat) { |                 if (it.chat is ChannelChat) { | ||||||
|                     delete(message) |                     delete(it) | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -36,9 +36,9 @@ data class ChatSettings( | |||||||
|             DefaultBoards.E926 -> DefaultImageBoards.E926 |             DefaultBoards.E926 -> DefaultImageBoards.E926 | ||||||
|         } |         } | ||||||
|  |  | ||||||
|     suspend fun makeRequest(): List<BoardImage> { |     suspend fun makeRequest(page: Int): List<BoardImage> { | ||||||
|         return withContext(Dispatchers.IO) { |         return withContext(Dispatchers.IO) { | ||||||
|             board.search(count, query).blocking() |             board.search(page, count, query).blocking() | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -17,7 +17,9 @@ class EnableArgsParser( | |||||||
|         it in 1 .. 10 |         it in 1 .. 10 | ||||||
|     } |     } | ||||||
|     val query by argument().multiple(required = true).help("Your query to booru. Use syntax \"-- -sometag\" to add excluding of some tag in query") |     val query by argument().multiple(required = true).help("Your query to booru. Use syntax \"-- -sometag\" to add excluding of some tag in query") | ||||||
|     val krontab by option("-k", "--krontab").required().help("Krontab in format * * * * *. See https://bookstack.inmo.dev/books/krontab/page/string-format") |     val krontab by option("-k", "--krontab").transformValues(5) { | ||||||
|  |         it.joinToString(" ") | ||||||
|  |     }.required().help("Krontab in format * * * * *. See https://bookstack.inmo.dev/books/krontab/page/string-format") | ||||||
|     val board by option("-b", "--board").convert { |     val board by option("-b", "--board").convert { | ||||||
|         ChatSettings.BoardSerializer.types.getValue(it) |         ChatSettings.BoardSerializer.types.getValue(it) | ||||||
|     }.required().help("Board type. Possible values: ${ChatSettings.BoardSerializer.types.keys.joinToString { it }}") |     }.required().help("Board type. Possible values: ${ChatSettings.BoardSerializer.types.keys.joinToString { it }}") | ||||||
| @@ -25,7 +27,7 @@ class EnableArgsParser( | |||||||
|  |  | ||||||
|     override fun run() { |     override fun run() { | ||||||
|         val chatSettings = ChatSettings( |         val chatSettings = ChatSettings( | ||||||
|             query.joinToString(" "), |             query.filterNot { it.isEmpty() }.joinToString(" ").trim(), | ||||||
|             krontab, |             krontab, | ||||||
|             board, |             board, | ||||||
|             count, |             count, | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user