mirror of
				https://github.com/InsanusMokrassar/TelegramBotAPI.git
				synced 2025-10-25 01:00:13 +00:00 
			
		
		
		
	fix of #560
This commit is contained in:
		| @@ -4,6 +4,7 @@ package dev.inmo.tgbotapi.extensions.behaviour_builder | ||||
|  | ||||
| import dev.inmo.micro_utils.coroutines.* | ||||
| import dev.inmo.tgbotapi.bot.TelegramBot | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.utils.handlers_registrar.TriggersHolder | ||||
| import dev.inmo.tgbotapi.types.update.abstracts.Update | ||||
| import dev.inmo.tgbotapi.updateshandlers.* | ||||
| import dev.inmo.tgbotapi.utils.RiskFeature | ||||
| @@ -47,12 +48,15 @@ interface BehaviourContext : FlowsUpdatesFilter, TelegramBot, CoroutineScope { | ||||
|     val flowsUpdatesFilter: FlowsUpdatesFilter | ||||
|         get() = this | ||||
|  | ||||
|     val triggersHolder: TriggersHolder | ||||
|  | ||||
|     fun copy( | ||||
|         bot: TelegramBot = this.bot, | ||||
|         scope: CoroutineScope = this.scope, | ||||
|         broadcastChannelsSize: Int = 100, | ||||
|         onBufferOverflow: BufferOverflow = BufferOverflow.SUSPEND, | ||||
|         upstreamUpdatesFlow: Flow<Update>? = null, | ||||
|         triggersHolder: TriggersHolder = TriggersHolder(), | ||||
|         updatesFilter: BehaviourContextAndTypeReceiver<Boolean, Update>? = null | ||||
|     ): BehaviourContext | ||||
| } | ||||
| @@ -63,6 +67,7 @@ class DefaultBehaviourContext( | ||||
|     broadcastChannelsSize: Int = 100, | ||||
|     onBufferOverflow: BufferOverflow = BufferOverflow.SUSPEND, | ||||
|     private val upstreamUpdatesFlow: Flow<Update>? = null, | ||||
|     override val triggersHolder: TriggersHolder = TriggersHolder(), | ||||
|     private val updatesFilter: BehaviourContextAndTypeReceiver<Boolean, Update>? = null | ||||
| ) : AbstractFlowsUpdatesFilter(), TelegramBot by bot, CoroutineScope by scope, BehaviourContext { | ||||
|  | ||||
| @@ -89,22 +94,25 @@ class DefaultBehaviourContext( | ||||
|         broadcastChannelsSize: Int, | ||||
|         onBufferOverflow: BufferOverflow, | ||||
|         upstreamUpdatesFlow: Flow<Update>?, | ||||
|         triggersHolder: TriggersHolder, | ||||
|         updatesFilter: BehaviourContextAndTypeReceiver<Boolean, Update>? | ||||
|     ): DefaultBehaviourContext = DefaultBehaviourContext(bot, scope, broadcastChannelsSize, onBufferOverflow, upstreamUpdatesFlow, updatesFilter) | ||||
|     ): DefaultBehaviourContext = DefaultBehaviourContext(bot, scope, broadcastChannelsSize, onBufferOverflow, upstreamUpdatesFlow, triggersHolder, updatesFilter) | ||||
| } | ||||
|  | ||||
| fun BehaviourContext( | ||||
|     bot: TelegramBot, | ||||
|     scope: CoroutineScope, | ||||
|     flowsUpdatesFilter: FlowsUpdatesFilter = FlowsUpdatesFilter() | ||||
| ) = DefaultBehaviourContext(bot, scope, upstreamUpdatesFlow = flowsUpdatesFilter.allUpdatesFlow) | ||||
|     flowsUpdatesFilter: FlowsUpdatesFilter = FlowsUpdatesFilter(), | ||||
|     triggersHolder: TriggersHolder = TriggersHolder(), | ||||
| ) = DefaultBehaviourContext(bot, scope, upstreamUpdatesFlow = flowsUpdatesFilter.allUpdatesFlow, triggersHolder = triggersHolder) | ||||
|  | ||||
| inline fun <T> BehaviourContext( | ||||
|     bot: TelegramBot, | ||||
|     scope: CoroutineScope, | ||||
|     flowsUpdatesFilter: FlowsUpdatesFilter = FlowsUpdatesFilter(), | ||||
|     triggersHolder: TriggersHolder = TriggersHolder(), | ||||
|     crossinline block: BehaviourContext.() -> T | ||||
| ) = DefaultBehaviourContext(bot, scope, upstreamUpdatesFlow = flowsUpdatesFilter.allUpdatesFlow).run(block) | ||||
| ) = DefaultBehaviourContext(bot, scope, upstreamUpdatesFlow = flowsUpdatesFilter.allUpdatesFlow, triggersHolder = triggersHolder).run(block) | ||||
|  | ||||
| /** | ||||
|  * Creates new one [BehaviourContext], adding subsequent [FlowsUpdatesFilter] in case [updatesFilter] is provided and | ||||
| @@ -115,6 +123,7 @@ suspend fun <T, BC : BehaviourContext> BC.doInSubContextWithUpdatesFilter( | ||||
|     stopOnCompletion: Boolean = true, | ||||
|     updatesUpstreamFlow: Flow<Update> = allUpdatesFlow, | ||||
|     scope: CoroutineScope = LinkedSupervisorScope(), | ||||
|     triggersHolder: TriggersHolder = this.triggersHolder, | ||||
|     behaviourContextReceiver: CustomBehaviourContextReceiver<BC, T> | ||||
| ): T { | ||||
|     val newContext = copy( | ||||
| @@ -126,7 +135,8 @@ suspend fun <T, BC : BehaviourContext> BC.doInSubContextWithUpdatesFilter( | ||||
|                 } ?: true | ||||
|             } | ||||
|         }, | ||||
|         upstreamUpdatesFlow = updatesUpstreamFlow | ||||
|         upstreamUpdatesFlow = updatesUpstreamFlow, | ||||
|         triggersHolder = triggersHolder | ||||
|     ) as BC | ||||
|     return withContext(currentCoroutineContext()) { | ||||
|         newContext.behaviourContextReceiver().also { if (stopOnCompletion) newContext.stop() } | ||||
| @@ -137,8 +147,9 @@ suspend fun <T> BehaviourContext.doInSubContext( | ||||
|     stopOnCompletion: Boolean = true, | ||||
|     updatesUpstreamFlow: Flow<Update> = allUpdatesFlow, | ||||
|     scope: CoroutineScope = LinkedSupervisorScope(), | ||||
|     triggersHolder: TriggersHolder = this.triggersHolder, | ||||
|     behaviourContextReceiver: BehaviourContextReceiver<T> | ||||
| ) = doInSubContextWithUpdatesFilter(updatesFilter = null, stopOnCompletion, updatesUpstreamFlow, scope, behaviourContextReceiver) | ||||
| ) = doInSubContextWithUpdatesFilter(updatesFilter = null, stopOnCompletion, updatesUpstreamFlow, scope, triggersHolder, behaviourContextReceiver) | ||||
|  | ||||
| /** | ||||
|  * This method will cancel ALL subsequent contexts, expectations and waiters | ||||
|   | ||||
| @@ -2,15 +2,17 @@ | ||||
|  | ||||
| package dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling | ||||
|  | ||||
| import dev.inmo.micro_utils.coroutines.launchSafelyWithoutExceptions | ||||
| import dev.inmo.micro_utils.coroutines.runCatchingSafely | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.* | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.filters.CallbackQueryFilterByUser | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.utils.SimpleFilter | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.utils.* | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.utils.marker_factories.ByUserCallbackQueryMarkerFactory | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.utils.marker_factories.MarkerFactory | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.utils.plus | ||||
| import dev.inmo.tgbotapi.extensions.utils.asCallbackQueryUpdate | ||||
| import dev.inmo.tgbotapi.types.CallbackQuery.* | ||||
| import dev.inmo.tgbotapi.types.update.abstracts.Update | ||||
| import kotlinx.coroutines.Job | ||||
|  | ||||
| internal suspend inline fun <BC : BehaviourContext, reified T : CallbackQuery> BC.onCallbackQuery( | ||||
|     noinline initialFilter: SimpleFilter<T>? = null, | ||||
| @@ -21,6 +23,48 @@ internal suspend inline fun <BC : BehaviourContext, reified T : CallbackQuery> B | ||||
|     (it.asCallbackQueryUpdate() ?.data as? T) ?.let(::listOfNotNull) | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @param initialFilter This filter will be called to remove unnecessary data BEFORE [scenarioReceiver] call | ||||
|  * @param subcontextUpdatesFilter This filter will be applied to each update inside of [scenarioReceiver]. For example, | ||||
|  * this filter will be used if you will call [dev.inmo.tgbotapi.extensions.behaviour_builder.expectations.waitContentMessage]. | ||||
|  * Use [dev.inmo.tgbotapi.extensions.behaviour_builder.BehaviourContextAndTwoTypesReceiver] function to create your own. | ||||
|  * Use [dev.inmo.tgbotapi.extensions.behaviour_builder.utils.plus] or [dev.inmo.tgbotapi.extensions.behaviour_builder.utils.times] | ||||
|  * to combinate several filters | ||||
|  * @param [markerFactory] Will be used to identify different "stream". [scenarioReceiver] will be called synchronously | ||||
|  * in one "stream". Output of [markerFactory] will be used as a key for "stream" | ||||
|  * @param scenarioReceiver Main callback which will be used to handle incoming data if [initialFilter] will pass that | ||||
|  * data | ||||
|  */ | ||||
| internal suspend inline fun <BC : BehaviourContext, reified T : DataCallbackQuery> BC.onDataCallbackQueryCounted( | ||||
|     noinline initialFilter: SimpleFilter<T>? = null, | ||||
|     noinline subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, T, Update>? = CallbackQueryFilterByUser, | ||||
|     markerFactory: MarkerFactory<in T, Any> = ByUserCallbackQueryMarkerFactory, | ||||
|     noinline scenarioReceiver: CustomBehaviourContextAndTypeReceiver<BC, Unit, T> | ||||
| ): Job { | ||||
|     val newInitialFilter = SimpleFilter<DataCallbackQuery> { | ||||
|         it is T && initialFilter ?.invoke(it) ?: true | ||||
|     } | ||||
|     return runCatchingSafely { | ||||
|         onCallbackQuery ( | ||||
|             initialFilter, | ||||
|             subcontextUpdatesFilter, | ||||
|             markerFactory, | ||||
|             scenarioReceiver | ||||
|         ) | ||||
|     }.onFailure { | ||||
|         triggersHolder.handleableCallbackQueriesDataHolder.unregisterHandleable(newInitialFilter) | ||||
|     }.onSuccess { | ||||
|         triggersHolder.handleableCallbackQueriesDataHolder.registerHandleable(newInitialFilter) | ||||
|         it.invokeOnCompletion { | ||||
|             runCatching { | ||||
|                 launchSafelyWithoutExceptions { | ||||
|                     triggersHolder.handleableCallbackQueriesDataHolder.unregisterHandleable(newInitialFilter) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     }.getOrThrow() | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @param initialFilter This filter will be called to remove unnecessary data BEFORE [scenarioReceiver] call | ||||
|  * @param subcontextUpdatesFilter This filter will be applied to each update inside of [scenarioReceiver]. For example, | ||||
| @@ -38,7 +82,7 @@ suspend fun <BC : BehaviourContext> BC.onDataCallbackQuery( | ||||
|     subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, DataCallbackQuery, Update>? = CallbackQueryFilterByUser, | ||||
|     markerFactory: MarkerFactory<in DataCallbackQuery, Any> = ByUserCallbackQueryMarkerFactory, | ||||
|     scenarioReceiver: CustomBehaviourContextAndTypeReceiver<BC, Unit, DataCallbackQuery> | ||||
| ) = onCallbackQuery( | ||||
| ) = onDataCallbackQueryCounted( | ||||
|     initialFilter, | ||||
|     subcontextUpdatesFilter, | ||||
|     markerFactory, | ||||
| @@ -166,7 +210,7 @@ suspend fun <BC : BehaviourContext> BC.onInlineMessageIdDataCallbackQuery( | ||||
|     subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, InlineMessageIdDataCallbackQuery, Update>? = CallbackQueryFilterByUser, | ||||
|     markerFactory: MarkerFactory<in InlineMessageIdDataCallbackQuery, Any> = ByUserCallbackQueryMarkerFactory, | ||||
|     scenarioReceiver: CustomBehaviourContextAndTypeReceiver<BC, Unit, InlineMessageIdDataCallbackQuery> | ||||
| ) = onCallbackQuery( | ||||
| ) = onDataCallbackQueryCounted( | ||||
|     initialFilter, | ||||
|     subcontextUpdatesFilter, | ||||
|     markerFactory, | ||||
| @@ -294,7 +338,7 @@ suspend fun <BC : BehaviourContext> BC.onMessageDataCallbackQuery( | ||||
|     subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, MessageDataCallbackQuery, Update>? = CallbackQueryFilterByUser, | ||||
|     markerFactory: MarkerFactory<in MessageDataCallbackQuery, Any> = ByUserCallbackQueryMarkerFactory, | ||||
|     scenarioReceiver: CustomBehaviourContextAndTypeReceiver<BC, Unit, MessageDataCallbackQuery> | ||||
| ) = onCallbackQuery( | ||||
| ) = onDataCallbackQueryCounted( | ||||
|     initialFilter, | ||||
|     subcontextUpdatesFilter, | ||||
|     markerFactory, | ||||
|   | ||||
| @@ -0,0 +1,87 @@ | ||||
| @file:Suppress("unused") | ||||
|  | ||||
| package dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling | ||||
|  | ||||
| import dev.inmo.micro_utils.coroutines.launchSafelyWithoutExceptions | ||||
| import dev.inmo.micro_utils.coroutines.runCatchingSafely | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.* | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.filters.CallbackQueryFilterByUser | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.utils.* | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.utils.marker_factories.ByUserCallbackQueryMarkerFactory | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.utils.marker_factories.MarkerFactory | ||||
| import dev.inmo.tgbotapi.extensions.utils.asCallbackQueryUpdate | ||||
| import dev.inmo.tgbotapi.types.CallbackQuery.* | ||||
| import dev.inmo.tgbotapi.types.update.abstracts.Update | ||||
| import kotlinx.coroutines.Job | ||||
|  | ||||
| /** | ||||
|  * @param initialFilter This filter will be called to remove unnecessary data BEFORE [scenarioReceiver] call | ||||
|  * @param subcontextUpdatesFilter This filter will be applied to each update inside of [scenarioReceiver]. For example, | ||||
|  * this filter will be used if you will call [dev.inmo.tgbotapi.extensions.behaviour_builder.expectations.waitContentMessage]. | ||||
|  * Use [dev.inmo.tgbotapi.extensions.behaviour_builder.BehaviourContextAndTwoTypesReceiver] function to create your own. | ||||
|  * Use [dev.inmo.tgbotapi.extensions.behaviour_builder.utils.plus] or [dev.inmo.tgbotapi.extensions.behaviour_builder.utils.times] | ||||
|  * to combinate several filters | ||||
|  * @param [markerFactory] Will be used to identify different "stream". [scenarioReceiver] will be called synchronously | ||||
|  * in one "stream". Output of [markerFactory] will be used as a key for "stream" | ||||
|  * @param scenarioReceiver Main callback which will be used to handle incoming data if [initialFilter] will pass that | ||||
|  * data | ||||
|  */ | ||||
| suspend fun <BC : BehaviourContext> BC.onUnhandledDataCallbackQuery( | ||||
|     initialFilter: SimpleFilter<DataCallbackQuery>? = null, | ||||
|     subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, DataCallbackQuery, Update>? = CallbackQueryFilterByUser, | ||||
|     markerFactory: MarkerFactory<in DataCallbackQuery, Any> = ByUserCallbackQueryMarkerFactory, | ||||
|     scenarioReceiver: CustomBehaviourContextAndTypeReceiver<BC, Unit, DataCallbackQuery> | ||||
| ) = onCallbackQuery ( | ||||
|     initialFilter * !SimpleFilter<MessageDataCallbackQuery> { triggersHolder.handleableCallbackQueriesDataHolder.isHandled(it) }, | ||||
|     subcontextUpdatesFilter, | ||||
|     markerFactory, | ||||
|     scenarioReceiver | ||||
| ) | ||||
|  | ||||
| /** | ||||
|  * @param initialFilter This filter will be called to remove unnecessary data BEFORE [scenarioReceiver] call | ||||
|  * @param subcontextUpdatesFilter This filter will be applied to each update inside of [scenarioReceiver]. For example, | ||||
|  * this filter will be used if you will call [dev.inmo.tgbotapi.extensions.behaviour_builder.expectations.waitContentMessage]. | ||||
|  * Use [dev.inmo.tgbotapi.extensions.behaviour_builder.BehaviourContextAndTwoTypesReceiver] function to create your own. | ||||
|  * Use [dev.inmo.tgbotapi.extensions.behaviour_builder.utils.plus] or [dev.inmo.tgbotapi.extensions.behaviour_builder.utils.times] | ||||
|  * to combinate several filters | ||||
|  * @param [markerFactory] Will be used to identify different "stream". [scenarioReceiver] will be called synchronously | ||||
|  * in one "stream". Output of [markerFactory] will be used as a key for "stream" | ||||
|  * @param scenarioReceiver Main callback which will be used to handle incoming data if [initialFilter] will pass that | ||||
|  * data | ||||
|  */ | ||||
| suspend fun <BC : BehaviourContext> BC.onUnhandledInlineMessageIdDataCallbackQuery( | ||||
|     initialFilter: SimpleFilter<InlineMessageIdDataCallbackQuery>? = null, | ||||
|     subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, InlineMessageIdDataCallbackQuery, Update>? = CallbackQueryFilterByUser, | ||||
|     markerFactory: MarkerFactory<in InlineMessageIdDataCallbackQuery, Any> = ByUserCallbackQueryMarkerFactory, | ||||
|     scenarioReceiver: CustomBehaviourContextAndTypeReceiver<BC, Unit, InlineMessageIdDataCallbackQuery> | ||||
| ) = onCallbackQuery ( | ||||
|     initialFilter * !SimpleFilter<MessageDataCallbackQuery> { triggersHolder.handleableCallbackQueriesDataHolder.isHandled(it) }, | ||||
|     subcontextUpdatesFilter, | ||||
|     markerFactory, | ||||
|     scenarioReceiver | ||||
| ) | ||||
|  | ||||
| /** | ||||
|  * @param initialFilter This filter will be called to remove unnecessary data BEFORE [scenarioReceiver] call | ||||
|  * @param subcontextUpdatesFilter This filter will be applied to each update inside of [scenarioReceiver]. For example, | ||||
|  * this filter will be used if you will call [dev.inmo.tgbotapi.extensions.behaviour_builder.expectations.waitContentMessage]. | ||||
|  * Use [dev.inmo.tgbotapi.extensions.behaviour_builder.BehaviourContextAndTwoTypesReceiver] function to create your own. | ||||
|  * Use [dev.inmo.tgbotapi.extensions.behaviour_builder.utils.plus] or [dev.inmo.tgbotapi.extensions.behaviour_builder.utils.times] | ||||
|  * to combinate several filters | ||||
|  * @param [markerFactory] Will be used to identify different "stream". [scenarioReceiver] will be called synchronously | ||||
|  * in one "stream". Output of [markerFactory] will be used as a key for "stream" | ||||
|  * @param scenarioReceiver Main callback which will be used to handle incoming data if [initialFilter] will pass that | ||||
|  * data | ||||
|  */ | ||||
| suspend fun <BC : BehaviourContext> BC.onUnhandledMessageDataCallbackQuery( | ||||
|     initialFilter: SimpleFilter<MessageDataCallbackQuery>? = null, | ||||
|     subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, MessageDataCallbackQuery, Update>? = CallbackQueryFilterByUser, | ||||
|     markerFactory: MarkerFactory<in MessageDataCallbackQuery, Any> = ByUserCallbackQueryMarkerFactory, | ||||
|     scenarioReceiver: CustomBehaviourContextAndTypeReceiver<BC, Unit, MessageDataCallbackQuery> | ||||
| ) = onCallbackQuery( | ||||
|     initialFilter * !SimpleFilter<MessageDataCallbackQuery> { triggersHolder.handleableCallbackQueriesDataHolder.isHandled(it) }, | ||||
|     subcontextUpdatesFilter, | ||||
|     markerFactory, | ||||
|     scenarioReceiver | ||||
| ) | ||||
| @@ -2,6 +2,8 @@ | ||||
|  | ||||
| package dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling | ||||
|  | ||||
| import dev.inmo.micro_utils.coroutines.launchSafelyWithoutExceptions | ||||
| import dev.inmo.micro_utils.coroutines.runCatchingSafely | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.* | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.filters.CommonMessageFilterExcludeMediaGroups | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.filters.MessageFilterByChat | ||||
| @@ -15,8 +17,7 @@ import dev.inmo.tgbotapi.types.message.content.TextContent | ||||
| import dev.inmo.tgbotapi.types.update.abstracts.Update | ||||
| import kotlinx.coroutines.Job | ||||
|  | ||||
|  | ||||
| suspend fun <BC : BehaviourContext> BC.command( | ||||
| internal suspend fun <BC : BehaviourContext> BC.commandUncounted( | ||||
|     commandRegex: Regex, | ||||
|     requireOnlyCommandInMessage: Boolean = true, | ||||
|     initialFilter: CommonMessageFilter<TextContent>? = CommonMessageFilterExcludeMediaGroups, | ||||
| @@ -43,6 +44,35 @@ suspend fun <BC : BehaviourContext> BC.command( | ||||
|     scenarioReceiver | ||||
| ) | ||||
|  | ||||
| suspend fun <BC : BehaviourContext> BC.command( | ||||
|     commandRegex: Regex, | ||||
|     requireOnlyCommandInMessage: Boolean = true, | ||||
|     initialFilter: CommonMessageFilter<TextContent>? = CommonMessageFilterExcludeMediaGroups, | ||||
|     subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, CommonMessage<TextContent>, Update> = MessageFilterByChat, | ||||
|     markerFactory: MarkerFactory<in CommonMessage<TextContent>, Any> = ByChatMessageMarkerFactory, | ||||
|     scenarioReceiver: CustomBehaviourContextAndTypeReceiver<BC, Unit, CommonMessage<TextContent>> | ||||
| ): Job = runCatchingSafely { | ||||
|     commandUncounted( | ||||
|         commandRegex, | ||||
|         requireOnlyCommandInMessage, | ||||
|         initialFilter, | ||||
|         subcontextUpdatesFilter, | ||||
|         markerFactory, | ||||
|         scenarioReceiver | ||||
|     ) | ||||
| }.onFailure { | ||||
|     triggersHolder.handleableCommandsHolder.unregisterHandleable(commandRegex) | ||||
| }.onSuccess { | ||||
|     triggersHolder.handleableCommandsHolder.registerHandleable(commandRegex) | ||||
|     it.invokeOnCompletion { | ||||
|         runCatching { | ||||
|             launchSafelyWithoutExceptions { | ||||
|                 triggersHolder.handleableCommandsHolder.unregisterHandleable(commandRegex) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| }.getOrThrow() | ||||
|  | ||||
| suspend fun <BC : BehaviourContext> BC.command( | ||||
|     command: String, | ||||
|     requireOnlyCommandInMessage: Boolean = true, | ||||
| @@ -52,22 +82,22 @@ suspend fun <BC : BehaviourContext> BC.command( | ||||
|     scenarioReceiver: CustomBehaviourContextAndTypeReceiver<BC, Unit, CommonMessage<TextContent>> | ||||
| ) = command(command.toRegex(), requireOnlyCommandInMessage, initialFilter, subcontextUpdatesFilter, markerFactory, scenarioReceiver) | ||||
|  | ||||
| suspend inline fun <BC : BehaviourContext> BC.onCommand( | ||||
| suspend fun <BC : BehaviourContext> BC.onCommand( | ||||
|     commandRegex: Regex, | ||||
|     requireOnlyCommandInMessage: Boolean = true, | ||||
|     noinline initialFilter: CommonMessageFilter<TextContent>? = CommonMessageFilterExcludeMediaGroups, | ||||
|     noinline subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, CommonMessage<TextContent>, Update> = MessageFilterByChat, | ||||
|     initialFilter: CommonMessageFilter<TextContent>? = CommonMessageFilterExcludeMediaGroups, | ||||
|     subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, CommonMessage<TextContent>, Update> = MessageFilterByChat, | ||||
|     markerFactory: MarkerFactory<in CommonMessage<TextContent>, Any> = ByChatMessageMarkerFactory, | ||||
|     noinline scenarioReceiver: CustomBehaviourContextAndTypeReceiver<BC, Unit, CommonMessage<TextContent>> | ||||
|     scenarioReceiver: CustomBehaviourContextAndTypeReceiver<BC, Unit, CommonMessage<TextContent>> | ||||
| ): Job = command(commandRegex, requireOnlyCommandInMessage, initialFilter, subcontextUpdatesFilter, markerFactory, scenarioReceiver) | ||||
|  | ||||
| suspend inline fun <BC : BehaviourContext> BC.onCommand( | ||||
| suspend fun <BC : BehaviourContext> BC.onCommand( | ||||
|     command: String, | ||||
|     requireOnlyCommandInMessage: Boolean = true, | ||||
|     noinline initialFilter: CommonMessageFilter<TextContent>? = CommonMessageFilterExcludeMediaGroups, | ||||
|     noinline subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, CommonMessage<TextContent>, Update> = MessageFilterByChat, | ||||
|     initialFilter: CommonMessageFilter<TextContent>? = CommonMessageFilterExcludeMediaGroups, | ||||
|     subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, CommonMessage<TextContent>, Update> = MessageFilterByChat, | ||||
|     markerFactory: MarkerFactory<in CommonMessage<TextContent>, Any> = ByChatMessageMarkerFactory, | ||||
|     noinline scenarioReceiver: CustomBehaviourContextAndTypeReceiver<BC, Unit, CommonMessage<TextContent>> | ||||
|     scenarioReceiver: CustomBehaviourContextAndTypeReceiver<BC, Unit, CommonMessage<TextContent>> | ||||
| ): Job = onCommand(command.toRegex(), requireOnlyCommandInMessage, initialFilter, subcontextUpdatesFilter, markerFactory, scenarioReceiver) | ||||
|  | ||||
| suspend fun <BC : BehaviourContext> BC.commandWithArgs( | ||||
| @@ -104,18 +134,18 @@ suspend fun <BC : BehaviourContext> BC.commandWithArgs( | ||||
|     scenarioReceiver = scenarioReceiver | ||||
| ) | ||||
|  | ||||
| suspend inline fun <BC : BehaviourContext> BC.onCommandWithArgs( | ||||
| suspend fun <BC : BehaviourContext> BC.onCommandWithArgs( | ||||
|     commandRegex: Regex, | ||||
|     noinline initialFilter: CommonMessageFilter<TextContent>? = CommonMessageFilterExcludeMediaGroups, | ||||
|     noinline subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, CommonMessage<TextContent>, Update> = MessageFilterByChat, | ||||
|     initialFilter: CommonMessageFilter<TextContent>? = CommonMessageFilterExcludeMediaGroups, | ||||
|     subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, CommonMessage<TextContent>, Update> = MessageFilterByChat, | ||||
|     markerFactory: MarkerFactory<in CommonMessage<TextContent>, Any> = ByChatMessageMarkerFactory, | ||||
|     noinline scenarioReceiver: CustomBehaviourContextAndTwoTypesReceiver<BC, Unit, CommonMessage<TextContent>, Array<String>> | ||||
|     scenarioReceiver: CustomBehaviourContextAndTwoTypesReceiver<BC, Unit, CommonMessage<TextContent>, Array<String>> | ||||
| ): Job = commandWithArgs(commandRegex, initialFilter, subcontextUpdatesFilter, markerFactory, scenarioReceiver) | ||||
|  | ||||
| suspend inline fun <BC : BehaviourContext> BC.onCommandWithArgs( | ||||
| suspend fun <BC : BehaviourContext> BC.onCommandWithArgs( | ||||
|     command: String, | ||||
|     noinline initialFilter: CommonMessageFilter<TextContent>? = CommonMessageFilterExcludeMediaGroups, | ||||
|     noinline subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, CommonMessage<TextContent>, Update> = MessageFilterByChat, | ||||
|     initialFilter: CommonMessageFilter<TextContent>? = CommonMessageFilterExcludeMediaGroups, | ||||
|     subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, CommonMessage<TextContent>, Update> = MessageFilterByChat, | ||||
|     markerFactory: MarkerFactory<in CommonMessage<TextContent>, Any> = ByChatMessageMarkerFactory, | ||||
|     noinline scenarioReceiver: CustomBehaviourContextAndTwoTypesReceiver<BC, Unit, CommonMessage<TextContent>, Array<String>> | ||||
|     scenarioReceiver: CustomBehaviourContextAndTwoTypesReceiver<BC, Unit, CommonMessage<TextContent>, Array<String>> | ||||
| ): Job = onCommandWithArgs(command.toRegex(), initialFilter, subcontextUpdatesFilter, markerFactory, scenarioReceiver) | ||||
|   | ||||
| @@ -0,0 +1,76 @@ | ||||
| @file:Suppress("unused") | ||||
|  | ||||
| package dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling | ||||
|  | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.* | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.filters.CommonMessageFilterExcludeMediaGroups | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.filters.MessageFilterByChat | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.utils.marker_factories.ByChatMessageMarkerFactory | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.utils.marker_factories.MarkerFactory | ||||
| import dev.inmo.tgbotapi.extensions.behaviour_builder.utils.times | ||||
| import dev.inmo.tgbotapi.extensions.utils.asBotCommandTextSource | ||||
| import dev.inmo.tgbotapi.extensions.utils.extensions.parseCommandsWithParams | ||||
| import dev.inmo.tgbotapi.types.message.abstracts.CommonMessage | ||||
| import dev.inmo.tgbotapi.types.message.content.TextContent | ||||
| import dev.inmo.tgbotapi.types.update.abstracts.Update | ||||
| import kotlinx.coroutines.Job | ||||
|  | ||||
|  | ||||
| suspend fun <BC : BehaviourContext> BC.unhandledCommand( | ||||
|     requireOnlyCommandInMessage: Boolean = true, | ||||
|     initialFilter: CommonMessageFilter<TextContent>? = CommonMessageFilterExcludeMediaGroups, | ||||
|     subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, CommonMessage<TextContent>, Update> = MessageFilterByChat, | ||||
|     markerFactory: MarkerFactory<in CommonMessage<TextContent>, Any> = ByChatMessageMarkerFactory, | ||||
|     scenarioReceiver: CustomBehaviourContextAndTypeReceiver<BC, Unit, CommonMessage<TextContent>> | ||||
| ): Job = onText( | ||||
|     CommonMessageFilter<TextContent> { message -> | ||||
|         val content = message.content | ||||
|         val textSources = content.textSources | ||||
|         val sizeRequirement = if (requireOnlyCommandInMessage) { | ||||
|             textSources.size == 1 | ||||
|         } else { | ||||
|             true | ||||
|         } | ||||
|         sizeRequirement && textSources.any { | ||||
|             val command = it.asBotCommandTextSource() ?.command ?: return@any false | ||||
|             !triggersHolder.handleableCommandsHolder.isHandled(command) | ||||
|         } | ||||
|     }.let { | ||||
|         initialFilter ?.times(it) ?: it | ||||
|     }, | ||||
|     subcontextUpdatesFilter, | ||||
|     markerFactory, | ||||
|     scenarioReceiver | ||||
| ) | ||||
|  | ||||
| suspend fun <BC : BehaviourContext> BC.onUnhandledCommand( | ||||
|     requireOnlyCommandInMessage: Boolean = true, | ||||
|     initialFilter: CommonMessageFilter<TextContent>? = CommonMessageFilterExcludeMediaGroups, | ||||
|     subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, CommonMessage<TextContent>, Update> = MessageFilterByChat, | ||||
|     markerFactory: MarkerFactory<in CommonMessage<TextContent>, Any> = ByChatMessageMarkerFactory, | ||||
|     scenarioReceiver: CustomBehaviourContextAndTypeReceiver<BC, Unit, CommonMessage<TextContent>> | ||||
| ): Job = unhandledCommand(requireOnlyCommandInMessage, initialFilter, subcontextUpdatesFilter, markerFactory, scenarioReceiver) | ||||
|  | ||||
| suspend fun <BC : BehaviourContext> BC.unhandledCommandWithArgs( | ||||
|     initialFilter: CommonMessageFilter<TextContent>? = CommonMessageFilterExcludeMediaGroups, | ||||
|     subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, CommonMessage<TextContent>, Update> = MessageFilterByChat, | ||||
|     markerFactory: MarkerFactory<in CommonMessage<TextContent>, Any> = ByChatMessageMarkerFactory, | ||||
|     scenarioReceiver: CustomBehaviourContextAndTwoTypesReceiver<BC, Unit, CommonMessage<TextContent>, Map<String, Array<String>>> | ||||
| ) = onUnhandledCommand( | ||||
|     requireOnlyCommandInMessage = false, | ||||
|     initialFilter = initialFilter, | ||||
|     subcontextUpdatesFilter = subcontextUpdatesFilter, | ||||
|     markerFactory = markerFactory | ||||
| ) { | ||||
|     val args = it.parseCommandsWithParams().let { commandsWithArgs -> | ||||
|         commandsWithArgs | ||||
|     } | ||||
|     scenarioReceiver(it, args) | ||||
| } | ||||
|  | ||||
| suspend fun <BC : BehaviourContext> BC.onUnhandledCommandWithArgs( | ||||
|     initialFilter: CommonMessageFilter<TextContent>? = CommonMessageFilterExcludeMediaGroups, | ||||
|     subcontextUpdatesFilter: CustomBehaviourContextAndTwoTypesReceiver<BC, Boolean, CommonMessage<TextContent>, Update> = MessageFilterByChat, | ||||
|     markerFactory: MarkerFactory<in CommonMessage<TextContent>, Any> = ByChatMessageMarkerFactory, | ||||
|     scenarioReceiver: CustomBehaviourContextAndTwoTypesReceiver<BC, Unit, CommonMessage<TextContent>, Map<String, Array<String>>> | ||||
| ): Job = unhandledCommandWithArgs(initialFilter, subcontextUpdatesFilter, markerFactory, scenarioReceiver) | ||||
| @@ -3,6 +3,7 @@ package dev.inmo.tgbotapi.extensions.behaviour_builder.utils | ||||
| typealias SimpleFilter<T> = suspend (T) -> Boolean | ||||
|  | ||||
| inline fun <T> SimpleFilter(noinline block: SimpleFilter<T>) = block | ||||
| val TrueSimpleFilter = SimpleFilter<Any?> { true } | ||||
|  | ||||
| /** | ||||
|  * @return [SimpleFilter] which will return true in case when all the items in incoming data passed [this] filter | ||||
|   | ||||
| @@ -0,0 +1,5 @@ | ||||
| package dev.inmo.tgbotapi.extensions.behaviour_builder.utils.handlers_registrar | ||||
|  | ||||
| class HandleableCallbackBasedHolder<T> : HandleableTriggersHolder<suspend (T) -> Boolean>() { | ||||
|     suspend fun isHandled(data: T) = handleable.any { it(data) } | ||||
| } | ||||
| @@ -0,0 +1,7 @@ | ||||
| package dev.inmo.tgbotapi.extensions.behaviour_builder.utils.handlers_registrar | ||||
|  | ||||
| class HandleableRegexesHolder : HandleableTriggersHolder<Regex>() { | ||||
|     fun isHandled(command: String) = handleable.any { | ||||
|         it.matches(command) | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,27 @@ | ||||
| package dev.inmo.tgbotapi.extensions.behaviour_builder.utils.handlers_registrar | ||||
|  | ||||
| import kotlinx.coroutines.sync.Mutex | ||||
| import kotlinx.coroutines.sync.withLock | ||||
|  | ||||
| open class HandleableTriggersHolder<T>( | ||||
|     preset: List<T> = emptyList() | ||||
| ) { | ||||
|     protected val commandsMutex = Mutex() | ||||
|     protected val _handleable = mutableListOf<T>().also { | ||||
|         it.addAll(preset) | ||||
|     } | ||||
|     val handleable: List<T> | ||||
|         get() = _handleable.toList() | ||||
|  | ||||
|     suspend fun registerHandleable(data: T) { | ||||
|         commandsMutex.withLock { | ||||
|             _handleable.add(data) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     suspend fun unregisterHandleable(data: T) { | ||||
|         commandsMutex.withLock { | ||||
|             _handleable.remove(data) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,9 @@ | ||||
| package dev.inmo.tgbotapi.extensions.behaviour_builder.utils.handlers_registrar | ||||
|  | ||||
| import dev.inmo.tgbotapi.types.CallbackQuery.DataCallbackQuery | ||||
| import dev.inmo.tgbotapi.types.CallbackQuery.MessageDataCallbackQuery | ||||
|  | ||||
| class TriggersHolder { | ||||
|     val handleableCommandsHolder = HandleableRegexesHolder() | ||||
|     val handleableCallbackQueriesDataHolder = HandleableCallbackBasedHolder<DataCallbackQuery>() | ||||
| } | ||||
		Reference in New Issue
	
	Block a user