mirror of
				https://github.com/InsanusMokrassar/TelegramBotAPI.git
				synced 2025-10-25 01:00:13 +00:00 
			
		
		
		
	| @@ -1,5 +1,13 @@ | ||||
| # TelegramBotAPI changelog | ||||
|  | ||||
| ## 1.1.2 | ||||
|  | ||||
| * `Core`: | ||||
|     * Rename of `TelegramAPIUrlsKeeper#checkWebAppLink` -> `TelegramAPIUrlsKeeper#checkWebAppData` (fix of [#591](https://github.com/InsanusMokrassar/TelegramBotAPI/issues/591)) | ||||
| * `Behaviour Builder with FSM`: | ||||
|     * New typealias `StateHandlingErrorHandler` | ||||
|     * `BehaviourBuilderWithFSM` now accepts new parameter `onStateHandlingErrorHandler` which will be used in case if state has not been successfully completed | ||||
|  | ||||
| ## 1.1.1 | ||||
|  | ||||
| * `Versions`: | ||||
|   | ||||
| @@ -20,6 +20,6 @@ javax_activation_version=1.1.1 | ||||
| dokka_version=1.6.21 | ||||
|  | ||||
| library_group=dev.inmo | ||||
| library_version=1.1.1 | ||||
| library_version=1.1.2 | ||||
|  | ||||
| github_release_plugin_version=2.3.7 | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| package dev.inmo.tgbotapi.extensions.behaviour_builder | ||||
|  | ||||
| import dev.inmo.micro_utils.coroutines.launchSafelyWithoutExceptions | ||||
| import dev.inmo.micro_utils.coroutines.subscribeSafelyWithoutExceptions | ||||
| import dev.inmo.micro_utils.coroutines.* | ||||
| import dev.inmo.micro_utils.fsm.common.* | ||||
| import dev.inmo.tgbotapi.bot.TelegramBot | ||||
| import dev.inmo.tgbotapi.types.update.abstracts.Update | ||||
| @@ -50,12 +49,32 @@ interface BehaviourContextWithFSM<T : State> : BehaviourContext, StatesMachine<T | ||||
|         updatesFilter: BehaviourContextAndTypeReceiver<Boolean, Update>? | ||||
|     ): BehaviourContextWithFSM<T> | ||||
|  | ||||
|     fun copy( | ||||
|         bot: TelegramBot = this.bot, | ||||
|         scope: CoroutineScope = this.scope, | ||||
|         broadcastChannelsSize: Int = 100, | ||||
|         onBufferOverflow: BufferOverflow = BufferOverflow.SUSPEND, | ||||
|         upstreamUpdatesFlow: Flow<Update>? = null, | ||||
|         triggersHolder: TriggersHolder = this.triggersHolder, | ||||
|         onStateHandlingErrorHandler: StateHandlingErrorHandler<T> = defaultStateHandlingErrorHandler(), | ||||
|         updatesFilter: BehaviourContextAndTypeReceiver<Boolean, Update>? = null | ||||
|     ): BehaviourContextWithFSM<T> = copy( | ||||
|         bot, | ||||
|         scope, | ||||
|         broadcastChannelsSize, | ||||
|         onBufferOverflow, | ||||
|         upstreamUpdatesFlow, | ||||
|         triggersHolder, | ||||
|         updatesFilter | ||||
|     ) | ||||
|  | ||||
|     companion object { | ||||
|         operator fun <T : State> invoke( | ||||
|             behaviourContext: BehaviourContext, | ||||
|             handlers: List<BehaviourWithFSMStateHandlerHolder<*, T>>, | ||||
|             statesManager: StatesManager<T> | ||||
|         ) = DefaultBehaviourContextWithFSM<T>(behaviourContext, statesManager, handlers) | ||||
|             statesManager: StatesManager<T>, | ||||
|             onStateHandlingErrorHandler: StateHandlingErrorHandler<T> = defaultStateHandlingErrorHandler() | ||||
|         ) = DefaultBehaviourContextWithFSM<T>(behaviourContext, statesManager, handlers, onStateHandlingErrorHandler) | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -84,16 +103,26 @@ inline fun <reified I : O, O: State> BehaviourContextWithFSM<O>.strictlyOn(handl | ||||
| /** | ||||
|  * Default realization of [BehaviourContextWithFSM]. It uses [behaviourContext] as a base for this object as | ||||
|  * [BehaviourContext], but managing substates contexts updates for avoiding of updates lost between states | ||||
|  * @param onStateHandlingErrorHandler Will be used in case if state handling has not been successfully completed in [launchStateHandling] | ||||
|  */ | ||||
| class DefaultBehaviourContextWithFSM<T : State>( | ||||
|     private val behaviourContext: BehaviourContext, | ||||
|     private val statesManager: StatesManager<T>, | ||||
|     private val handlers: List<BehaviourWithFSMStateHandlerHolder<*, T>> | ||||
|     private val handlers: List<BehaviourWithFSMStateHandlerHolder<*, T>>, | ||||
|     private val onStateHandlingErrorHandler: StateHandlingErrorHandler<T> = defaultStateHandlingErrorHandler() | ||||
| ) : BehaviourContext by behaviourContext, BehaviourContextWithFSM<T> { | ||||
|     private val updatesFlows = mutableMapOf<Any, DefaultBehaviourContextWithFSM<T>>() | ||||
|     private val additionalHandlers = mutableListOf<BehaviourWithFSMStateHandlerHolder<*, T>>() | ||||
|     private var actualHandlersList = additionalHandlers + handlers | ||||
|  | ||||
|     override suspend fun launchStateHandling(state: T, handlers: List<CheckableHandlerHolder<in T, T>>): T? { | ||||
|         return runCatchingSafely { | ||||
|             super.launchStateHandling(state, handlers) | ||||
|         }.getOrElse { | ||||
|             onStateHandlingErrorHandler(state, it) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun getSubContext(context: Any) = updatesFlows.getOrPut(context) { | ||||
|         createSubContext() | ||||
|     } | ||||
| @@ -174,6 +203,23 @@ class DefaultBehaviourContextWithFSM<T : State>( | ||||
|     ): DefaultBehaviourContextWithFSM<T> = BehaviourContextWithFSM( | ||||
|         behaviourContext.copy(bot, scope, broadcastChannelsSize, onBufferOverflow, upstreamUpdatesFlow, triggersHolder, updatesFilter), | ||||
|         handlers, | ||||
|         statesManager | ||||
|         statesManager, | ||||
|         onStateHandlingErrorHandler | ||||
|     ) | ||||
|  | ||||
|     override fun copy( | ||||
|         bot: TelegramBot, | ||||
|         scope: CoroutineScope, | ||||
|         broadcastChannelsSize: Int, | ||||
|         onBufferOverflow: BufferOverflow, | ||||
|         upstreamUpdatesFlow: Flow<Update>?, | ||||
|         triggersHolder: TriggersHolder, | ||||
|         onStateHandlingErrorHandler: StateHandlingErrorHandler<T>, | ||||
|         updatesFilter: BehaviourContextAndTypeReceiver<Boolean, Update>? | ||||
|     ): DefaultBehaviourContextWithFSM<T> = BehaviourContextWithFSM( | ||||
|         behaviourContext.copy(bot, scope, broadcastChannelsSize, onBufferOverflow, upstreamUpdatesFlow, triggersHolder, updatesFilter), | ||||
|         handlers, | ||||
|         statesManager, | ||||
|         onStateHandlingErrorHandler | ||||
|     ) | ||||
| } | ||||
|   | ||||
| @@ -6,13 +6,10 @@ import dev.inmo.micro_utils.fsm.common.managers.DefaultStatesManager | ||||
| import dev.inmo.micro_utils.fsm.common.managers.InMemoryDefaultStatesManagerRepo | ||||
| import dev.inmo.tgbotapi.bot.TelegramBot | ||||
| import dev.inmo.tgbotapi.extensions.utils.updates.retrieving.longPolling | ||||
| import dev.inmo.tgbotapi.extensions.utils.updates.retrieving.startGettingOfUpdatesByLongPolling | ||||
| import dev.inmo.tgbotapi.types.update.abstracts.Update | ||||
| import dev.inmo.tgbotapi.updateshandlers.FlowsUpdatesFilter | ||||
| import dev.inmo.tgbotapi.utils.PreviewFeature | ||||
| import kotlinx.coroutines.* | ||||
| import kotlinx.coroutines.flow.Flow | ||||
| import kotlin.reflect.KClass | ||||
|  | ||||
| @Deprecated("Will be removed soon") | ||||
| typealias BehaviourContextWithFSMBuilder<T> = BehaviourContextWithFSM<T> | ||||
| @@ -33,6 +30,7 @@ suspend fun <T : State> TelegramBot.buildBehaviourWithFSM( | ||||
|     defaultExceptionsHandler: ExceptionHandler<Unit>? = null, | ||||
|     statesManager: StatesManager<T> = DefaultStatesManager(InMemoryDefaultStatesManagerRepo()), | ||||
|     presetHandlers: List<BehaviourWithFSMStateHandlerHolder<*, T>> = listOf(), | ||||
|     onStateHandlingErrorHandler: StateHandlingErrorHandler<T> = defaultStateHandlingErrorHandler(), | ||||
|     block: CustomBehaviourContextReceiver<DefaultBehaviourContextWithFSM<T>, Unit> | ||||
| ): DefaultBehaviourContextWithFSM<T> = BehaviourContextWithFSM( | ||||
|     DefaultBehaviourContext( | ||||
| @@ -41,7 +39,8 @@ suspend fun <T : State> TelegramBot.buildBehaviourWithFSM( | ||||
|         upstreamUpdatesFlow = upstreamUpdatesFlow | ||||
|     ), | ||||
|     presetHandlers, | ||||
|     statesManager | ||||
|     statesManager, | ||||
|     onStateHandlingErrorHandler | ||||
| ).apply { block() } | ||||
|  | ||||
| /** | ||||
| @@ -55,6 +54,7 @@ suspend fun <T : State> TelegramBot.buildBehaviourWithFSMAndStartLongPolling( | ||||
|     defaultExceptionsHandler: ExceptionHandler<Unit>? = null, | ||||
|     statesManager: StatesManager<T> = DefaultStatesManager(InMemoryDefaultStatesManagerRepo()), | ||||
|     presetHandlers: List<BehaviourWithFSMStateHandlerHolder<*, T>> = listOf(), | ||||
|     onStateHandlingErrorHandler: StateHandlingErrorHandler<T> = defaultStateHandlingErrorHandler(), | ||||
|     block: CustomBehaviourContextReceiver<DefaultBehaviourContextWithFSM<T>, Unit> | ||||
| ): Pair<DefaultBehaviourContextWithFSM<T>, Job> = buildBehaviourWithFSM( | ||||
|     upstreamUpdatesFlow, | ||||
| @@ -62,6 +62,7 @@ suspend fun <T : State> TelegramBot.buildBehaviourWithFSMAndStartLongPolling( | ||||
|     defaultExceptionsHandler, | ||||
|     statesManager, | ||||
|     presetHandlers, | ||||
|     onStateHandlingErrorHandler, | ||||
|     block | ||||
| ).run { | ||||
|     this to scope.launch { | ||||
| @@ -94,6 +95,7 @@ suspend fun <T : State> TelegramBot.buildBehaviourWithFSM( | ||||
|     defaultExceptionsHandler: ExceptionHandler<Unit>? = null, | ||||
|     statesManager: StatesManager<T> = DefaultStatesManager(InMemoryDefaultStatesManagerRepo()), | ||||
|     presetHandlers: List<BehaviourWithFSMStateHandlerHolder<*, T>> = listOf(), | ||||
|     onStateHandlingErrorHandler: StateHandlingErrorHandler<T> = defaultStateHandlingErrorHandler(), | ||||
|     block: CustomBehaviourContextReceiver<DefaultBehaviourContextWithFSM<T>, Unit> | ||||
| ): DefaultBehaviourContextWithFSM<T> = BehaviourContextWithFSM( | ||||
|     DefaultBehaviourContext( | ||||
| @@ -102,7 +104,8 @@ suspend fun <T : State> TelegramBot.buildBehaviourWithFSM( | ||||
|         upstreamUpdatesFlow = flowUpdatesFilter.allUpdatesFlow | ||||
|     ), | ||||
|     presetHandlers, | ||||
|     statesManager | ||||
|     statesManager, | ||||
|     onStateHandlingErrorHandler | ||||
| ).apply { block() } | ||||
|  | ||||
| /** | ||||
| @@ -121,6 +124,7 @@ suspend fun <T : State> TelegramBot.buildBehaviourWithFSMAndStartLongPolling( | ||||
|     defaultExceptionsHandler: ExceptionHandler<Unit>? = null, | ||||
|     statesManager: StatesManager<T> = DefaultStatesManager(InMemoryDefaultStatesManagerRepo()), | ||||
|     presetHandlers: List<BehaviourWithFSMStateHandlerHolder<*, T>> = listOf(), | ||||
|     onStateHandlingErrorHandler: StateHandlingErrorHandler<T> = defaultStateHandlingErrorHandler(), | ||||
|     block: CustomBehaviourContextReceiver<DefaultBehaviourContextWithFSM<T>, Unit> | ||||
| ) = FlowsUpdatesFilter().let { | ||||
|     buildBehaviourWithFSM( | ||||
| @@ -129,6 +133,7 @@ suspend fun <T : State> TelegramBot.buildBehaviourWithFSMAndStartLongPolling( | ||||
|         defaultExceptionsHandler, | ||||
|         statesManager, | ||||
|         presetHandlers, | ||||
|         onStateHandlingErrorHandler, | ||||
|         block | ||||
|     ).run { | ||||
|         start() | ||||
|   | ||||
| @@ -0,0 +1,5 @@ | ||||
| package dev.inmo.tgbotapi.extensions.behaviour_builder | ||||
|  | ||||
| typealias StateHandlingErrorHandler<T> = suspend (T, Throwable) -> T? | ||||
| val DefaultStateHandlingErrorHandler: StateHandlingErrorHandler<*> = { _, _ -> null } | ||||
| inline fun <T> defaultStateHandlingErrorHandler(): StateHandlingErrorHandler<T> = DefaultStateHandlingErrorHandler as StateHandlingErrorHandler<T> | ||||
| @@ -39,6 +39,7 @@ suspend fun <T : State> telegramBotWithBehaviourAndFSM( | ||||
|     statesManager: StatesManager<T> = DefaultStatesManager(InMemoryDefaultStatesManagerRepo()), | ||||
|     presetHandlers: List<BehaviourWithFSMStateHandlerHolder<*, T>> = listOf(), | ||||
|     testServer: Boolean = false, | ||||
|     onStateHandlingErrorHandler: StateHandlingErrorHandler<T> = defaultStateHandlingErrorHandler(), | ||||
|     block: CustomBehaviourContextReceiver<DefaultBehaviourContextWithFSM<T>, Unit> | ||||
| ): TelegramBot = telegramBot( | ||||
|     token, | ||||
| @@ -52,6 +53,7 @@ suspend fun <T : State> telegramBotWithBehaviourAndFSM( | ||||
|         defaultExceptionsHandler, | ||||
|         statesManager, | ||||
|         presetHandlers, | ||||
|         onStateHandlingErrorHandler, | ||||
|         block | ||||
|     ) | ||||
| } | ||||
| @@ -76,6 +78,7 @@ suspend fun <T : State> telegramBotWithBehaviourAndFSMAndStartLongPolling( | ||||
|     statesManager: StatesManager<T> = DefaultStatesManager(InMemoryDefaultStatesManagerRepo()), | ||||
|     presetHandlers: List<BehaviourWithFSMStateHandlerHolder<*, T>> = listOf(), | ||||
|     testServer: Boolean = false, | ||||
|     onStateHandlingErrorHandler: StateHandlingErrorHandler<T> = defaultStateHandlingErrorHandler(), | ||||
|     block: CustomBehaviourContextReceiver<DefaultBehaviourContextWithFSM<T>, Unit> | ||||
| ): Pair<TelegramBot, Job> { | ||||
|     return telegramBot( | ||||
| @@ -89,6 +92,7 @@ suspend fun <T : State> telegramBotWithBehaviourAndFSMAndStartLongPolling( | ||||
|             defaultExceptionsHandler, | ||||
|             statesManager, | ||||
|             presetHandlers, | ||||
|             onStateHandlingErrorHandler, | ||||
|             block | ||||
|         ) | ||||
|     } | ||||
|   | ||||
| @@ -50,7 +50,7 @@ class TelegramAPIUrlsKeeper( | ||||
|      * @param rawData Data from [dev.inmo.tgbotapi.webapps.WebApp.initData] | ||||
|      * @param hash Data from [dev.inmo.tgbotapi.webapps.WebApp.initDataUnsafe] from the field [dev.inmo.tgbotapi.webapps.WebAppInitData.hash] | ||||
|      */ | ||||
|     fun checkWebAppLink(rawData: String, hash: String): Boolean { | ||||
|     fun checkWebAppData(rawData: String, hash: String): Boolean { | ||||
|         val preparedData = rawData | ||||
|             .decodeURLQueryComponent() | ||||
|             .split("&") | ||||
| @@ -60,4 +60,11 @@ class TelegramAPIUrlsKeeper( | ||||
|  | ||||
|         return HMAC.hmacSHA256(webAppDataSecretKeyHash.bytes, preparedData.toByteArray()).hexLower == hash.lowercase() | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @param rawData Data from [dev.inmo.tgbotapi.webapps.WebApp.initData] | ||||
|      * @param hash Data from [dev.inmo.tgbotapi.webapps.WebApp.initDataUnsafe] from the field [dev.inmo.tgbotapi.webapps.WebAppInitData.hash] | ||||
|      */ | ||||
|     @Deprecated("Renamed", ReplaceWith("checkWebAppData")) | ||||
|     inline fun checkWebAppLink(rawData: String, hash: String) = checkWebAppData(rawData, hash) | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| package dev.inmo.tgbotapi.webapps | ||||
|  | ||||
| import dev.inmo.micro_utils.crypto.CryptoJS | ||||
| import dev.inmo.tgbotapi.utils.TelegramAPIUrlsKeeper | ||||
|  | ||||
| external class WebApp { | ||||
| @@ -77,7 +76,7 @@ fun WebApp.onMainButtonClicked(eventHandler: EventHandler) = onEvent(EventType.M | ||||
|  */ | ||||
| fun WebApp.onViewportChanged(eventHandler: ViewportChangedEventHandler) = onEvent(EventType.ViewportChanged, eventHandler) | ||||
|  | ||||
| fun WebApp.isInitDataSafe(botToken: String) = TelegramAPIUrlsKeeper(botToken).checkWebAppLink( | ||||
| fun WebApp.isInitDataSafe(botToken: String) = TelegramAPIUrlsKeeper(botToken).checkWebAppData( | ||||
|     initData, | ||||
|     initDataUnsafe.hash | ||||
| ) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user