mirror of
https://github.com/InsanusMokrassar/TelegramBotAPI.git
synced 2025-11-17 20:40:20 +00:00
Compare commits
39 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| e8a7ea9ce4 | |||
| 30f35e5488 | |||
| b626974ea7 | |||
| 128b782a27 | |||
| dc83b8d77a | |||
| 2aac6c0bde | |||
| b637d0d2a6 | |||
| d12cba2343 | |||
| 450c7e6474 | |||
| a35fb40a66 | |||
| 8731aa2c26 | |||
| 32e305537d | |||
| ec0a10e36e | |||
| 131ec4d6d5 | |||
| ecc608f51a | |||
| 040654f131 | |||
| 9e73d0c461 | |||
| d67c80bc99 | |||
| b0eef4f82d | |||
| fbe91a6321 | |||
| 15066c9d63 | |||
| e0bf67d8f9 | |||
| 1dc3ce2fb5 | |||
| 0c71133969 | |||
| 1359dd549f | |||
| c425e2ecc3 | |||
| 4d63e3a17d | |||
| d34deade0d | |||
| 2f52ad45a4 | |||
| 1d99e632a4 | |||
| cdcfaf5a29 | |||
| f3590762f3 | |||
| e0ff14b7fe | |||
| 9983e111ec | |||
| 08b8710772 | |||
| fc71e028c4 | |||
| 30a4a7bd8b | |||
| b973278b0a | |||
| ae2f4579e2 |
11
.github/dependabot.yml
vendored
Normal file
11
.github/dependabot.yml
vendored
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
# To get started with Dependabot version updates, you'll need to specify which
|
||||||
|
# package ecosystems to update and where the package manifests are located.
|
||||||
|
# Please see the documentation for all configuration options:
|
||||||
|
# https://help.github.com/github/administering-a-repository/configuration-options-for-dependency-updates
|
||||||
|
|
||||||
|
version: 2
|
||||||
|
updates:
|
||||||
|
- package-ecosystem: "gradle" # See documentation for possible values
|
||||||
|
directory: "/tgbotapi.core" # Location of package manifests
|
||||||
|
schedule:
|
||||||
|
interval: "daily"
|
||||||
16
.github/workflows/regular-build.yml
vendored
Normal file
16
.github/workflows/regular-build.yml
vendored
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
name: Build
|
||||||
|
|
||||||
|
on: [push]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- name: Set up JDK 1.8
|
||||||
|
uses: actions/setup-java@v1
|
||||||
|
with:
|
||||||
|
java-version: 1.8
|
||||||
|
- name: Build with Gradle
|
||||||
|
run: ./gradlew build
|
||||||
71
CHANGELOG.md
71
CHANGELOG.md
@@ -1,10 +1,77 @@
|
|||||||
# TelegramBotAPI changelog
|
# TelegramBotAPI changelog
|
||||||
|
|
||||||
## 0.31.1
|
## 0.32.3
|
||||||
|
|
||||||
|
* `Behaviour Builder`:
|
||||||
|
* Add expectators and waiters for inline queries
|
||||||
|
|
||||||
|
## 0.32.2
|
||||||
|
|
||||||
|
* `Core`:
|
||||||
|
* Fix of [#275](https://github.com/InsanusMokrassar/TelegramBotAPI/issues/275)
|
||||||
|
|
||||||
|
## 0.32.1
|
||||||
|
|
||||||
|
* `Core`:
|
||||||
|
* Fix of [#272](https://github.com/InsanusMokrassar/TelegramBotAPI/issues/272)
|
||||||
|
* `Utils`:
|
||||||
|
* Fix of [#273](https://github.com/InsanusMokrassar/TelegramBotAPI/issues/273)
|
||||||
|
|
||||||
|
## 0.32.0
|
||||||
|
|
||||||
|
**THIS UPDATE CONTAINS BREAKING CHANGES**
|
||||||
|
|
||||||
* `Common`:
|
* `Common`:
|
||||||
* `Version`:
|
* `Version`:
|
||||||
* `MicroUtils`: `0.4.16` -> `0.4.18`
|
* `MicroUtils`: `0.4.16` -> `0.4.23`
|
||||||
|
* `Klock`: `0.2.3` -> `0.2.4`
|
||||||
|
* `Ktor`: `1.5.0` -> `1.5.1`
|
||||||
|
* `Core`:
|
||||||
|
* **BREAKING CHANGE** Now `MediaGroupMessage` have a generic type related to `MediaGroupContent`
|
||||||
|
* Methods and types related to `MediaGroupMessage` have been modified according to their meanings
|
||||||
|
* **Important Change** `FlowsUpdatesFilter` now is an interface. Old class has been renamed to
|
||||||
|
`DefaultFlowsUpdatesFilter` and factory method `FlowsUpdatesFilter` has been added
|
||||||
|
* **PASSPORT** Full support of `Telegram Passport API`
|
||||||
|
* `PassportData`
|
||||||
|
* All variants of `EncryptedPassportElement`
|
||||||
|
* All variants of `SecureValue`
|
||||||
|
* All variants of `PassportElementError`
|
||||||
|
* New request `SetPassportDataErrors`
|
||||||
|
* `Credentials`:
|
||||||
|
* `EncryptedCredentials`
|
||||||
|
* `DeryptedCredentials`
|
||||||
|
* `EndDataCredentials`
|
||||||
|
* `Behaviour Builder`:
|
||||||
|
* Trigger and expectation extensions for `MessageContent` (`onContentMessage` and `waitContentMessage`)
|
||||||
|
* `onMediaGroup` has been replaced
|
||||||
|
* `waitMediaGroup` has been added
|
||||||
|
* `onVisualMediaGroup` now is just an alternative to `onVisualGallery`
|
||||||
|
* `command` and `onCommand` expectations has been added for commands `String` variant
|
||||||
|
* New extensions `BehaviourContext#oneOf`, `BehaviourContext#parallel` and `Deferred<T>#withAction`
|
||||||
|
* Several renames:
|
||||||
|
* `waitAudioMediaGroup` -> `waitAudioMediaGroupContent`
|
||||||
|
* `waitDocumentMediaGroup` -> `waitDocumentMediaGroupContent`
|
||||||
|
* `waitMediaGroup` -> `waitAnyMediaGroupContent`
|
||||||
|
* `waitVisualMediaGroup` -> `waitVisualMediaGroupContent`
|
||||||
|
* New extensions `BehaviourContext#waitPassportMessagesWith` and `BehaviourContext#waitAnyPassportMessages`
|
||||||
|
* New extensions `BehaviourContext#onPassportMessage` and `BehaviourContext#onPassportMessageWith`
|
||||||
|
* `Utils`:
|
||||||
|
* New `ClassCasts` for
|
||||||
|
* `Message`
|
||||||
|
* **PASSPORT** `EncryptedPassportElement`
|
||||||
|
* **PASSPORT** `PassportElementError`
|
||||||
|
* **PASSPORT** `SecureValue`
|
||||||
|
* Several tools for decryption have been added:
|
||||||
|
* `AESDecryptor` is available for `JVM` platform
|
||||||
|
* Extensions `EncryptedCredentials#decryptWithPKCS8PrivateKey` are available for `JVM`
|
||||||
|
platform
|
||||||
|
* Extensions `EndDataCredentials#decryptData` and `FileCredentials#decryptFile` have been added
|
||||||
|
* Several extensions `createDecryptor`
|
||||||
|
* Several extensions `doInDecryptionContextWithPKCS8Key`
|
||||||
|
* New extension `Flow#passportMessages`
|
||||||
|
* In most of webhook setting up functions/methods now available parameter `mediaGroupsDebounceTimeMillis`
|
||||||
|
* `API`:
|
||||||
|
* **PASSPORT** New extensions `TelegramBot#setPassportDataErrors`
|
||||||
|
|
||||||
## 0.31.0
|
## 0.31.0
|
||||||
|
|
||||||
|
|||||||
@@ -8,15 +8,15 @@ kotlin.incremental.js=true
|
|||||||
kotlin_version=1.4.21
|
kotlin_version=1.4.21
|
||||||
kotlin_coroutines_version=1.4.2
|
kotlin_coroutines_version=1.4.2
|
||||||
kotlin_serialisation_runtime_version=1.0.1
|
kotlin_serialisation_runtime_version=1.0.1
|
||||||
klock_version=2.0.3
|
klock_version=2.0.4
|
||||||
uuid_version=0.2.3
|
uuid_version=0.2.3
|
||||||
ktor_version=1.5.0
|
ktor_version=1.5.1
|
||||||
|
|
||||||
micro_utils_version=0.4.18
|
micro_utils_version=0.4.23
|
||||||
|
|
||||||
javax_activation_version=1.1.1
|
javax_activation_version=1.1.1
|
||||||
|
|
||||||
library_group=dev.inmo
|
library_group=dev.inmo
|
||||||
library_version=0.31.1
|
library_version=0.32.3
|
||||||
|
|
||||||
github_release_plugin_version=2.2.12
|
github_release_plugin_version=2.2.12
|
||||||
|
|||||||
2
gradle/wrapper/gradle-wrapper.properties
vendored
2
gradle/wrapper/gradle-wrapper.properties
vendored
@@ -2,4 +2,4 @@ distributionBase=GRADLE_USER_HOME
|
|||||||
distributionPath=wrapper/dists
|
distributionPath=wrapper/dists
|
||||||
zipStoreBase=GRADLE_USER_HOME
|
zipStoreBase=GRADLE_USER_HOME
|
||||||
zipStorePath=wrapper/dists
|
zipStorePath=wrapper/dists
|
||||||
distributionUrl=https\://services.gradle.org/distributions/gradle-6.8-bin.zip
|
distributionUrl=https\://services.gradle.org/distributions/gradle-6.8.1-bin.zip
|
||||||
|
|||||||
@@ -11,9 +11,6 @@ moments are describing by official [Telegram Bot API](https://core.telegram.org/
|
|||||||
## Compatibility
|
## Compatibility
|
||||||
|
|
||||||
This version compatible with [4th of November 2020 update of TelegramBotAPI (version 5.0)](https://core.telegram.org/bots/api#november-4-2020).
|
This version compatible with [4th of November 2020 update of TelegramBotAPI (version 5.0)](https://core.telegram.org/bots/api#november-4-2020).
|
||||||
There is only one exception of implemented functionality - Telegram Passport API, which was presented in
|
|
||||||
[August 2018 update of TelegramBotAPI](https://core.telegram.org/bots/api-changelog#august-27-2018) update. It will be implemented
|
|
||||||
as soon as possible.
|
|
||||||
|
|
||||||
## How to implement library?
|
## How to implement library?
|
||||||
|
|
||||||
@@ -149,3 +146,18 @@ Here was used `okhttp` realisation of client, but there are several others engin
|
|||||||
available on ktor.io site for [client](https://ktor.io/clients/http-client/engines.html) and [server](https://ktor.io/quickstart/artifacts.html)
|
available on ktor.io site for [client](https://ktor.io/clients/http-client/engines.html) and [server](https://ktor.io/quickstart/artifacts.html)
|
||||||
engines.
|
engines.
|
||||||
|
|
||||||
|
### Passport
|
||||||
|
|
||||||
|
In case you wish to work with `Telegram Passport`, currently there are several useful things, but most part of working
|
||||||
|
with decryption and handling is available only on JVM. Next snippet contains example of data decryption on JVM platform:
|
||||||
|
|
||||||
|
```kotlin
|
||||||
|
passportMessage.passportData.doInDecryptionContextWithPKCS8Key(privateKey) {
|
||||||
|
val passportDataSecureValue = passport ?.data ?: return@doInDecryptionContextWithPKCS8Key
|
||||||
|
val passportData = (passportMessage.passportData.data.firstOrNull { it is CommonPassport } ?: return@doInDecryptionContextWithPKCS8Key) as CommonPassport
|
||||||
|
val decrypted = passportDataSecureValue.decrypt(
|
||||||
|
passportData.data
|
||||||
|
) ?.decodeToString() ?: return@doInDecryptionContextWithPKCS8Key
|
||||||
|
println(decrypted)
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|||||||
@@ -46,8 +46,10 @@ kotlin {
|
|||||||
api "com.soywiz.korlibs.klock:klock:$klock_version"
|
api "com.soywiz.korlibs.klock:klock:$klock_version"
|
||||||
api "com.benasher44:uuid:$uuid_version"
|
api "com.benasher44:uuid:$uuid_version"
|
||||||
|
|
||||||
|
api "dev.inmo:micro_utils.crypto:$micro_utils_version"
|
||||||
api "dev.inmo:micro_utils.coroutines:$micro_utils_version"
|
api "dev.inmo:micro_utils.coroutines:$micro_utils_version"
|
||||||
api "dev.inmo:micro_utils.serialization.base64:$micro_utils_version"
|
api "dev.inmo:micro_utils.serialization.base64:$micro_utils_version"
|
||||||
|
api "dev.inmo:micro_utils.serialization.encapsulator:$micro_utils_version"
|
||||||
|
|
||||||
api "io.ktor:ktor-client-core:$ktor_version"
|
api "io.ktor:ktor-client-core:$ktor_version"
|
||||||
}
|
}
|
||||||
@@ -74,7 +76,6 @@ kotlin {
|
|||||||
implementation kotlin('test-junit')
|
implementation kotlin('test-junit')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
jsTest {
|
jsTest {
|
||||||
dependencies {
|
dependencies {
|
||||||
implementation kotlin('test-junit')
|
implementation kotlin('test-junit')
|
||||||
|
|||||||
@@ -0,0 +1,22 @@
|
|||||||
|
package dev.inmo.tgbotapi.requests
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.requests.abstracts.Request
|
||||||
|
import dev.inmo.tgbotapi.requests.abstracts.SimpleRequest
|
||||||
|
import dev.inmo.tgbotapi.types.*
|
||||||
|
import dev.inmo.tgbotapi.types.passport.PassportElementError
|
||||||
|
import kotlinx.serialization.*
|
||||||
|
import kotlinx.serialization.builtins.serializer
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class SetPassportDataErrors(
|
||||||
|
@SerialName(userIdField)
|
||||||
|
val user: UserId,
|
||||||
|
@SerialName(errorsField)
|
||||||
|
val errors: List<PassportElementError>
|
||||||
|
) : SimpleRequest<Boolean> {
|
||||||
|
override val resultDeserializer: DeserializationStrategy<Boolean>
|
||||||
|
get() = Boolean.serializer()
|
||||||
|
override fun method(): String = "setPassportDataErrors"
|
||||||
|
override val requestSerializer: SerializationStrategy<*>
|
||||||
|
get() = serializer()
|
||||||
|
}
|
||||||
@@ -16,7 +16,7 @@ data class AnswerCallbackQuery(
|
|||||||
val showAlert: Boolean? = null,
|
val showAlert: Boolean? = null,
|
||||||
@SerialName(urlField)
|
@SerialName(urlField)
|
||||||
val url: String? = null,
|
val url: String? = null,
|
||||||
@SerialName(cachedTimeField)
|
@SerialName(cacheTimeField)
|
||||||
val cachedTimeSeconds: Int? = null
|
val cachedTimeSeconds: Int? = null
|
||||||
) : SimpleRequest<Boolean> {
|
) : SimpleRequest<Boolean> {
|
||||||
override fun method(): String = "answerCallbackQuery"
|
override fun method(): String = "answerCallbackQuery"
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ data class AnswerInlineQuery(
|
|||||||
@Serializable(InlineQueryAnswersResultsSerializer::class)
|
@Serializable(InlineQueryAnswersResultsSerializer::class)
|
||||||
@SerialName(resultsField)
|
@SerialName(resultsField)
|
||||||
val results: List<InlineQueryResult> = emptyList(),
|
val results: List<InlineQueryResult> = emptyList(),
|
||||||
@SerialName(cachedTimeField)
|
@SerialName(cacheTimeField)
|
||||||
val cachedTime: Int? = null,
|
val cachedTime: Int? = null,
|
||||||
@SerialName(isPersonalField)
|
@SerialName(isPersonalField)
|
||||||
val isPersonal: Boolean? = null,
|
val isPersonal: Boolean? = null,
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ import dev.inmo.tgbotapi.utils.throwRangeError
|
|||||||
import kotlinx.serialization.*
|
import kotlinx.serialization.*
|
||||||
|
|
||||||
private val commonResultDeserializer = TelegramBotAPIMessageDeserializationStrategyClass<ContentMessage<LocationContent>>()
|
private val commonResultDeserializer = TelegramBotAPIMessageDeserializationStrategyClass<ContentMessage<LocationContent>>()
|
||||||
|
const val editMessageLiveLocationMethod = "editMessageLiveLocation"
|
||||||
|
|
||||||
@Serializable
|
@Serializable
|
||||||
data class EditChatMessageLiveLocation(
|
data class EditChatMessageLiveLocation(
|
||||||
@@ -30,7 +31,7 @@ data class EditChatMessageLiveLocation(
|
|||||||
@SerialName(replyMarkupField)
|
@SerialName(replyMarkupField)
|
||||||
override val replyMarkup: InlineKeyboardMarkup? = null
|
override val replyMarkup: InlineKeyboardMarkup? = null
|
||||||
) : EditChatMessage<LocationContent>, EditReplyMessage, EditLocationMessage {
|
) : EditChatMessage<LocationContent>, EditReplyMessage, EditLocationMessage {
|
||||||
override fun method(): String = "editMessageLiveLocation"
|
override fun method(): String = editMessageLiveLocationMethod
|
||||||
override val resultDeserializer: DeserializationStrategy<ContentMessage<LocationContent>>
|
override val resultDeserializer: DeserializationStrategy<ContentMessage<LocationContent>>
|
||||||
get() = commonResultDeserializer
|
get() = commonResultDeserializer
|
||||||
override val requestSerializer: SerializationStrategy<*>
|
override val requestSerializer: SerializationStrategy<*>
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ data class EditInlineMessageLiveLocation(
|
|||||||
@SerialName(replyMarkupField)
|
@SerialName(replyMarkupField)
|
||||||
override val replyMarkup: InlineKeyboardMarkup? = null
|
override val replyMarkup: InlineKeyboardMarkup? = null
|
||||||
) : EditInlineMessage, EditReplyMessage, EditLocationMessage {
|
) : EditInlineMessage, EditReplyMessage, EditLocationMessage {
|
||||||
override fun method(): String = "editMessageLiveLocation"
|
override fun method(): String = editMessageLiveLocationMethod
|
||||||
override val requestSerializer: SerializationStrategy<*>
|
override val requestSerializer: SerializationStrategy<*>
|
||||||
get() = serializer()
|
get() = serializer()
|
||||||
|
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ import dev.inmo.tgbotapi.types.message.content.LocationContent
|
|||||||
import kotlinx.serialization.*
|
import kotlinx.serialization.*
|
||||||
|
|
||||||
private val commonResultDeserializer = TelegramBotAPIMessageDeserializationStrategyClass<ContentMessage<LocationContent>>()
|
private val commonResultDeserializer = TelegramBotAPIMessageDeserializationStrategyClass<ContentMessage<LocationContent>>()
|
||||||
|
const val stopMessageLiveLocationMethod = "stopMessageLiveLocation"
|
||||||
|
|
||||||
@Serializable
|
@Serializable
|
||||||
data class StopChatMessageLiveLocation(
|
data class StopChatMessageLiveLocation(
|
||||||
@@ -20,7 +21,7 @@ data class StopChatMessageLiveLocation(
|
|||||||
@SerialName(replyMarkupField)
|
@SerialName(replyMarkupField)
|
||||||
override val replyMarkup: InlineKeyboardMarkup? = null
|
override val replyMarkup: InlineKeyboardMarkup? = null
|
||||||
) : EditChatMessage<LocationContent>, EditReplyMessage {
|
) : EditChatMessage<LocationContent>, EditReplyMessage {
|
||||||
override fun method(): String = "stopMessageLiveLocation"
|
override fun method(): String = stopMessageLiveLocationMethod
|
||||||
override val resultDeserializer: DeserializationStrategy<ContentMessage<LocationContent>>
|
override val resultDeserializer: DeserializationStrategy<ContentMessage<LocationContent>>
|
||||||
get() = commonResultDeserializer
|
get() = commonResultDeserializer
|
||||||
override val requestSerializer: SerializationStrategy<*>
|
override val requestSerializer: SerializationStrategy<*>
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ data class StopInlineMessageLiveLocation(
|
|||||||
@SerialName(replyMarkupField)
|
@SerialName(replyMarkupField)
|
||||||
override val replyMarkup: InlineKeyboardMarkup? = null
|
override val replyMarkup: InlineKeyboardMarkup? = null
|
||||||
) : EditInlineMessage, EditReplyMessage {
|
) : EditInlineMessage, EditReplyMessage {
|
||||||
override fun method(): String = "stopMessageLiveLocation"
|
override fun method(): String = stopMessageLiveLocationMethod
|
||||||
override val requestSerializer: SerializationStrategy<*>
|
override val requestSerializer: SerializationStrategy<*>
|
||||||
get() = serializer()
|
get() = serializer()
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ package dev.inmo.tgbotapi.requests.edit.text
|
|||||||
|
|
||||||
import dev.inmo.tgbotapi.CommonAbstracts.*
|
import dev.inmo.tgbotapi.CommonAbstracts.*
|
||||||
import dev.inmo.tgbotapi.requests.edit.abstracts.*
|
import dev.inmo.tgbotapi.requests.edit.abstracts.*
|
||||||
import dev.inmo.tgbotapi.requests.edit.media.editMessageMediaMethod
|
|
||||||
import dev.inmo.tgbotapi.types.*
|
import dev.inmo.tgbotapi.types.*
|
||||||
import dev.inmo.tgbotapi.types.MessageEntity.*
|
import dev.inmo.tgbotapi.types.MessageEntity.*
|
||||||
import dev.inmo.tgbotapi.types.ParseMode.ParseMode
|
import dev.inmo.tgbotapi.types.ParseMode.ParseMode
|
||||||
@@ -55,10 +54,10 @@ data class EditInlineMessageText internal constructor(
|
|||||||
override val replyMarkup: InlineKeyboardMarkup? = null
|
override val replyMarkup: InlineKeyboardMarkup? = null
|
||||||
) : EditInlineMessage, EditTextChatMessage, EditReplyMessage, EditDisableWebPagePreviewMessage {
|
) : EditInlineMessage, EditTextChatMessage, EditReplyMessage, EditDisableWebPagePreviewMessage {
|
||||||
override val entities: List<TextSource>? by lazy {
|
override val entities: List<TextSource>? by lazy {
|
||||||
rawEntities ?.asTextParts(text ?: return@lazy null) ?.justTextSources()
|
rawEntities ?.asTextParts(text) ?.justTextSources()
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun method(): String = editMessageMediaMethod
|
override fun method(): String = editMessageTextMethod
|
||||||
override val requestSerializer: SerializationStrategy<*>
|
override val requestSerializer: SerializationStrategy<*>
|
||||||
get() = serializer()
|
get() = serializer()
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,6 +8,10 @@ import dev.inmo.tgbotapi.types.*
|
|||||||
import dev.inmo.tgbotapi.types.InputMedia.*
|
import dev.inmo.tgbotapi.types.InputMedia.*
|
||||||
import dev.inmo.tgbotapi.types.message.abstracts.MediaGroupMessage
|
import dev.inmo.tgbotapi.types.message.abstracts.MediaGroupMessage
|
||||||
import dev.inmo.tgbotapi.types.message.abstracts.TelegramBotAPIMessageDeserializeOnlySerializerClass
|
import dev.inmo.tgbotapi.types.message.abstracts.TelegramBotAPIMessageDeserializeOnlySerializerClass
|
||||||
|
import dev.inmo.tgbotapi.types.message.content.abstracts.MediaGroupContent
|
||||||
|
import dev.inmo.tgbotapi.types.message.content.abstracts.VisualMediaGroupContent
|
||||||
|
import dev.inmo.tgbotapi.types.message.content.media.AudioContent
|
||||||
|
import dev.inmo.tgbotapi.types.message.content.media.DocumentContent
|
||||||
import dev.inmo.tgbotapi.utils.*
|
import dev.inmo.tgbotapi.utils.*
|
||||||
import kotlinx.serialization.*
|
import kotlinx.serialization.*
|
||||||
import kotlinx.serialization.builtins.ListSerializer
|
import kotlinx.serialization.builtins.ListSerializer
|
||||||
@@ -17,13 +21,13 @@ const val rawSendingMediaGroupsWarning = "Media groups contains restrictions rel
|
|||||||
" types. Currently it is possible to combine photo + video OR audio OR documents"
|
" types. Currently it is possible to combine photo + video OR audio OR documents"
|
||||||
|
|
||||||
@RiskFeature(rawSendingMediaGroupsWarning)
|
@RiskFeature(rawSendingMediaGroupsWarning)
|
||||||
fun SendMediaGroup(
|
fun <T : MediaGroupContent> SendMediaGroup(
|
||||||
chatId: ChatIdentifier,
|
chatId: ChatIdentifier,
|
||||||
media: List<MediaGroupMemberInputMedia>,
|
media: List<MediaGroupMemberInputMedia>,
|
||||||
disableNotification: Boolean = false,
|
disableNotification: Boolean = false,
|
||||||
replyToMessageId: MessageIdentifier? = null,
|
replyToMessageId: MessageIdentifier? = null,
|
||||||
allowSendingWithoutReply: Boolean? = null
|
allowSendingWithoutReply: Boolean? = null
|
||||||
): Request<List<MediaGroupMessage>> {
|
): Request<List<MediaGroupMessage<T>>> {
|
||||||
if (media.size !in mediaCountInMediaGroup) {
|
if (media.size !in mediaCountInMediaGroup) {
|
||||||
throwRangeError("Count of members in media group", mediaCountInMediaGroup, media.size)
|
throwRangeError("Count of members in media group", mediaCountInMediaGroup, media.size)
|
||||||
}
|
}
|
||||||
@@ -47,14 +51,14 @@ fun SendMediaGroup(
|
|||||||
allowSendingWithoutReply
|
allowSendingWithoutReply
|
||||||
)
|
)
|
||||||
|
|
||||||
return if (files.isEmpty()) {
|
return (if (files.isEmpty()) {
|
||||||
data
|
data
|
||||||
} else {
|
} else {
|
||||||
MultipartRequestImpl(
|
MultipartRequestImpl(
|
||||||
data,
|
data,
|
||||||
SendMediaGroupFiles(files)
|
SendMediaGroupFiles(files)
|
||||||
)
|
)
|
||||||
}
|
}) as Request<List<MediaGroupMessage<T>>>
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -69,7 +73,7 @@ inline fun SendPlaylist(
|
|||||||
disableNotification: Boolean = false,
|
disableNotification: Boolean = false,
|
||||||
replyToMessageId: MessageIdentifier? = null,
|
replyToMessageId: MessageIdentifier? = null,
|
||||||
allowSendingWithoutReply: Boolean? = null
|
allowSendingWithoutReply: Boolean? = null
|
||||||
) = SendMediaGroup(chatId, media, disableNotification, replyToMessageId, allowSendingWithoutReply)
|
) = SendMediaGroup<AudioContent>(chatId, media, disableNotification, replyToMessageId, allowSendingWithoutReply)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Use this method to be sure that you are correctly sending documents media group
|
* Use this method to be sure that you are correctly sending documents media group
|
||||||
@@ -83,7 +87,7 @@ inline fun SendDocumentsGroup(
|
|||||||
disableNotification: Boolean = false,
|
disableNotification: Boolean = false,
|
||||||
replyToMessageId: MessageIdentifier? = null,
|
replyToMessageId: MessageIdentifier? = null,
|
||||||
allowSendingWithoutReply: Boolean? = null
|
allowSendingWithoutReply: Boolean? = null
|
||||||
) = SendMediaGroup(chatId, media, disableNotification, replyToMessageId, allowSendingWithoutReply)
|
) = SendMediaGroup<DocumentContent>(chatId, media, disableNotification, replyToMessageId, allowSendingWithoutReply)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Use this method to be sure that you are correctly sending visual media group
|
* Use this method to be sure that you are correctly sending visual media group
|
||||||
@@ -98,9 +102,9 @@ inline fun SendVisualMediaGroup(
|
|||||||
disableNotification: Boolean = false,
|
disableNotification: Boolean = false,
|
||||||
replyToMessageId: MessageIdentifier? = null,
|
replyToMessageId: MessageIdentifier? = null,
|
||||||
allowSendingWithoutReply: Boolean? = null
|
allowSendingWithoutReply: Boolean? = null
|
||||||
) = SendMediaGroup(chatId, media, disableNotification, replyToMessageId, allowSendingWithoutReply)
|
) = SendMediaGroup<VisualMediaGroupContent>(chatId, media, disableNotification, replyToMessageId, allowSendingWithoutReply)
|
||||||
|
|
||||||
private val messagesListSerializer: KSerializer<List<MediaGroupMessage>>
|
private val messagesListSerializer: KSerializer<List<MediaGroupMessage<MediaGroupContent>>>
|
||||||
= ListSerializer(TelegramBotAPIMessageDeserializeOnlySerializerClass())
|
= ListSerializer(TelegramBotAPIMessageDeserializeOnlySerializerClass())
|
||||||
|
|
||||||
@Serializable
|
@Serializable
|
||||||
@@ -114,7 +118,7 @@ data class SendMediaGroupData internal constructor(
|
|||||||
override val replyToMessageId: MessageIdentifier? = null,
|
override val replyToMessageId: MessageIdentifier? = null,
|
||||||
@SerialName(allowSendingWithoutReplyField)
|
@SerialName(allowSendingWithoutReplyField)
|
||||||
override val allowSendingWithoutReply: Boolean? = null
|
override val allowSendingWithoutReply: Boolean? = null
|
||||||
) : DataRequest<List<MediaGroupMessage>>, SendMessageRequest<List<MediaGroupMessage>> {
|
) : DataRequest<List<MediaGroupMessage<MediaGroupContent>>>, SendMessageRequest<List<MediaGroupMessage<MediaGroupContent>>> {
|
||||||
@SerialName(mediaField)
|
@SerialName(mediaField)
|
||||||
private val convertedMedia: String
|
private val convertedMedia: String
|
||||||
get() = buildJsonArray {
|
get() = buildJsonArray {
|
||||||
@@ -127,7 +131,7 @@ data class SendMediaGroupData internal constructor(
|
|||||||
override fun method(): String = "sendMediaGroup"
|
override fun method(): String = "sendMediaGroup"
|
||||||
override val requestSerializer: SerializationStrategy<*>
|
override val requestSerializer: SerializationStrategy<*>
|
||||||
get() = serializer()
|
get() = serializer()
|
||||||
override val resultDeserializer: DeserializationStrategy<List<MediaGroupMessage>>
|
override val resultDeserializer: DeserializationStrategy<List<MediaGroupMessage<MediaGroupContent>>>
|
||||||
get() = messagesListSerializer
|
get() = messagesListSerializer
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -123,7 +123,7 @@ const val callbackQueryIdField = "callback_query_id"
|
|||||||
const val inlineQueryIdField = "inline_query_id"
|
const val inlineQueryIdField = "inline_query_id"
|
||||||
const val inlineKeyboardField = "inline_keyboard"
|
const val inlineKeyboardField = "inline_keyboard"
|
||||||
const val showAlertField = "show_alert"
|
const val showAlertField = "show_alert"
|
||||||
const val cachedTimeField = "cached_time"
|
const val cacheTimeField = "cache_time"
|
||||||
const val foursquareIdField = "foursquare_id"
|
const val foursquareIdField = "foursquare_id"
|
||||||
const val foursquareTypeField = "foursquare_type"
|
const val foursquareTypeField = "foursquare_type"
|
||||||
const val googlePlaceIdField = "google_place_id"
|
const val googlePlaceIdField = "google_place_id"
|
||||||
@@ -362,8 +362,34 @@ const val dataField = "data"
|
|||||||
const val credentialsField = "credentials"
|
const val credentialsField = "credentials"
|
||||||
const val hashField = "hash"
|
const val hashField = "hash"
|
||||||
const val translationField = "translation"
|
const val translationField = "translation"
|
||||||
|
const val translationFileField = "translation_file"
|
||||||
|
const val fileField = "file"
|
||||||
const val filesField = "files"
|
const val filesField = "files"
|
||||||
|
const val translationFilesField = "translation_files"
|
||||||
const val frontSideField = "front_side"
|
const val frontSideField = "front_side"
|
||||||
const val reverseSideField = "reverse_side"
|
const val reverseSideField = "reverse_side"
|
||||||
const val selfieField = "selfie"
|
const val selfieField = "selfie"
|
||||||
const val secretField = "secret"
|
const val secretField = "secret"
|
||||||
|
|
||||||
|
const val errorsField = "errors"
|
||||||
|
const val sourceField = "source"
|
||||||
|
const val fieldNameField = "field_name"
|
||||||
|
const val dataHashField = "data_hash"
|
||||||
|
const val fileHashField = "file_hash"
|
||||||
|
const val fileHashesField = "file_hashes"
|
||||||
|
const val messageField = "message"
|
||||||
|
const val unspecifiedField = "unspecified"
|
||||||
|
|
||||||
|
const val secureDataField = "secure_data"
|
||||||
|
const val nonceField = "nonce"
|
||||||
|
|
||||||
|
const val personalDetailsField = "personal_details"
|
||||||
|
const val passportField = "passport"
|
||||||
|
const val internalPassportField = "internal_passport"
|
||||||
|
const val driverLicenseField = "driver_license"
|
||||||
|
const val identityCardField = "identity_card"
|
||||||
|
const val utilityBillField = "utility_bill"
|
||||||
|
const val bankStatementField = "bank_statement"
|
||||||
|
const val rentalAgreementField = "rental_agreement"
|
||||||
|
const val passportRegistrationField = "passport_registration"
|
||||||
|
const val temporaryRegistrationField = "temporary_registration"
|
||||||
|
|||||||
@@ -9,14 +9,14 @@ import dev.inmo.tgbotapi.types.message.abstracts.MediaGroupMessage
|
|||||||
import dev.inmo.tgbotapi.types.message.abstracts.Message
|
import dev.inmo.tgbotapi.types.message.abstracts.Message
|
||||||
import dev.inmo.tgbotapi.types.message.content.abstracts.MediaGroupContent
|
import dev.inmo.tgbotapi.types.message.content.abstracts.MediaGroupContent
|
||||||
|
|
||||||
data class ChannelMediaGroupMessage(
|
data class ChannelMediaGroupMessage<T : MediaGroupContent>(
|
||||||
override val messageId: MessageIdentifier,
|
override val messageId: MessageIdentifier,
|
||||||
override val chat: Chat,
|
override val chat: Chat,
|
||||||
override val date: DateTime,
|
override val date: DateTime,
|
||||||
override val mediaGroupId: MediaGroupIdentifier,
|
override val mediaGroupId: MediaGroupIdentifier,
|
||||||
override val content: MediaGroupContent,
|
override val content: T,
|
||||||
override val editDate: DateTime?,
|
override val editDate: DateTime?,
|
||||||
override val forwardInfo: ForwardInfo?,
|
override val forwardInfo: ForwardInfo?,
|
||||||
override val replyTo: Message?,
|
override val replyTo: Message?,
|
||||||
override val replyMarkup: InlineKeyboardMarkup?
|
override val replyMarkup: InlineKeyboardMarkup?
|
||||||
) : MediaGroupMessage
|
) : MediaGroupMessage<T>
|
||||||
|
|||||||
@@ -7,15 +7,15 @@ import dev.inmo.tgbotapi.types.chat.abstracts.Chat
|
|||||||
import dev.inmo.tgbotapi.types.message.abstracts.*
|
import dev.inmo.tgbotapi.types.message.abstracts.*
|
||||||
import dev.inmo.tgbotapi.types.message.content.abstracts.MediaGroupContent
|
import dev.inmo.tgbotapi.types.message.content.abstracts.MediaGroupContent
|
||||||
|
|
||||||
data class CommonMediaGroupMessage(
|
data class CommonMediaGroupMessage<T : MediaGroupContent>(
|
||||||
override val messageId: MessageIdentifier,
|
override val messageId: MessageIdentifier,
|
||||||
override val user: User,
|
override val user: User,
|
||||||
override val chat: Chat,
|
override val chat: Chat,
|
||||||
override val date: DateTime,
|
override val date: DateTime,
|
||||||
override val mediaGroupId: MediaGroupIdentifier,
|
override val mediaGroupId: MediaGroupIdentifier,
|
||||||
override val content: MediaGroupContent,
|
override val content: T,
|
||||||
override val editDate: DateTime?,
|
override val editDate: DateTime?,
|
||||||
override val forwardInfo: ForwardInfo?,
|
override val forwardInfo: ForwardInfo?,
|
||||||
override val replyTo: Message?,
|
override val replyTo: Message?,
|
||||||
override val replyMarkup: InlineKeyboardMarkup?
|
override val replyMarkup: InlineKeyboardMarkup?
|
||||||
) : MediaGroupMessage, FromUserMessage
|
) : MediaGroupMessage<T>, FromUserMessage
|
||||||
|
|||||||
@@ -3,6 +3,6 @@ package dev.inmo.tgbotapi.types.message.abstracts
|
|||||||
import dev.inmo.tgbotapi.types.MediaGroupIdentifier
|
import dev.inmo.tgbotapi.types.MediaGroupIdentifier
|
||||||
import dev.inmo.tgbotapi.types.message.content.abstracts.MediaGroupContent
|
import dev.inmo.tgbotapi.types.message.content.abstracts.MediaGroupContent
|
||||||
|
|
||||||
interface MediaGroupMessage : CommonMessage<MediaGroupContent> {
|
interface MediaGroupMessage<T : MediaGroupContent> : CommonMessage<T> {
|
||||||
val mediaGroupId: MediaGroupIdentifier
|
val mediaGroupId: MediaGroupIdentifier
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,12 +1,18 @@
|
|||||||
package dev.inmo.tgbotapi.types.message.content.abstracts
|
package dev.inmo.tgbotapi.types.message.content.abstracts
|
||||||
|
|
||||||
import dev.inmo.tgbotapi.CommonAbstracts.CaptionedInput
|
import dev.inmo.tgbotapi.CommonAbstracts.CaptionedInput
|
||||||
import dev.inmo.tgbotapi.types.InputMedia.MediaGroupMemberInputMedia
|
import dev.inmo.tgbotapi.types.InputMedia.*
|
||||||
|
|
||||||
interface MediaGroupContent : MediaContent, CaptionedInput {
|
interface MediaGroupContent : MediaContent, CaptionedInput {
|
||||||
fun toMediaGroupMemberInputMedia(): MediaGroupMemberInputMedia
|
fun toMediaGroupMemberInputMedia(): MediaGroupMemberInputMedia
|
||||||
}
|
}
|
||||||
|
|
||||||
interface VisualMediaGroupContent : MediaGroupContent
|
interface VisualMediaGroupContent : MediaGroupContent {
|
||||||
interface AudioMediaGroupContent : MediaGroupContent
|
override fun toMediaGroupMemberInputMedia(): VisualMediaGroupMemberInputMedia
|
||||||
interface DocumentMediaGroupContent : MediaGroupContent
|
}
|
||||||
|
interface AudioMediaGroupContent : MediaGroupContent {
|
||||||
|
override fun toMediaGroupMemberInputMedia(): AudioMediaGroupMemberInputMedia
|
||||||
|
}
|
||||||
|
interface DocumentMediaGroupContent : MediaGroupContent {
|
||||||
|
override fun toMediaGroupMemberInputMedia(): DocumentMediaGroupMemberInputMedia
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,23 +0,0 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport
|
|
||||||
|
|
||||||
import dev.inmo.micro_utils.serialization.base64.Base64StringSerializer
|
|
||||||
import dev.inmo.tgbotapi.types.*
|
|
||||||
import kotlinx.serialization.SerialName
|
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
|
|
||||||
typealias EncryptedAndBase64EncodedData = String
|
|
||||||
typealias EncryptedByBotPublicKeyData = String
|
|
||||||
typealias EncryptedData = String
|
|
||||||
|
|
||||||
@Serializable
|
|
||||||
data class EncryptedCredentials(
|
|
||||||
@SerialName(dataField)
|
|
||||||
@Serializable(Base64StringSerializer::class)
|
|
||||||
val data: EncryptedData,
|
|
||||||
@SerialName(hashField)
|
|
||||||
@Serializable(Base64StringSerializer::class)
|
|
||||||
val hash: String,
|
|
||||||
@SerialName(secretField)
|
|
||||||
@Serializable(Base64StringSerializer::class)
|
|
||||||
val secret: EncryptedByBotPublicKeyData
|
|
||||||
)
|
|
||||||
@@ -2,7 +2,8 @@ package dev.inmo.tgbotapi.types.passport
|
|||||||
|
|
||||||
import dev.inmo.tgbotapi.types.credentialsField
|
import dev.inmo.tgbotapi.types.credentialsField
|
||||||
import dev.inmo.tgbotapi.types.dataField
|
import dev.inmo.tgbotapi.types.dataField
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts.EncryptedPassportElement
|
import dev.inmo.tgbotapi.types.passport.credentials.EncryptedCredentials
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.EncryptedPassportElement
|
||||||
import kotlinx.serialization.SerialName
|
import kotlinx.serialization.SerialName
|
||||||
import kotlinx.serialization.Serializable
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,266 @@
|
|||||||
|
@file:Suppress("unused", "EXPERIMENTAL_API_USAGE")
|
||||||
|
|
||||||
|
package dev.inmo.tgbotapi.types.passport
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.crypto.MD5
|
||||||
|
import dev.inmo.micro_utils.crypto.md5
|
||||||
|
import dev.inmo.micro_utils.serialization.base64.Base64BytesToFromStringSerializer
|
||||||
|
import dev.inmo.tgbotapi.types.*
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.*
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.type
|
||||||
|
import dev.inmo.tgbotapi.utils.nonstrictJsonFormat
|
||||||
|
import kotlinx.serialization.*
|
||||||
|
import kotlinx.serialization.descriptors.SerialDescriptor
|
||||||
|
import kotlinx.serialization.encoding.Decoder
|
||||||
|
import kotlinx.serialization.encoding.Encoder
|
||||||
|
import kotlinx.serialization.json.*
|
||||||
|
|
||||||
|
val ByteArray.passportFileHash: MD5
|
||||||
|
get() = md5()
|
||||||
|
|
||||||
|
@Serializable(PassportElementErrorSerializer::class)
|
||||||
|
sealed class PassportElementError {
|
||||||
|
abstract val source: String
|
||||||
|
abstract val type: String
|
||||||
|
abstract val message: String
|
||||||
|
}
|
||||||
|
|
||||||
|
data class UnknownPassportElementError(
|
||||||
|
val raw: JsonObject
|
||||||
|
) : PassportElementError() {
|
||||||
|
override val source: String = raw[sourceField] ?.jsonPrimitive ?.contentOrNull ?: ""
|
||||||
|
override val type: String = raw[typeField] ?.jsonPrimitive ?.contentOrNull ?: ""
|
||||||
|
override val message: String = raw[messageField] ?.jsonPrimitive ?.contentOrNull ?: ""
|
||||||
|
}
|
||||||
|
|
||||||
|
object PassportElementErrorSerializer : KSerializer<PassportElementError> {
|
||||||
|
private val jsonObjectSerializer = JsonObject.serializer()
|
||||||
|
override val descriptor: SerialDescriptor
|
||||||
|
get() = jsonObjectSerializer.descriptor
|
||||||
|
override fun deserialize(decoder: Decoder): PassportElementError {
|
||||||
|
val json = jsonObjectSerializer.deserialize(decoder)
|
||||||
|
return when (json[sourceField] ?.jsonPrimitive ?.content) {
|
||||||
|
"dataField" -> nonstrictJsonFormat.decodeFromJsonElement(PassportElementErrorDataField.serializer(), json)
|
||||||
|
"frontSideField" -> nonstrictJsonFormat.decodeFromJsonElement(PassportElementErrorFrontSide.serializer(), json)
|
||||||
|
"reverseSideField" -> nonstrictJsonFormat.decodeFromJsonElement(PassportElementErrorReverseSide.serializer(), json)
|
||||||
|
"selfieField" -> nonstrictJsonFormat.decodeFromJsonElement(PassportElementErrorSelfie.serializer(), json)
|
||||||
|
"fileField" -> nonstrictJsonFormat.decodeFromJsonElement(PassportElementFileError.serializer(), json)
|
||||||
|
"filesField" -> nonstrictJsonFormat.decodeFromJsonElement(PassportElementFilesError.serializer(), json)
|
||||||
|
"translationFileField" -> nonstrictJsonFormat.decodeFromJsonElement(PassportElementErrorTranslationFile.serializer(), json)
|
||||||
|
"translationFilesField" -> nonstrictJsonFormat.decodeFromJsonElement(PassportElementErrorTranslationFiles.serializer(), json)
|
||||||
|
"unspecifiedField" -> nonstrictJsonFormat.decodeFromJsonElement(PassportElementErrorUnspecified.serializer(), json)
|
||||||
|
else -> UnknownPassportElementError(json)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
override fun serialize(encoder: Encoder, value: PassportElementError) {
|
||||||
|
val neverMindAboutThisVariable = when (value) {
|
||||||
|
is PassportElementErrorFrontSide -> PassportElementErrorFrontSide.serializer().serialize(encoder, value)
|
||||||
|
is PassportElementErrorReverseSide -> PassportElementErrorReverseSide.serializer().serialize(encoder, value)
|
||||||
|
is PassportElementErrorSelfie -> PassportElementErrorSelfie.serializer().serialize(encoder, value)
|
||||||
|
is PassportElementErrorFile -> PassportElementErrorFile.serializer().serialize(encoder, value)
|
||||||
|
is PassportElementErrorTranslationFile -> PassportElementErrorTranslationFile.serializer().serialize(encoder, value)
|
||||||
|
is PassportElementErrorUnspecified -> PassportElementErrorUnspecified.serializer().serialize(encoder, value)
|
||||||
|
is PassportElementErrorDataField -> PassportElementErrorDataField.serializer().serialize(encoder, value)
|
||||||
|
is PassportElementErrorFiles -> PassportElementErrorFiles.serializer().serialize(encoder, value)
|
||||||
|
is PassportElementErrorTranslationFiles -> PassportElementErrorTranslationFiles.serializer().serialize(encoder, value)
|
||||||
|
is UnknownPassportElementError -> jsonObjectSerializer.serialize(encoder, value.raw)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
sealed class PassportSingleElementError : PassportElementError() {
|
||||||
|
abstract val elementHash: PassportElementHash
|
||||||
|
}
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
sealed class PassportMultipleElementsError : PassportElementError() {
|
||||||
|
abstract val elementsHashes: List<PassportElementHash>
|
||||||
|
}
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
sealed class PassportElementFileError : PassportSingleElementError()
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
sealed class PassportElementFilesError : PassportMultipleElementsError()
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class PassportElementErrorDataField(
|
||||||
|
@SerialName(typeField)
|
||||||
|
override val type: String,
|
||||||
|
@SerialName(fieldNameField)
|
||||||
|
val fieldName: String,
|
||||||
|
@SerialName(dataHashField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val elementHash: PassportElementHash,
|
||||||
|
@SerialName(messageField)
|
||||||
|
override val message: String
|
||||||
|
) : PassportSingleElementError() {
|
||||||
|
@SerialName(sourceField)
|
||||||
|
@Required
|
||||||
|
override val source: String = dataField
|
||||||
|
}
|
||||||
|
fun EncryptedPassportElementWithData.createDataError(field: String, message: String) = PassportElementErrorDataField(
|
||||||
|
type,
|
||||||
|
field,
|
||||||
|
hash,
|
||||||
|
message
|
||||||
|
)
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class PassportElementErrorFrontSide(
|
||||||
|
@SerialName(typeField)
|
||||||
|
override val type: String,
|
||||||
|
@SerialName(fileHashField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val elementHash: PassportElementHash,
|
||||||
|
@SerialName(messageField)
|
||||||
|
override val message: String
|
||||||
|
) : PassportElementFileError() {
|
||||||
|
@SerialName(sourceField)
|
||||||
|
@Required
|
||||||
|
override val source: String = frontSideField
|
||||||
|
}
|
||||||
|
fun EncryptedPassportElementWithFrontSide.createFrontSideError(message: String, unencryptedFileHash: PassportElementHash) = PassportElementErrorFrontSide(
|
||||||
|
type,
|
||||||
|
unencryptedFileHash,
|
||||||
|
message
|
||||||
|
)
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class PassportElementErrorReverseSide(
|
||||||
|
@SerialName(typeField)
|
||||||
|
override val type: String,
|
||||||
|
@SerialName(fileHashField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val elementHash: PassportElementHash,
|
||||||
|
@SerialName(messageField)
|
||||||
|
override val message: String
|
||||||
|
) : PassportElementFileError() {
|
||||||
|
@SerialName(sourceField)
|
||||||
|
@Required
|
||||||
|
override val source: String = reverseSideField
|
||||||
|
}
|
||||||
|
fun EncryptedPassportElementWithReverseSide.createReverseSideError(message: String, unencryptedFileHash: PassportElementHash) = PassportElementErrorReverseSide(
|
||||||
|
type,
|
||||||
|
unencryptedFileHash,
|
||||||
|
message
|
||||||
|
)
|
||||||
|
@Serializable
|
||||||
|
data class PassportElementErrorSelfie(
|
||||||
|
@SerialName(typeField)
|
||||||
|
override val type: String,
|
||||||
|
@SerialName(fileHashField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val elementHash: PassportElementHash,
|
||||||
|
@SerialName(messageField)
|
||||||
|
override val message: String
|
||||||
|
) : PassportElementFileError() {
|
||||||
|
@SerialName(sourceField)
|
||||||
|
@Required
|
||||||
|
override val source: String = selfieField
|
||||||
|
}
|
||||||
|
fun EncryptedPassportElementWithSelfie.createSelfieError(message: String, unencryptedFileHash: PassportElementHash) = PassportElementErrorSelfie(
|
||||||
|
type,
|
||||||
|
unencryptedFileHash,
|
||||||
|
message
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class PassportElementErrorFile(
|
||||||
|
@SerialName(typeField)
|
||||||
|
override val type: String,
|
||||||
|
@SerialName(fileHashField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val elementHash: PassportElementHash,
|
||||||
|
@SerialName(messageField)
|
||||||
|
override val message: String
|
||||||
|
) : PassportElementFileError() {
|
||||||
|
@SerialName(sourceField)
|
||||||
|
@Required
|
||||||
|
override val source: String = fileField
|
||||||
|
}
|
||||||
|
fun EncryptedPassportElementWithFilesCollection.createFileError(message: String, unencryptedFileHash: PassportElementHash) = PassportElementErrorFile(
|
||||||
|
type,
|
||||||
|
unencryptedFileHash,
|
||||||
|
message
|
||||||
|
)
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class PassportElementErrorFiles(
|
||||||
|
@SerialName(typeField)
|
||||||
|
override val type: String,
|
||||||
|
@SerialName(fileHashesField)
|
||||||
|
override val elementsHashes: List<@Serializable(Base64BytesToFromStringSerializer::class) PassportElementHash>,
|
||||||
|
@SerialName(messageField)
|
||||||
|
override val message: String
|
||||||
|
) : PassportElementFilesError() {
|
||||||
|
@SerialName(sourceField)
|
||||||
|
@Required
|
||||||
|
override val source: String = filesField
|
||||||
|
}
|
||||||
|
fun EncryptedPassportElementWithFilesCollection.createFilesError(message: String, unencryptedFileHashes: List<PassportElementHash>) = PassportElementErrorFiles(
|
||||||
|
type,
|
||||||
|
unencryptedFileHashes,
|
||||||
|
message
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class PassportElementErrorTranslationFile(
|
||||||
|
@SerialName(typeField)
|
||||||
|
override val type: String,
|
||||||
|
@SerialName(fileHashField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val elementHash: PassportElementHash,
|
||||||
|
@SerialName(messageField)
|
||||||
|
override val message: String
|
||||||
|
) : PassportElementFileError() {
|
||||||
|
@SerialName(sourceField)
|
||||||
|
@Required
|
||||||
|
override val source: String = translationFileField
|
||||||
|
}
|
||||||
|
fun EncryptedPassportElementTranslatable.createFileError(message: String, unencryptedFileHash: PassportElementHash) = PassportElementErrorTranslationFile(
|
||||||
|
type,
|
||||||
|
unencryptedFileHash,
|
||||||
|
message
|
||||||
|
)
|
||||||
|
@Serializable
|
||||||
|
data class PassportElementErrorTranslationFiles(
|
||||||
|
@SerialName(typeField)
|
||||||
|
override val type: String,
|
||||||
|
@SerialName(fileHashesField)
|
||||||
|
override val elementsHashes: List<@Serializable(Base64BytesToFromStringSerializer::class) PassportElementHash>,
|
||||||
|
@SerialName(messageField)
|
||||||
|
override val message: String
|
||||||
|
) : PassportElementFilesError() {
|
||||||
|
@SerialName(sourceField)
|
||||||
|
@Required
|
||||||
|
override val source: String = translationFilesField
|
||||||
|
}
|
||||||
|
fun EncryptedPassportElementTranslatable.createFilesError(message: String, unencryptedFileHashes: List<PassportElementHash>) = PassportElementErrorTranslationFiles(
|
||||||
|
type,
|
||||||
|
unencryptedFileHashes,
|
||||||
|
message
|
||||||
|
)
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class PassportElementErrorUnspecified(
|
||||||
|
@SerialName(typeField)
|
||||||
|
override val type: String,
|
||||||
|
@SerialName(fileHashField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val elementHash: PassportElementHash,
|
||||||
|
@SerialName(messageField)
|
||||||
|
override val message: String
|
||||||
|
) : PassportElementFileError() {
|
||||||
|
@SerialName(sourceField)
|
||||||
|
@Required
|
||||||
|
override val source: String = unspecifiedField
|
||||||
|
}
|
||||||
|
fun EncryptedPassportElement.createUnspecifiedError(message: String, elementHash: PassportElementHash) = PassportElementErrorUnspecified(
|
||||||
|
type,
|
||||||
|
elementHash,
|
||||||
|
message
|
||||||
|
)
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.credentials
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.nonceField
|
||||||
|
import dev.inmo.tgbotapi.types.passport.decrypted.SecureData
|
||||||
|
import dev.inmo.tgbotapi.types.secureDataField
|
||||||
|
import kotlinx.serialization.SerialName
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class DecryptedCredentials(
|
||||||
|
@SerialName(secureDataField)
|
||||||
|
val secureData: SecureData,
|
||||||
|
@SerialName(nonceField)
|
||||||
|
val nonce: String
|
||||||
|
)
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.credentials
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.crypto.SourceBytes
|
||||||
|
import dev.inmo.micro_utils.serialization.base64.Base64BytesToFromStringSerializer
|
||||||
|
import dev.inmo.tgbotapi.types.*
|
||||||
|
import kotlinx.serialization.SerialName
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
typealias EncryptedByBotPublicKeyData = SourceBytes
|
||||||
|
typealias EncryptedData = SourceBytes
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class EncryptedCredentials(
|
||||||
|
@SerialName(dataField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
val data: EncryptedData,
|
||||||
|
@SerialName(hashField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
val hash: SourceBytes,
|
||||||
|
@SerialName(secretField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
val secret: EncryptedByBotPublicKeyData
|
||||||
|
)
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.credentials
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.crypto.SourceBytes
|
||||||
|
import dev.inmo.micro_utils.serialization.base64.Base64BytesToFromStringSerializer
|
||||||
|
import dev.inmo.tgbotapi.types.*
|
||||||
|
import kotlinx.serialization.SerialName
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
sealed class EndDataCredentials {
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
abstract val hash: SourceBytes
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
abstract val secret: SourceBytes
|
||||||
|
}
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class DataCredentials(
|
||||||
|
@SerialName(dataHashField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val hash: SourceBytes,
|
||||||
|
@SerialName(secretField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val secret: SourceBytes
|
||||||
|
) : EndDataCredentials()
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class FileCredentials(
|
||||||
|
@SerialName(fileHashField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val hash: SourceBytes,
|
||||||
|
@SerialName(secretField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val secret: SourceBytes
|
||||||
|
) : EndDataCredentials()
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.decrypted
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.dataField
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.DataCredentials
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.EndDataCredentials
|
||||||
|
import dev.inmo.tgbotapi.types.passport.decrypted.abstracts.SecureValueWithData
|
||||||
|
import kotlinx.serialization.SerialName
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class AddressSecureValue(
|
||||||
|
@SerialName(dataField)
|
||||||
|
override val data: DataCredentials
|
||||||
|
) : SecureValueWithData {
|
||||||
|
override val credentials: List<EndDataCredentials> = listOf(data)
|
||||||
|
}
|
||||||
@@ -0,0 +1,41 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.decrypted
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.*
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.*
|
||||||
|
import dev.inmo.tgbotapi.types.passport.decrypted.abstracts.*
|
||||||
|
import kotlinx.serialization.SerialName
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
sealed class IdentityWithReverseSideSecureValue : SecureValueIdentity, SecureValueWithData, SecureValueWithTranslations, SecureValueWithReverseSide {
|
||||||
|
override val credentials: List<EndDataCredentials>
|
||||||
|
get() = listOfNotNull(data, frontSide, reverseSide, selfie) + translation
|
||||||
|
}
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class DriverLicenseSecureValue(
|
||||||
|
@SerialName(dataField)
|
||||||
|
override val data: DataCredentials? = null,
|
||||||
|
@SerialName(frontSideField)
|
||||||
|
override val frontSide: FileCredentials? = null,
|
||||||
|
@SerialName(reverseSideField)
|
||||||
|
override val reverseSide: FileCredentials? = null,
|
||||||
|
@SerialName(selfieField)
|
||||||
|
override val selfie: FileCredentials? = null,
|
||||||
|
@SerialName(translationField)
|
||||||
|
override val translation: List<FileCredentials> = emptyList()
|
||||||
|
) : IdentityWithReverseSideSecureValue()
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class IdentityCardSecureValue(
|
||||||
|
@SerialName(dataField)
|
||||||
|
override val data: DataCredentials? = null,
|
||||||
|
@SerialName(frontSideField)
|
||||||
|
override val frontSide: FileCredentials? = null,
|
||||||
|
@SerialName(reverseSideField)
|
||||||
|
override val reverseSide: FileCredentials? = null,
|
||||||
|
@SerialName(selfieField)
|
||||||
|
override val selfie: FileCredentials? = null,
|
||||||
|
@SerialName(translationField)
|
||||||
|
override val translation: List<FileCredentials> = emptyList()
|
||||||
|
) : IdentityWithReverseSideSecureValue()
|
||||||
@@ -0,0 +1,56 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.decrypted
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.filesField
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.EndDataCredentials
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.FileCredentials
|
||||||
|
import dev.inmo.tgbotapi.types.passport.decrypted.abstracts.SecureValueWithFiles
|
||||||
|
import dev.inmo.tgbotapi.types.passport.decrypted.abstracts.SecureValueWithTranslations
|
||||||
|
import dev.inmo.tgbotapi.types.translationField
|
||||||
|
import kotlinx.serialization.SerialName
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
sealed class OtherDocumentsSecureValue : SecureValueWithTranslations, SecureValueWithFiles {
|
||||||
|
override val credentials: List<EndDataCredentials>
|
||||||
|
get() = translation + files
|
||||||
|
}
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class UtilityBillSecureValue(
|
||||||
|
@SerialName(translationField)
|
||||||
|
override val translation: List<FileCredentials> = emptyList(),
|
||||||
|
@SerialName(filesField)
|
||||||
|
override val files: List<FileCredentials> = emptyList()
|
||||||
|
) : OtherDocumentsSecureValue()
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class BankStatementSecureValue(
|
||||||
|
@SerialName(translationField)
|
||||||
|
override val translation: List<FileCredentials> = emptyList(),
|
||||||
|
@SerialName(filesField)
|
||||||
|
override val files: List<FileCredentials> = emptyList()
|
||||||
|
) : OtherDocumentsSecureValue()
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class RentalAgreementSecureValue(
|
||||||
|
@SerialName(translationField)
|
||||||
|
override val translation: List<FileCredentials> = emptyList(),
|
||||||
|
@SerialName(filesField)
|
||||||
|
override val files: List<FileCredentials> = emptyList()
|
||||||
|
) : OtherDocumentsSecureValue()
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class PassportRegistrationSecureValue(
|
||||||
|
@SerialName(translationField)
|
||||||
|
override val translation: List<FileCredentials> = emptyList(),
|
||||||
|
@SerialName(filesField)
|
||||||
|
override val files: List<FileCredentials> = emptyList()
|
||||||
|
) : OtherDocumentsSecureValue()
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class TemporalRegistrationSecureValue(
|
||||||
|
@SerialName(translationField)
|
||||||
|
override val translation: List<FileCredentials> = emptyList(),
|
||||||
|
@SerialName(filesField)
|
||||||
|
override val files: List<FileCredentials> = emptyList()
|
||||||
|
) : OtherDocumentsSecureValue()
|
||||||
@@ -0,0 +1,38 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.decrypted
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.*
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.*
|
||||||
|
import dev.inmo.tgbotapi.types.passport.decrypted.abstracts.*
|
||||||
|
import kotlinx.serialization.SerialName
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
sealed class PassportSecureValue : SecureValueIdentity, SecureValueWithData, SecureValueWithTranslations {
|
||||||
|
override val credentials: List<EndDataCredentials>
|
||||||
|
get() = listOfNotNull(data, frontSide, selfie) + translation
|
||||||
|
}
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class CommonPassportSecureValue(
|
||||||
|
@SerialName(dataField)
|
||||||
|
override val data: DataCredentials? = null,
|
||||||
|
@SerialName(frontSideField)
|
||||||
|
override val frontSide: FileCredentials? = null,
|
||||||
|
@SerialName(selfieField)
|
||||||
|
override val selfie: FileCredentials? = null,
|
||||||
|
@SerialName(translationField)
|
||||||
|
override val translation: List<FileCredentials> = emptyList()
|
||||||
|
) : PassportSecureValue()
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class InternalPassportSecureValue(
|
||||||
|
@SerialName(dataField)
|
||||||
|
override val data: DataCredentials? = null,
|
||||||
|
@SerialName(frontSideField)
|
||||||
|
override val frontSide: FileCredentials? = null,
|
||||||
|
@SerialName(selfieField)
|
||||||
|
override val selfie: FileCredentials? = null,
|
||||||
|
@SerialName(translationField)
|
||||||
|
override val translation: List<FileCredentials> = emptyList()
|
||||||
|
) : PassportSecureValue()
|
||||||
|
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.decrypted
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.dataField
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.DataCredentials
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.EndDataCredentials
|
||||||
|
import dev.inmo.tgbotapi.types.passport.decrypted.abstracts.SecureValueWithData
|
||||||
|
import kotlinx.serialization.SerialName
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class PersonalDetailsSecureValue(
|
||||||
|
@SerialName(dataField)
|
||||||
|
override val data: DataCredentials
|
||||||
|
) : SecureValueWithData {
|
||||||
|
override val credentials: List<EndDataCredentials> = listOf(data)
|
||||||
|
}
|
||||||
@@ -0,0 +1,42 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.decrypted
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.*
|
||||||
|
import kotlinx.serialization.SerialName
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class SecureData(
|
||||||
|
@SerialName(personalDetailsField)
|
||||||
|
val personalDetails: PersonalDetailsSecureValue? = null,
|
||||||
|
@SerialName(passportField)
|
||||||
|
val passport: CommonPassportSecureValue? = null,
|
||||||
|
@SerialName(internalPassportField)
|
||||||
|
val internalPassport: InternalPassportSecureValue? = null,
|
||||||
|
@SerialName(driverLicenseField)
|
||||||
|
val driverLicense: DriverLicenseSecureValue? = null,
|
||||||
|
@SerialName(identityCardField)
|
||||||
|
val identityCard: IdentityCardSecureValue? = null,
|
||||||
|
@SerialName(utilityBillField)
|
||||||
|
val utilityBill: UtilityBillSecureValue? = null,
|
||||||
|
@SerialName(bankStatementField)
|
||||||
|
val bankStatement: BankStatementSecureValue? = null,
|
||||||
|
@SerialName(rentalAgreementField)
|
||||||
|
val rentalAgreement: RentalAgreementSecureValue? = null,
|
||||||
|
@SerialName(passportRegistrationField)
|
||||||
|
val passportRegistration: PassportRegistrationSecureValue? = null,
|
||||||
|
@SerialName(temporaryRegistrationField)
|
||||||
|
val temporaryRegistration: TemporalRegistrationSecureValue? = null,
|
||||||
|
) {
|
||||||
|
val allCredentials by lazy {
|
||||||
|
(personalDetails ?.credentials ?: emptyList()) +
|
||||||
|
(passport ?.credentials ?: emptyList()) +
|
||||||
|
(internalPassport ?.credentials ?: emptyList()) +
|
||||||
|
(driverLicense ?.credentials ?: emptyList()) +
|
||||||
|
(identityCard ?.credentials ?: emptyList()) +
|
||||||
|
(utilityBill ?.credentials ?: emptyList()) +
|
||||||
|
(bankStatement ?.credentials ?: emptyList()) +
|
||||||
|
(rentalAgreement ?.credentials ?: emptyList()) +
|
||||||
|
(passportRegistration ?.credentials ?: emptyList()) +
|
||||||
|
(temporaryRegistration ?.credentials ?: emptyList())
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.decrypted.abstracts
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.EndDataCredentials
|
||||||
|
|
||||||
|
interface SecureValue {
|
||||||
|
val credentials: List<EndDataCredentials>
|
||||||
|
}
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.decrypted.abstracts
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.FileCredentials
|
||||||
|
|
||||||
|
interface SecureValueIdentity : SecureValue {
|
||||||
|
val frontSide: FileCredentials?
|
||||||
|
val selfie: FileCredentials?
|
||||||
|
}
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.decrypted.abstracts
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.DataCredentials
|
||||||
|
|
||||||
|
interface SecureValueWithData : SecureValue {
|
||||||
|
val data: DataCredentials?
|
||||||
|
}
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.decrypted.abstracts
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.FileCredentials
|
||||||
|
|
||||||
|
interface SecureValueWithFiles : SecureValue {
|
||||||
|
val files: List<FileCredentials>
|
||||||
|
}
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.decrypted.abstracts
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.FileCredentials
|
||||||
|
|
||||||
|
interface SecureValueWithReverseSide : SecureValue {
|
||||||
|
val reverseSide: FileCredentials?
|
||||||
|
}
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.decrypted.abstracts
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.FileCredentials
|
||||||
|
|
||||||
|
interface SecureValueWithTranslations : SecureValue {
|
||||||
|
val translation: List<FileCredentials>
|
||||||
|
}
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.encrypted
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.serialization.base64.Base64BytesToFromStringSerializer
|
||||||
|
import dev.inmo.tgbotapi.types.emailField
|
||||||
|
import dev.inmo.tgbotapi.types.hashField
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.PassportElementHash
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.EncryptedPassportElementWithEmail
|
||||||
|
import kotlinx.serialization.SerialName
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class Email(
|
||||||
|
@SerialName(emailField)
|
||||||
|
override val email: String,
|
||||||
|
@SerialName(hashField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val hash: PassportElementHash
|
||||||
|
) : EncryptedPassportElementWithEmail
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.encrypted
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.serialization.base64.Base64BytesToFromStringSerializer
|
||||||
|
import dev.inmo.tgbotapi.types.dataField
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.EncryptedData
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.PassportElementHash
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.EncryptedPassportElementWithData
|
||||||
|
import kotlinx.serialization.SerialName
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class EncryptedAddress(
|
||||||
|
@SerialName(dataField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val data: EncryptedData,
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val hash: PassportElementHash
|
||||||
|
) : EncryptedPassportElementWithData
|
||||||
@@ -0,0 +1,64 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.encrypted
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.crypto.decodeBase64
|
||||||
|
import dev.inmo.micro_utils.serialization.encapsulator.Encapsulator
|
||||||
|
import dev.inmo.tgbotapi.types.hashField
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.EncryptedPassportElement
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.UnknownEncryptedPassportElement
|
||||||
|
import dev.inmo.tgbotapi.types.typeField
|
||||||
|
import dev.inmo.tgbotapi.utils.RiskFeature
|
||||||
|
import dev.inmo.tgbotapi.utils.nonstrictJsonFormat
|
||||||
|
import kotlinx.serialization.KSerializer
|
||||||
|
import kotlinx.serialization.Serializer
|
||||||
|
import kotlinx.serialization.descriptors.SerialDescriptor
|
||||||
|
import kotlinx.serialization.encoding.Decoder
|
||||||
|
import kotlinx.serialization.encoding.Encoder
|
||||||
|
import kotlinx.serialization.json.*
|
||||||
|
|
||||||
|
val encryptedElementsClassesByTypes = mapOf(
|
||||||
|
"personal_details" to Encapsulator(EncryptedPersonalDetails::class, EncryptedPersonalDetails.serializer()),
|
||||||
|
"passport" to Encapsulator(CommonPassport::class, CommonPassport.serializer()),
|
||||||
|
"driver_license" to Encapsulator(DriverLicense::class, DriverLicense.serializer()),
|
||||||
|
"identity_card" to Encapsulator(IdentityCard::class, IdentityCard.serializer()),
|
||||||
|
"internal_passport" to Encapsulator(InternalPassport::class, InternalPassport.serializer()),
|
||||||
|
"address" to Encapsulator(EncryptedAddress::class, EncryptedAddress.serializer()),
|
||||||
|
"utility_bill" to Encapsulator(UtilityBill::class, UtilityBill.serializer()),
|
||||||
|
"bank_statement" to Encapsulator(BankStatement::class, BankStatement.serializer()),
|
||||||
|
"rental_agreement" to Encapsulator(RentalAgreement::class, RentalAgreement.serializer()),
|
||||||
|
"passport_registration" to Encapsulator(PassportRegistration::class, PassportRegistration.serializer()),
|
||||||
|
"temporary_registration" to Encapsulator(TemporaryRegistration::class, TemporaryRegistration.serializer()),
|
||||||
|
"phone_number" to Encapsulator(PhoneNumber::class, PhoneNumber.serializer()),
|
||||||
|
"email" to Encapsulator(Email::class, Email.serializer())
|
||||||
|
)
|
||||||
|
|
||||||
|
@RiskFeature("Remember that this method may return \"unknown\" in case if encrypted element was not defined in library")
|
||||||
|
val EncryptedPassportElement.type: String
|
||||||
|
get() = encryptedElementsClassesByTypes.keys.firstOrNull { encryptedElementsClassesByTypes.getValue(it).klass.isInstance(this) } ?: "unknown"
|
||||||
|
|
||||||
|
@Serializer(EncryptedPassportElement::class)
|
||||||
|
object EncryptedElementSerializer : KSerializer<EncryptedPassportElement> {
|
||||||
|
private val jsonSerializer = JsonObject.serializer()
|
||||||
|
override val descriptor: SerialDescriptor = jsonSerializer.descriptor
|
||||||
|
|
||||||
|
override fun deserialize(decoder: Decoder): EncryptedPassportElement {
|
||||||
|
val json = jsonSerializer.deserialize(decoder)
|
||||||
|
return json[typeField] ?.jsonPrimitive ?.content ?.let { type ->
|
||||||
|
encryptedElementsClassesByTypes[type] ?.serializer ?.let { deserializer ->
|
||||||
|
nonstrictJsonFormat.decodeFromJsonElement(deserializer, json)
|
||||||
|
}
|
||||||
|
} ?: UnknownEncryptedPassportElement(json, json[hashField] ?.jsonPrimitive ?.content ?.decodeBase64() ?: byteArrayOf())
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun serialize(encoder: Encoder, value: EncryptedPassportElement) {
|
||||||
|
val json = value.let {
|
||||||
|
encryptedElementsClassesByTypes.forEach { (key, encapsulator) ->
|
||||||
|
val json = encapsulator.encapsulate(value) { data ->
|
||||||
|
nonstrictJsonFormat.encodeToJsonElement(this as KSerializer<EncryptedPassportElement>, data).jsonObject
|
||||||
|
} ?: return@forEach
|
||||||
|
return@let JsonObject(json + (typeField to JsonPrimitive(key)))
|
||||||
|
}
|
||||||
|
(value as? UnknownEncryptedPassportElement) ?.rawJson ?: return
|
||||||
|
}
|
||||||
|
jsonSerializer.serialize(encoder, json)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,14 +1,13 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data
|
package dev.inmo.tgbotapi.types.passport.encrypted
|
||||||
|
|
||||||
import dev.inmo.micro_utils.serialization.base64.Base64StringSerializer
|
import dev.inmo.micro_utils.serialization.base64.Base64BytesToFromStringSerializer
|
||||||
import dev.inmo.tgbotapi.types.*
|
import dev.inmo.tgbotapi.types.*
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts.FilesCollection
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.*
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts.Translatable
|
|
||||||
import kotlinx.serialization.SerialName
|
import kotlinx.serialization.SerialName
|
||||||
import kotlinx.serialization.Serializable
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
@Serializable(EncryptedElementSerializer::class)
|
@Serializable(EncryptedElementSerializer::class)
|
||||||
sealed class TranslatableFilesCollection : Translatable, FilesCollection
|
sealed class EncryptedPassportElementWithTranslatableFilesCollection : EncryptedPassportElementTranslatable, EncryptedPassportElementWithFilesCollection
|
||||||
|
|
||||||
@Serializable
|
@Serializable
|
||||||
data class UtilityBill(
|
data class UtilityBill(
|
||||||
@@ -17,9 +16,9 @@ data class UtilityBill(
|
|||||||
@SerialName(translationField)
|
@SerialName(translationField)
|
||||||
override val translations: List<PassportFile> = emptyList(),
|
override val translations: List<PassportFile> = emptyList(),
|
||||||
@SerialName(hashField)
|
@SerialName(hashField)
|
||||||
@Serializable(Base64StringSerializer::class)
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
override val hash: String
|
override val hash: PassportElementHash
|
||||||
) : TranslatableFilesCollection()
|
) : EncryptedPassportElementWithTranslatableFilesCollection()
|
||||||
@Serializable
|
@Serializable
|
||||||
data class BankStatement(
|
data class BankStatement(
|
||||||
@SerialName(filesField)
|
@SerialName(filesField)
|
||||||
@@ -27,9 +26,9 @@ data class BankStatement(
|
|||||||
@SerialName(translationField)
|
@SerialName(translationField)
|
||||||
override val translations: List<PassportFile> = emptyList(),
|
override val translations: List<PassportFile> = emptyList(),
|
||||||
@SerialName(hashField)
|
@SerialName(hashField)
|
||||||
@Serializable(Base64StringSerializer::class)
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
override val hash: String
|
override val hash: PassportElementHash
|
||||||
) : TranslatableFilesCollection()
|
) : EncryptedPassportElementWithTranslatableFilesCollection()
|
||||||
@Serializable
|
@Serializable
|
||||||
data class RentalAgreement(
|
data class RentalAgreement(
|
||||||
@SerialName(filesField)
|
@SerialName(filesField)
|
||||||
@@ -37,9 +36,9 @@ data class RentalAgreement(
|
|||||||
@SerialName(translationField)
|
@SerialName(translationField)
|
||||||
override val translations: List<PassportFile> = emptyList(),
|
override val translations: List<PassportFile> = emptyList(),
|
||||||
@SerialName(hashField)
|
@SerialName(hashField)
|
||||||
@Serializable(Base64StringSerializer::class)
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
override val hash: String
|
override val hash: PassportElementHash
|
||||||
) : TranslatableFilesCollection()
|
) : EncryptedPassportElementWithTranslatableFilesCollection()
|
||||||
@Serializable
|
@Serializable
|
||||||
data class PassportRegistration(
|
data class PassportRegistration(
|
||||||
@SerialName(filesField)
|
@SerialName(filesField)
|
||||||
@@ -47,9 +46,9 @@ data class PassportRegistration(
|
|||||||
@SerialName(translationField)
|
@SerialName(translationField)
|
||||||
override val translations: List<PassportFile> = emptyList(),
|
override val translations: List<PassportFile> = emptyList(),
|
||||||
@SerialName(hashField)
|
@SerialName(hashField)
|
||||||
@Serializable(Base64StringSerializer::class)
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
override val hash: String
|
override val hash: PassportElementHash
|
||||||
) : TranslatableFilesCollection()
|
) : EncryptedPassportElementWithTranslatableFilesCollection()
|
||||||
@Serializable
|
@Serializable
|
||||||
data class TemporaryRegistration(
|
data class TemporaryRegistration(
|
||||||
@SerialName(filesField)
|
@SerialName(filesField)
|
||||||
@@ -57,7 +56,7 @@ data class TemporaryRegistration(
|
|||||||
@SerialName(translationField)
|
@SerialName(translationField)
|
||||||
override val translations: List<PassportFile> = emptyList(),
|
override val translations: List<PassportFile> = emptyList(),
|
||||||
@SerialName(hashField)
|
@SerialName(hashField)
|
||||||
@Serializable(Base64StringSerializer::class)
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
override val hash: String
|
override val hash: PassportElementHash
|
||||||
) : TranslatableFilesCollection()
|
) : EncryptedPassportElementWithTranslatableFilesCollection()
|
||||||
|
|
||||||
@@ -1,19 +1,19 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data
|
package dev.inmo.tgbotapi.types.passport.encrypted
|
||||||
|
|
||||||
import dev.inmo.micro_utils.serialization.base64.Base64StringSerializer
|
import dev.inmo.micro_utils.serialization.base64.Base64BytesToFromStringSerializer
|
||||||
import dev.inmo.tgbotapi.types.*
|
import dev.inmo.tgbotapi.types.*
|
||||||
import dev.inmo.tgbotapi.types.passport.EncryptedData
|
import dev.inmo.tgbotapi.types.passport.credentials.EncryptedData
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts.*
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.*
|
||||||
import kotlinx.serialization.SerialName
|
import kotlinx.serialization.SerialName
|
||||||
import kotlinx.serialization.Serializable
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
@Serializable(EncryptedElementSerializer::class)
|
@Serializable(EncryptedElementSerializer::class)
|
||||||
sealed class TranslatableIDDocument : WithData, WithFrontSide, WithReverseSide, WithSelfie, Translatable
|
sealed class EncryptedPassportElementWithTranslatableIDDocument : EncryptedPassportElementWithData, EncryptedPassportElementWithFrontSide, EncryptedPassportElementWithReverseSide, EncryptedPassportElementWithSelfie, EncryptedPassportElementTranslatable
|
||||||
|
|
||||||
@Serializable
|
@Serializable
|
||||||
data class DriverLicense(
|
data class DriverLicense(
|
||||||
@SerialName(dataField)
|
@SerialName(dataField)
|
||||||
@Serializable(Base64StringSerializer::class)
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
override val data: EncryptedData,
|
override val data: EncryptedData,
|
||||||
@SerialName(frontSideField)
|
@SerialName(frontSideField)
|
||||||
override val frontSide: PassportFile? = null,
|
override val frontSide: PassportFile? = null,
|
||||||
@@ -24,14 +24,14 @@ data class DriverLicense(
|
|||||||
@SerialName(translationField)
|
@SerialName(translationField)
|
||||||
override val translations: List<PassportFile> = emptyList(),
|
override val translations: List<PassportFile> = emptyList(),
|
||||||
@SerialName(hashField)
|
@SerialName(hashField)
|
||||||
@Serializable(Base64StringSerializer::class)
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
override val hash: String
|
override val hash: PassportElementHash
|
||||||
) : TranslatableIDDocument()
|
) : EncryptedPassportElementWithTranslatableIDDocument()
|
||||||
|
|
||||||
@Serializable
|
@Serializable
|
||||||
data class IdentityCard(
|
data class IdentityCard(
|
||||||
@SerialName(dataField)
|
@SerialName(dataField)
|
||||||
@Serializable(Base64StringSerializer::class)
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
override val data: EncryptedData,
|
override val data: EncryptedData,
|
||||||
@SerialName(frontSideField)
|
@SerialName(frontSideField)
|
||||||
override val frontSide: PassportFile? = null,
|
override val frontSide: PassportFile? = null,
|
||||||
@@ -42,6 +42,6 @@ data class IdentityCard(
|
|||||||
@SerialName(translationField)
|
@SerialName(translationField)
|
||||||
override val translations: List<PassportFile> = emptyList(),
|
override val translations: List<PassportFile> = emptyList(),
|
||||||
@SerialName(hashField)
|
@SerialName(hashField)
|
||||||
@Serializable(Base64StringSerializer::class)
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
override val hash: String
|
override val hash: PassportElementHash
|
||||||
) : TranslatableIDDocument()
|
) : EncryptedPassportElementWithTranslatableIDDocument()
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.encrypted
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.serialization.base64.Base64BytesToFromStringSerializer
|
||||||
|
import dev.inmo.tgbotapi.types.dataField
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.EncryptedData
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.PassportElementHash
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.EncryptedPassportElementWithData
|
||||||
|
import kotlinx.serialization.SerialName
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class EncryptedPersonalDetails(
|
||||||
|
@SerialName(dataField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val data: EncryptedData,
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val hash: PassportElementHash
|
||||||
|
) : EncryptedPassportElementWithData
|
||||||
@@ -1,19 +1,19 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data
|
package dev.inmo.tgbotapi.types.passport.encrypted
|
||||||
|
|
||||||
import dev.inmo.micro_utils.serialization.base64.Base64StringSerializer
|
import dev.inmo.micro_utils.serialization.base64.Base64BytesToFromStringSerializer
|
||||||
import dev.inmo.tgbotapi.types.*
|
import dev.inmo.tgbotapi.types.*
|
||||||
import dev.inmo.tgbotapi.types.passport.EncryptedData
|
import dev.inmo.tgbotapi.types.passport.credentials.EncryptedData
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts.*
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.*
|
||||||
import kotlinx.serialization.SerialName
|
import kotlinx.serialization.SerialName
|
||||||
import kotlinx.serialization.Serializable
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
@Serializable(EncryptedElementSerializer::class)
|
@Serializable(EncryptedElementSerializer::class)
|
||||||
sealed class Passport : WithData, WithFrontSide, WithSelfie, Translatable
|
sealed class Passport : EncryptedPassportElementWithData, EncryptedPassportElementWithFrontSide, EncryptedPassportElementWithSelfie, EncryptedPassportElementTranslatable
|
||||||
|
|
||||||
@Serializable
|
@Serializable
|
||||||
data class CommonPassport(
|
data class CommonPassport(
|
||||||
@SerialName(dataField)
|
@SerialName(dataField)
|
||||||
@Serializable(Base64StringSerializer::class)
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
override val data: EncryptedData,
|
override val data: EncryptedData,
|
||||||
@SerialName(frontSideField)
|
@SerialName(frontSideField)
|
||||||
override val frontSide: PassportFile? = null,
|
override val frontSide: PassportFile? = null,
|
||||||
@@ -22,13 +22,13 @@ data class CommonPassport(
|
|||||||
@SerialName(translationField)
|
@SerialName(translationField)
|
||||||
override val translations: List<PassportFile> = emptyList(),
|
override val translations: List<PassportFile> = emptyList(),
|
||||||
@SerialName(hashField)
|
@SerialName(hashField)
|
||||||
@Serializable(Base64StringSerializer::class)
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
override val hash: String
|
override val hash: PassportElementHash
|
||||||
) : Passport()
|
) : Passport()
|
||||||
@Serializable
|
@Serializable
|
||||||
data class InternalPassport(
|
data class InternalPassport(
|
||||||
@SerialName(dataField)
|
@SerialName(dataField)
|
||||||
@Serializable(Base64StringSerializer::class)
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
override val data: EncryptedData,
|
override val data: EncryptedData,
|
||||||
@SerialName(frontSideField)
|
@SerialName(frontSideField)
|
||||||
override val frontSide: PassportFile? = null,
|
override val frontSide: PassportFile? = null,
|
||||||
@@ -37,6 +37,6 @@ data class InternalPassport(
|
|||||||
@SerialName(translationField)
|
@SerialName(translationField)
|
||||||
override val translations: List<PassportFile> = emptyList(),
|
override val translations: List<PassportFile> = emptyList(),
|
||||||
@SerialName(hashField)
|
@SerialName(hashField)
|
||||||
@Serializable(Base64StringSerializer::class)
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
override val hash: String
|
override val hash: PassportElementHash
|
||||||
) : Passport()
|
) : Passport()
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data
|
package dev.inmo.tgbotapi.types.passport.encrypted
|
||||||
|
|
||||||
import dev.inmo.tgbotapi.requests.abstracts.FileId
|
import dev.inmo.tgbotapi.requests.abstracts.FileId
|
||||||
import dev.inmo.tgbotapi.types.*
|
import dev.inmo.tgbotapi.types.*
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.encrypted
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.serialization.base64.Base64BytesToFromStringSerializer
|
||||||
|
import dev.inmo.tgbotapi.types.hashField
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.PassportElementHash
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.EncryptedPassportElementWithPhoneNumber
|
||||||
|
import dev.inmo.tgbotapi.types.phoneNumberField
|
||||||
|
import kotlinx.serialization.SerialName
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class PhoneNumber(
|
||||||
|
@SerialName(phoneNumberField)
|
||||||
|
override val phoneNumber: String,
|
||||||
|
@SerialName(hashField)
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val hash: PassportElementHash
|
||||||
|
) : EncryptedPassportElementWithPhoneNumber
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.encrypted.abstracts
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.crypto.SourceBytes
|
||||||
|
import dev.inmo.micro_utils.serialization.base64.Base64BytesToFromStringSerializer
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.EncryptedElementSerializer
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
import kotlinx.serialization.json.JsonObject
|
||||||
|
|
||||||
|
typealias PassportElementHash = SourceBytes
|
||||||
|
|
||||||
|
@Serializable(EncryptedElementSerializer::class)
|
||||||
|
interface EncryptedPassportElement {
|
||||||
|
val hash: PassportElementHash
|
||||||
|
}
|
||||||
|
|
||||||
|
@Serializable(EncryptedElementSerializer::class)
|
||||||
|
data class UnknownEncryptedPassportElement(
|
||||||
|
val rawJson: JsonObject,
|
||||||
|
@Serializable(Base64BytesToFromStringSerializer::class)
|
||||||
|
override val hash: PassportElementHash
|
||||||
|
) : EncryptedPassportElement
|
||||||
@@ -0,0 +1,10 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.encrypted.abstracts
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.EncryptedElementSerializer
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.PassportFile
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable(EncryptedElementSerializer::class)
|
||||||
|
interface EncryptedPassportElementTranslatable : EncryptedPassportElement {
|
||||||
|
val translations: List<PassportFile>
|
||||||
|
}
|
||||||
@@ -0,0 +1,10 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.encrypted.abstracts
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.EncryptedData
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.EncryptedElementSerializer
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable(EncryptedElementSerializer::class)
|
||||||
|
interface EncryptedPassportElementWithData : EncryptedPassportElement {
|
||||||
|
val data: EncryptedData
|
||||||
|
}
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.encrypted.abstracts
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.EncryptedElementSerializer
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable(EncryptedElementSerializer::class)
|
||||||
|
interface EncryptedPassportElementWithEmail : EncryptedPassportElement {
|
||||||
|
val email: String
|
||||||
|
}
|
||||||
@@ -0,0 +1,10 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.encrypted.abstracts
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.EncryptedElementSerializer
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.PassportFile
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable(EncryptedElementSerializer::class)
|
||||||
|
interface EncryptedPassportElementWithFilesCollection : EncryptedPassportElement {
|
||||||
|
val files: List<PassportFile>
|
||||||
|
}
|
||||||
@@ -0,0 +1,10 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.encrypted.abstracts
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.EncryptedElementSerializer
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.PassportFile
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable(EncryptedElementSerializer::class)
|
||||||
|
interface EncryptedPassportElementWithFrontSide : EncryptedPassportElement {
|
||||||
|
val frontSide: PassportFile?
|
||||||
|
}
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.encrypted.abstracts
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.EncryptedElementSerializer
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable(EncryptedElementSerializer::class)
|
||||||
|
interface EncryptedPassportElementWithPhoneNumber : EncryptedPassportElement {
|
||||||
|
val phoneNumber: String
|
||||||
|
}
|
||||||
@@ -0,0 +1,10 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.encrypted.abstracts
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.EncryptedElementSerializer
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.PassportFile
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable(EncryptedElementSerializer::class)
|
||||||
|
interface EncryptedPassportElementWithReverseSide : EncryptedPassportElement {
|
||||||
|
val reverseSide: PassportFile?
|
||||||
|
}
|
||||||
@@ -0,0 +1,10 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport.encrypted.abstracts
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.EncryptedElementSerializer
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.PassportFile
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
|
||||||
|
@Serializable(EncryptedElementSerializer::class)
|
||||||
|
interface EncryptedPassportElementWithSelfie : EncryptedPassportElement {
|
||||||
|
val selfie: PassportFile?
|
||||||
|
}
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data
|
|
||||||
|
|
||||||
import dev.inmo.micro_utils.serialization.base64.Base64StringSerializer
|
|
||||||
import dev.inmo.tgbotapi.types.emailField
|
|
||||||
import dev.inmo.tgbotapi.types.hashField
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts.WithEmail
|
|
||||||
import kotlinx.serialization.SerialName
|
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
|
|
||||||
@Serializable
|
|
||||||
data class Email(
|
|
||||||
@SerialName(emailField)
|
|
||||||
override val email: String,
|
|
||||||
@SerialName(hashField)
|
|
||||||
@Serializable(Base64StringSerializer::class)
|
|
||||||
override val hash: String
|
|
||||||
) : WithEmail {
|
|
||||||
}
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data
|
|
||||||
|
|
||||||
import dev.inmo.micro_utils.serialization.base64.Base64StringSerializer
|
|
||||||
import dev.inmo.tgbotapi.types.dataField
|
|
||||||
import dev.inmo.tgbotapi.types.passport.EncryptedData
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts.WithData
|
|
||||||
import kotlinx.serialization.SerialName
|
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
|
|
||||||
@Serializable
|
|
||||||
data class EncryptedAddress(
|
|
||||||
@SerialName(dataField)
|
|
||||||
@Serializable(Base64StringSerializer::class)
|
|
||||||
override val data: EncryptedData,
|
|
||||||
@Serializable(Base64StringSerializer::class)
|
|
||||||
override val hash: String
|
|
||||||
) : WithData
|
|
||||||
@@ -1,87 +0,0 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data
|
|
||||||
|
|
||||||
import dev.inmo.tgbotapi.types.hashField
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts.EncryptedPassportElement
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts.UnknownEncryptedPassportElement
|
|
||||||
import dev.inmo.tgbotapi.types.typeField
|
|
||||||
import dev.inmo.tgbotapi.utils.nonstrictJsonFormat
|
|
||||||
import kotlinx.serialization.KSerializer
|
|
||||||
import kotlinx.serialization.Serializer
|
|
||||||
import kotlinx.serialization.descriptors.SerialDescriptor
|
|
||||||
import kotlinx.serialization.encoding.Decoder
|
|
||||||
import kotlinx.serialization.encoding.Encoder
|
|
||||||
import kotlinx.serialization.json.*
|
|
||||||
|
|
||||||
@Serializer(EncryptedPassportElement::class)
|
|
||||||
object EncryptedElementSerializer : KSerializer<EncryptedPassportElement> {
|
|
||||||
private val jsonSerializer = JsonObject.serializer()
|
|
||||||
override val descriptor: SerialDescriptor = jsonSerializer.descriptor
|
|
||||||
|
|
||||||
override fun deserialize(decoder: Decoder): EncryptedPassportElement {
|
|
||||||
val json = jsonSerializer.deserialize(decoder)
|
|
||||||
return when (json[typeField] ?.jsonPrimitive ?.content) {
|
|
||||||
"personal_details" -> nonstrictJsonFormat.decodeFromJsonElement(EncryptedPersonalDetails.serializer(), json)
|
|
||||||
"passport" -> nonstrictJsonFormat.decodeFromJsonElement(CommonPassport.serializer(), json)
|
|
||||||
"driver_license" -> nonstrictJsonFormat.decodeFromJsonElement(DriverLicense.serializer(), json)
|
|
||||||
"identity_card" -> nonstrictJsonFormat.decodeFromJsonElement(IdentityCard.serializer(), json)
|
|
||||||
"internal_passport" -> nonstrictJsonFormat.decodeFromJsonElement(InternalPassport.serializer(), json)
|
|
||||||
"address" -> nonstrictJsonFormat.decodeFromJsonElement(EncryptedAddress.serializer(), json)
|
|
||||||
"utility_bill" -> nonstrictJsonFormat.decodeFromJsonElement(UtilityBill.serializer(), json)
|
|
||||||
"bank_statement" -> nonstrictJsonFormat.decodeFromJsonElement(BankStatement.serializer(), json)
|
|
||||||
"rental_agreement" -> nonstrictJsonFormat.decodeFromJsonElement(RentalAgreement.serializer(), json)
|
|
||||||
"passport_registration" -> nonstrictJsonFormat.decodeFromJsonElement(PassportRegistration.serializer(), json)
|
|
||||||
"temporary_registration" -> nonstrictJsonFormat.decodeFromJsonElement(TemporaryRegistration.serializer(), json)
|
|
||||||
"phone_number" -> nonstrictJsonFormat.decodeFromJsonElement(PhoneNumber.serializer(), json)
|
|
||||||
"email" -> nonstrictJsonFormat.decodeFromJsonElement(Email.serializer(), json)
|
|
||||||
else -> UnknownEncryptedPassportElement(json, json[hashField] ?.jsonPrimitive ?.content ?: "")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun serialize(encoder: Encoder, value: EncryptedPassportElement) {
|
|
||||||
val json = when (value) {
|
|
||||||
is EncryptedPersonalDetails -> JsonObject(
|
|
||||||
nonstrictJsonFormat.encodeToJsonElement(EncryptedPersonalDetails.serializer(), value).jsonObject + (typeField to JsonPrimitive("personal_details"))
|
|
||||||
)
|
|
||||||
is CommonPassport -> JsonObject(
|
|
||||||
nonstrictJsonFormat.encodeToJsonElement(CommonPassport.serializer(), value).jsonObject + (typeField to JsonPrimitive("passport"))
|
|
||||||
)
|
|
||||||
is DriverLicense -> JsonObject(
|
|
||||||
nonstrictJsonFormat.encodeToJsonElement(DriverLicense.serializer(), value).jsonObject + (typeField to JsonPrimitive("driver_license"))
|
|
||||||
)
|
|
||||||
is IdentityCard -> JsonObject(
|
|
||||||
nonstrictJsonFormat.encodeToJsonElement(IdentityCard.serializer(), value).jsonObject + (typeField to JsonPrimitive("identity_card"))
|
|
||||||
)
|
|
||||||
is InternalPassport -> JsonObject(
|
|
||||||
nonstrictJsonFormat.encodeToJsonElement(InternalPassport.serializer(), value).jsonObject + (typeField to JsonPrimitive("internal_passport"))
|
|
||||||
)
|
|
||||||
is EncryptedAddress -> JsonObject(
|
|
||||||
nonstrictJsonFormat.encodeToJsonElement(EncryptedAddress.serializer(), value).jsonObject + (typeField to JsonPrimitive("address"))
|
|
||||||
)
|
|
||||||
is UtilityBill -> JsonObject(
|
|
||||||
nonstrictJsonFormat.encodeToJsonElement(UtilityBill.serializer(), value).jsonObject + (typeField to JsonPrimitive("utility_bill"))
|
|
||||||
)
|
|
||||||
is BankStatement -> JsonObject(
|
|
||||||
nonstrictJsonFormat.encodeToJsonElement(BankStatement.serializer(), value).jsonObject + (typeField to JsonPrimitive("bank_statement"))
|
|
||||||
)
|
|
||||||
is RentalAgreement -> JsonObject(
|
|
||||||
nonstrictJsonFormat.encodeToJsonElement(RentalAgreement.serializer(), value).jsonObject + (typeField to JsonPrimitive("rental_agreement"))
|
|
||||||
)
|
|
||||||
is PassportRegistration -> JsonObject(
|
|
||||||
nonstrictJsonFormat.encodeToJsonElement(PassportRegistration.serializer(), value).jsonObject + (typeField to JsonPrimitive("passport_registration"))
|
|
||||||
)
|
|
||||||
is TemporaryRegistration -> JsonObject(
|
|
||||||
nonstrictJsonFormat.encodeToJsonElement(TemporaryRegistration.serializer(), value).jsonObject + (typeField to JsonPrimitive("temporary_registration"))
|
|
||||||
)
|
|
||||||
is PhoneNumber -> JsonObject(
|
|
||||||
nonstrictJsonFormat.encodeToJsonElement(PhoneNumber.serializer(), value).jsonObject + (typeField to JsonPrimitive("phone_number"))
|
|
||||||
)
|
|
||||||
is Email -> JsonObject(
|
|
||||||
nonstrictJsonFormat.encodeToJsonElement(Email.serializer(), value).jsonObject + (typeField to JsonPrimitive("email"))
|
|
||||||
)
|
|
||||||
is UnknownEncryptedPassportElement -> value.rawJson
|
|
||||||
else -> return
|
|
||||||
}
|
|
||||||
jsonSerializer.serialize(encoder, json)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data
|
|
||||||
|
|
||||||
import dev.inmo.micro_utils.serialization.base64.Base64StringSerializer
|
|
||||||
import dev.inmo.tgbotapi.types.dataField
|
|
||||||
import dev.inmo.tgbotapi.types.passport.EncryptedData
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts.WithData
|
|
||||||
import kotlinx.serialization.SerialName
|
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
|
|
||||||
@Serializable
|
|
||||||
data class EncryptedPersonalDetails(
|
|
||||||
@SerialName(dataField)
|
|
||||||
@Serializable(Base64StringSerializer::class)
|
|
||||||
override val data: EncryptedData,
|
|
||||||
@Serializable(Base64StringSerializer::class)
|
|
||||||
override val hash: String
|
|
||||||
) : WithData
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data
|
|
||||||
|
|
||||||
import dev.inmo.micro_utils.serialization.base64.Base64StringSerializer
|
|
||||||
import dev.inmo.tgbotapi.types.hashField
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts.WithPhoneNumber
|
|
||||||
import dev.inmo.tgbotapi.types.phoneNumberField
|
|
||||||
import kotlinx.serialization.SerialName
|
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
|
|
||||||
@Serializable
|
|
||||||
data class PhoneNumber(
|
|
||||||
@SerialName(phoneNumberField)
|
|
||||||
override val phoneNumber: String,
|
|
||||||
@SerialName(hashField)
|
|
||||||
@Serializable(Base64StringSerializer::class)
|
|
||||||
override val hash: String
|
|
||||||
) : WithPhoneNumber {
|
|
||||||
}
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts
|
|
||||||
|
|
||||||
import dev.inmo.micro_utils.serialization.base64.Base64StringSerializer
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.EncryptedElementSerializer
|
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
import kotlinx.serialization.json.JsonObject
|
|
||||||
|
|
||||||
@Serializable(EncryptedElementSerializer::class)
|
|
||||||
interface EncryptedPassportElement {
|
|
||||||
val hash: String
|
|
||||||
}
|
|
||||||
|
|
||||||
@Serializable(EncryptedElementSerializer::class)
|
|
||||||
data class UnknownEncryptedPassportElement(
|
|
||||||
val rawJson: JsonObject,
|
|
||||||
@Serializable(Base64StringSerializer::class)
|
|
||||||
override val hash: String
|
|
||||||
) : EncryptedPassportElement
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts
|
|
||||||
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.EncryptedElementSerializer
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.PassportFile
|
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
|
|
||||||
@Serializable(EncryptedElementSerializer::class)
|
|
||||||
interface FilesCollection : EncryptedPassportElement {
|
|
||||||
val files: List<PassportFile>
|
|
||||||
}
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts
|
|
||||||
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.EncryptedElementSerializer
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.PassportFile
|
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
|
|
||||||
@Serializable(EncryptedElementSerializer::class)
|
|
||||||
interface Translatable : EncryptedPassportElement {
|
|
||||||
val translations: List<PassportFile>
|
|
||||||
}
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts
|
|
||||||
|
|
||||||
import dev.inmo.tgbotapi.types.passport.EncryptedData
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.EncryptedElementSerializer
|
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
|
|
||||||
@Serializable(EncryptedElementSerializer::class)
|
|
||||||
interface WithData : EncryptedPassportElement {
|
|
||||||
val data: EncryptedData
|
|
||||||
}
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts
|
|
||||||
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.EncryptedElementSerializer
|
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
|
|
||||||
@Serializable(EncryptedElementSerializer::class)
|
|
||||||
interface WithEmail : EncryptedPassportElement {
|
|
||||||
val email: String
|
|
||||||
}
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts
|
|
||||||
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.EncryptedElementSerializer
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.PassportFile
|
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
|
|
||||||
@Serializable(EncryptedElementSerializer::class)
|
|
||||||
interface WithFrontSide : EncryptedPassportElement {
|
|
||||||
val frontSide: PassportFile?
|
|
||||||
}
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts
|
|
||||||
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.EncryptedElementSerializer
|
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
|
|
||||||
@Serializable(EncryptedElementSerializer::class)
|
|
||||||
interface WithPhoneNumber : EncryptedPassportElement {
|
|
||||||
val phoneNumber: String
|
|
||||||
}
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts
|
|
||||||
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.EncryptedElementSerializer
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.PassportFile
|
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
|
|
||||||
@Serializable(EncryptedElementSerializer::class)
|
|
||||||
interface WithReverseSide : EncryptedPassportElement {
|
|
||||||
val reverseSide: PassportFile?
|
|
||||||
}
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
package dev.inmo.tgbotapi.types.passport.encrypted_data.abstracts
|
|
||||||
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.EncryptedElementSerializer
|
|
||||||
import dev.inmo.tgbotapi.types.passport.encrypted_data.PassportFile
|
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
|
|
||||||
@Serializable(EncryptedElementSerializer::class)
|
|
||||||
interface WithSelfie : EncryptedPassportElement {
|
|
||||||
val selfie: PassportFile?
|
|
||||||
}
|
|
||||||
@@ -2,11 +2,12 @@ package dev.inmo.tgbotapi.types.update.MediaGroupUpdates
|
|||||||
|
|
||||||
import dev.inmo.tgbotapi.types.UpdateIdentifier
|
import dev.inmo.tgbotapi.types.UpdateIdentifier
|
||||||
import dev.inmo.tgbotapi.types.message.abstracts.MediaGroupMessage
|
import dev.inmo.tgbotapi.types.message.abstracts.MediaGroupMessage
|
||||||
|
import dev.inmo.tgbotapi.types.message.content.abstracts.MediaGroupContent
|
||||||
import dev.inmo.tgbotapi.types.update.abstracts.BaseMessageUpdate
|
import dev.inmo.tgbotapi.types.update.abstracts.BaseMessageUpdate
|
||||||
|
|
||||||
data class ChannelPostMediaGroupUpdate(
|
data class ChannelPostMediaGroupUpdate(
|
||||||
override val origins: List<BaseMessageUpdate>
|
override val origins: List<BaseMessageUpdate>
|
||||||
) : SentMediaGroupUpdate {
|
) : SentMediaGroupUpdate {
|
||||||
override val updateId: UpdateIdentifier = origins.last().updateId
|
override val updateId: UpdateIdentifier = origins.last().updateId
|
||||||
override val data: List<MediaGroupMessage> = origins.mapNotNull { it.data as? MediaGroupMessage }
|
override val data: List<MediaGroupMessage<MediaGroupContent>> = origins.mapNotNull { it.data as? MediaGroupMessage<MediaGroupContent> }
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,11 +2,12 @@ package dev.inmo.tgbotapi.types.update.MediaGroupUpdates
|
|||||||
|
|
||||||
import dev.inmo.tgbotapi.types.UpdateIdentifier
|
import dev.inmo.tgbotapi.types.UpdateIdentifier
|
||||||
import dev.inmo.tgbotapi.types.message.abstracts.MediaGroupMessage
|
import dev.inmo.tgbotapi.types.message.abstracts.MediaGroupMessage
|
||||||
|
import dev.inmo.tgbotapi.types.message.content.abstracts.MediaGroupContent
|
||||||
import dev.inmo.tgbotapi.types.update.EditChannelPostUpdate
|
import dev.inmo.tgbotapi.types.update.EditChannelPostUpdate
|
||||||
|
|
||||||
data class EditChannelPostMediaGroupUpdate(
|
data class EditChannelPostMediaGroupUpdate(
|
||||||
override val origin: EditChannelPostUpdate
|
override val origin: EditChannelPostUpdate
|
||||||
) : EditMediaGroupUpdate {
|
) : EditMediaGroupUpdate {
|
||||||
override val updateId: UpdateIdentifier = origin.updateId
|
override val updateId: UpdateIdentifier = origin.updateId
|
||||||
override val data: MediaGroupMessage = origin.data as MediaGroupMessage
|
override val data: MediaGroupMessage<MediaGroupContent> = origin.data as MediaGroupMessage<MediaGroupContent>
|
||||||
}
|
}
|
||||||
@@ -2,11 +2,12 @@ package dev.inmo.tgbotapi.types.update.MediaGroupUpdates
|
|||||||
|
|
||||||
import dev.inmo.tgbotapi.types.UpdateIdentifier
|
import dev.inmo.tgbotapi.types.UpdateIdentifier
|
||||||
import dev.inmo.tgbotapi.types.message.abstracts.MediaGroupMessage
|
import dev.inmo.tgbotapi.types.message.abstracts.MediaGroupMessage
|
||||||
|
import dev.inmo.tgbotapi.types.message.content.abstracts.MediaGroupContent
|
||||||
import dev.inmo.tgbotapi.types.update.EditMessageUpdate
|
import dev.inmo.tgbotapi.types.update.EditMessageUpdate
|
||||||
|
|
||||||
data class EditMessageMediaGroupUpdate(
|
data class EditMessageMediaGroupUpdate(
|
||||||
override val origin: EditMessageUpdate
|
override val origin: EditMessageUpdate
|
||||||
) : EditMediaGroupUpdate {
|
) : EditMediaGroupUpdate {
|
||||||
override val updateId: UpdateIdentifier = origin.updateId
|
override val updateId: UpdateIdentifier = origin.updateId
|
||||||
override val data: MediaGroupMessage = origin.data as MediaGroupMessage
|
override val data: MediaGroupMessage<MediaGroupContent> = origin.data as MediaGroupMessage<MediaGroupContent>
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
package dev.inmo.tgbotapi.types.update.MediaGroupUpdates
|
package dev.inmo.tgbotapi.types.update.MediaGroupUpdates
|
||||||
|
|
||||||
import dev.inmo.tgbotapi.types.message.abstracts.MediaGroupMessage
|
import dev.inmo.tgbotapi.types.message.abstracts.MediaGroupMessage
|
||||||
|
import dev.inmo.tgbotapi.types.message.content.abstracts.MediaGroupContent
|
||||||
import dev.inmo.tgbotapi.types.update.abstracts.*
|
import dev.inmo.tgbotapi.types.update.abstracts.*
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -13,11 +14,11 @@ import dev.inmo.tgbotapi.types.update.abstracts.*
|
|||||||
interface MediaGroupUpdate : Update
|
interface MediaGroupUpdate : Update
|
||||||
|
|
||||||
interface SentMediaGroupUpdate: MediaGroupUpdate {
|
interface SentMediaGroupUpdate: MediaGroupUpdate {
|
||||||
override val data: List<MediaGroupMessage>
|
override val data: List<MediaGroupMessage<MediaGroupContent>>
|
||||||
val origins: List<BaseMessageUpdate>
|
val origins: List<BaseMessageUpdate>
|
||||||
}
|
}
|
||||||
|
|
||||||
interface EditMediaGroupUpdate : BaseEditMessageUpdate, MediaGroupUpdate {
|
interface EditMediaGroupUpdate : BaseEditMessageUpdate, MediaGroupUpdate {
|
||||||
override val data: MediaGroupMessage
|
override val data: MediaGroupMessage<MediaGroupContent>
|
||||||
val origin: BaseMessageUpdate
|
val origin: BaseMessageUpdate
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,11 +2,12 @@ package dev.inmo.tgbotapi.types.update.MediaGroupUpdates
|
|||||||
|
|
||||||
import dev.inmo.tgbotapi.types.UpdateIdentifier
|
import dev.inmo.tgbotapi.types.UpdateIdentifier
|
||||||
import dev.inmo.tgbotapi.types.message.abstracts.MediaGroupMessage
|
import dev.inmo.tgbotapi.types.message.abstracts.MediaGroupMessage
|
||||||
|
import dev.inmo.tgbotapi.types.message.content.abstracts.MediaGroupContent
|
||||||
import dev.inmo.tgbotapi.types.update.abstracts.BaseMessageUpdate
|
import dev.inmo.tgbotapi.types.update.abstracts.BaseMessageUpdate
|
||||||
|
|
||||||
data class MessageMediaGroupUpdate(
|
data class MessageMediaGroupUpdate(
|
||||||
override val origins: List<BaseMessageUpdate>
|
override val origins: List<BaseMessageUpdate>
|
||||||
) : SentMediaGroupUpdate {
|
) : SentMediaGroupUpdate {
|
||||||
override val updateId: UpdateIdentifier = origins.last().updateId
|
override val updateId: UpdateIdentifier = origins.last().updateId
|
||||||
override val data: List<MediaGroupMessage> = origins.mapNotNull { it.data as? MediaGroupMessage }
|
override val data: List<MediaGroupMessage<MediaGroupContent>> = origins.mapNotNull { it.data as? MediaGroupMessage<MediaGroupContent> }
|
||||||
}
|
}
|
||||||
@@ -7,15 +7,47 @@ import dev.inmo.tgbotapi.types.update.abstracts.UnknownUpdate
|
|||||||
import dev.inmo.tgbotapi.types.update.abstracts.Update
|
import dev.inmo.tgbotapi.types.update.abstracts.Update
|
||||||
import kotlinx.coroutines.flow.*
|
import kotlinx.coroutines.flow.*
|
||||||
|
|
||||||
@Suppress("EXPERIMENTAL_API_USAGE", "unused")
|
interface FlowsUpdatesFilter : UpdatesFilter {
|
||||||
class FlowsUpdatesFilter(
|
override val allowedUpdates: List<String>
|
||||||
|
get() = ALL_UPDATES_LIST
|
||||||
|
val allUpdatesFlow: Flow<Update>
|
||||||
|
val allUpdatesWithoutMediaGroupsGroupingFlow: Flow<Update>
|
||||||
|
|
||||||
|
val messageFlow: Flow<MessageUpdate>
|
||||||
|
val messageMediaGroupFlow: Flow<MessageMediaGroupUpdate>
|
||||||
|
val editedMessageFlow: Flow<EditMessageUpdate>
|
||||||
|
val editedMessageMediaGroupFlow: Flow<EditMessageMediaGroupUpdate>
|
||||||
|
val channelPostFlow: Flow<ChannelPostUpdate>
|
||||||
|
val channelPostMediaGroupFlow: Flow<ChannelPostMediaGroupUpdate>
|
||||||
|
val editedChannelPostFlow: Flow<EditChannelPostUpdate>
|
||||||
|
val editedChannelPostMediaGroupFlow: Flow<EditChannelPostMediaGroupUpdate>
|
||||||
|
val chosenInlineResultFlow: Flow<ChosenInlineResultUpdate>
|
||||||
|
val inlineQueryFlow: Flow<InlineQueryUpdate>
|
||||||
|
val callbackQueryFlow: Flow<CallbackQueryUpdate>
|
||||||
|
val shippingQueryFlow: Flow<ShippingQueryUpdate>
|
||||||
|
val preCheckoutQueryFlow: Flow<PreCheckoutQueryUpdate>
|
||||||
|
val pollFlow: Flow<PollUpdate>
|
||||||
|
val pollAnswerFlow: Flow<PollAnswerUpdate>
|
||||||
|
val unknownUpdateTypeFlow: Flow<UnknownUpdate>
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates [DefaultFlowsUpdatesFilter]
|
||||||
|
*/
|
||||||
|
@Suppress("FunctionName")
|
||||||
|
fun FlowsUpdatesFilter(
|
||||||
broadcastChannelsSize: Int = 100
|
broadcastChannelsSize: Int = 100
|
||||||
): UpdatesFilter {
|
) = DefaultFlowsUpdatesFilter(broadcastChannelsSize)
|
||||||
|
|
||||||
|
@Suppress("EXPERIMENTAL_API_USAGE", "unused")
|
||||||
|
class DefaultFlowsUpdatesFilter(
|
||||||
|
broadcastChannelsSize: Int = 100
|
||||||
|
): FlowsUpdatesFilter {
|
||||||
private val updatesSharedFlow = MutableSharedFlow<Update>(extraBufferCapacity = broadcastChannelsSize)
|
private val updatesSharedFlow = MutableSharedFlow<Update>(extraBufferCapacity = broadcastChannelsSize)
|
||||||
@Suppress("MemberVisibilityCanBePrivate")
|
@Suppress("MemberVisibilityCanBePrivate")
|
||||||
val allUpdatesFlow: Flow<Update> = updatesSharedFlow.asSharedFlow()
|
override val allUpdatesFlow: Flow<Update> = updatesSharedFlow.asSharedFlow()
|
||||||
@Suppress("MemberVisibilityCanBePrivate")
|
@Suppress("MemberVisibilityCanBePrivate")
|
||||||
val allUpdatesWithoutMediaGroupsGroupingFlow: Flow<Update> = updatesSharedFlow.flatMapConcat {
|
override val allUpdatesWithoutMediaGroupsGroupingFlow: Flow<Update> = allUpdatesFlow.flatMapConcat {
|
||||||
when (it) {
|
when (it) {
|
||||||
is SentMediaGroupUpdate -> it.origins.asFlow()
|
is SentMediaGroupUpdate -> it.origins.asFlow()
|
||||||
is EditMediaGroupUpdate -> flowOf(it.origin)
|
is EditMediaGroupUpdate -> flowOf(it.origin)
|
||||||
@@ -23,26 +55,24 @@ class FlowsUpdatesFilter(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override val allowedUpdates: List<String>
|
|
||||||
get() = ALL_UPDATES_LIST
|
|
||||||
override val asUpdateReceiver: UpdateReceiver<Update> = {
|
override val asUpdateReceiver: UpdateReceiver<Update> = {
|
||||||
updatesSharedFlow.emit(it)
|
updatesSharedFlow.emit(it)
|
||||||
}
|
}
|
||||||
|
|
||||||
val messageFlow: Flow<MessageUpdate> = allUpdatesFlow.filterIsInstance()
|
override val messageFlow: Flow<MessageUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
val messageMediaGroupFlow: Flow<MessageMediaGroupUpdate> = allUpdatesFlow.filterIsInstance()
|
override val messageMediaGroupFlow: Flow<MessageMediaGroupUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
val editedMessageFlow: Flow<EditMessageUpdate> = allUpdatesFlow.filterIsInstance()
|
override val editedMessageFlow: Flow<EditMessageUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
val editedMessageMediaGroupFlow: Flow<EditMessageMediaGroupUpdate> = allUpdatesFlow.filterIsInstance()
|
override val editedMessageMediaGroupFlow: Flow<EditMessageMediaGroupUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
val channelPostFlow: Flow<ChannelPostUpdate> = allUpdatesFlow.filterIsInstance()
|
override val channelPostFlow: Flow<ChannelPostUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
val channelPostMediaGroupFlow: Flow<ChannelPostMediaGroupUpdate> = allUpdatesFlow.filterIsInstance()
|
override val channelPostMediaGroupFlow: Flow<ChannelPostMediaGroupUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
val editedChannelPostFlow: Flow<EditChannelPostUpdate> = allUpdatesFlow.filterIsInstance()
|
override val editedChannelPostFlow: Flow<EditChannelPostUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
val editedChannelPostMediaGroupFlow: Flow<EditChannelPostMediaGroupUpdate> = allUpdatesFlow.filterIsInstance()
|
override val editedChannelPostMediaGroupFlow: Flow<EditChannelPostMediaGroupUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
val chosenInlineResultFlow: Flow<ChosenInlineResultUpdate> = allUpdatesFlow.filterIsInstance()
|
override val chosenInlineResultFlow: Flow<ChosenInlineResultUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
val inlineQueryFlow: Flow<InlineQueryUpdate> = allUpdatesFlow.filterIsInstance()
|
override val inlineQueryFlow: Flow<InlineQueryUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
val callbackQueryFlow: Flow<CallbackQueryUpdate> = allUpdatesFlow.filterIsInstance()
|
override val callbackQueryFlow: Flow<CallbackQueryUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
val shippingQueryFlow: Flow<ShippingQueryUpdate> = allUpdatesFlow.filterIsInstance()
|
override val shippingQueryFlow: Flow<ShippingQueryUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
val preCheckoutQueryFlow: Flow<PreCheckoutQueryUpdate> = allUpdatesFlow.filterIsInstance()
|
override val preCheckoutQueryFlow: Flow<PreCheckoutQueryUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
val pollFlow: Flow<PollUpdate> = allUpdatesFlow.filterIsInstance()
|
override val pollFlow: Flow<PollUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
val pollAnswerFlow: Flow<PollAnswerUpdate> = allUpdatesFlow.filterIsInstance()
|
override val pollAnswerFlow: Flow<PollAnswerUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
val unknownUpdateTypeFlow: Flow<UnknownUpdate> = allUpdatesFlow.filterIsInstance()
|
override val unknownUpdateTypeFlow: Flow<UnknownUpdate> = allUpdatesFlow.filterIsInstance()
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
package dev.inmo.tgbotapi.utils.passport
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.crypto.SourceBytes
|
||||||
|
import dev.inmo.tgbotapi.bot.TelegramBot
|
||||||
|
import dev.inmo.tgbotapi.requests.DownloadFile
|
||||||
|
import dev.inmo.tgbotapi.requests.get.GetFile
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.EncryptedCredentials
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.EncryptedData
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.PassportFile
|
||||||
|
import dev.inmo.tgbotapi.utils.nonstrictJsonFormat
|
||||||
|
import kotlinx.serialization.json.JsonObject
|
||||||
|
|
||||||
|
interface Decryptor {
|
||||||
|
fun decrypt(data: EncryptedData): SourceBytes
|
||||||
|
}
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
package dev.inmo.tgbotapi.types.passport
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.crypto.MD5
|
||||||
|
import java.io.File
|
||||||
|
|
||||||
|
val File.passportFileHash: MD5
|
||||||
|
get() = readBytes().passportFileHash
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
package dev.inmo.tgbotapi.utils.passport
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.crypto.SourceBytes
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.EncryptedData
|
||||||
|
import javax.crypto.Cipher
|
||||||
|
import javax.crypto.spec.IvParameterSpec
|
||||||
|
import javax.crypto.spec.SecretKeySpec
|
||||||
|
|
||||||
|
class AESDecryptor(key: SourceBytes, private val iv: ByteArray) : Decryptor {
|
||||||
|
private val key = SecretKeySpec(key, "AES");
|
||||||
|
|
||||||
|
override fun decrypt(data: EncryptedData): SourceBytes {
|
||||||
|
return Cipher.getInstance("AES/CBC/NOPADDING").run {
|
||||||
|
init(Cipher.DECRYPT_MODE, key, IvParameterSpec(this@AESDecryptor.iv))
|
||||||
|
val decryptedCredentials = doFinal(data)
|
||||||
|
|
||||||
|
val padding = decryptedCredentials.first()
|
||||||
|
decryptedCredentials.copyOfRange(padding.toInt(), decryptedCredentials.size)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,29 @@
|
|||||||
|
package dev.inmo.tgbotapi.utils.passport
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.crypto.decodeBase64
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.DecryptedCredentials
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.EncryptedCredentials
|
||||||
|
import dev.inmo.tgbotapi.utils.nonstrictJsonFormat
|
||||||
|
import java.security.*
|
||||||
|
import java.security.spec.PKCS8EncodedKeySpec
|
||||||
|
import javax.crypto.Cipher
|
||||||
|
import javax.crypto.spec.IvParameterSpec
|
||||||
|
import javax.crypto.spec.SecretKeySpec
|
||||||
|
|
||||||
|
private val regexToRemoveFromKey = Regex("(-----(BEGIN|END) ((?:.*? KEY)|CERTIFICATE)-----|[\\s])")
|
||||||
|
|
||||||
|
fun EncryptedCredentials.decryptWithPKCS8PrivateKey(privateKey: PrivateKey): DecryptedCredentials {
|
||||||
|
val decrypted = Cipher.getInstance("RSA/ECB/OAEPWithSHA-1AndMGF1Padding").run {
|
||||||
|
init(Cipher.DECRYPT_MODE, privateKey)
|
||||||
|
doFinal(secret)
|
||||||
|
}
|
||||||
|
val dataDecryptor = (decrypted to hash).createDecryptor()
|
||||||
|
val decryptedCredentials = dataDecryptor.decrypt(data).decodeToString()
|
||||||
|
return nonstrictJsonFormat.decodeFromString(DecryptedCredentials.serializer(), decryptedCredentials)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun EncryptedCredentials.decryptWithPKCS8PrivateKey(key: String) = decryptWithPKCS8PrivateKey(
|
||||||
|
KeyFactory.getInstance("RSA").generatePrivate(
|
||||||
|
PKCS8EncodedKeySpec(key.replace(regexToRemoveFromKey, "").decodeBase64())
|
||||||
|
)
|
||||||
|
)
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
package dev.inmo.tgbotapi.utils.passport
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.crypto.SourceBytes
|
||||||
|
import dev.inmo.tgbotapi.bot.TelegramBot
|
||||||
|
import dev.inmo.tgbotapi.requests.DownloadFile
|
||||||
|
import dev.inmo.tgbotapi.requests.get.GetFile
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.*
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.PassportFile
|
||||||
|
|
||||||
|
fun EndDataCredentials.decryptData(
|
||||||
|
bytes: EncryptedData
|
||||||
|
): SourceBytes {
|
||||||
|
return createDecryptor().decrypt(bytes)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun FileCredentials.decryptFile(
|
||||||
|
fileBytes: ByteArray
|
||||||
|
): SourceBytes {
|
||||||
|
return createDecryptor().decrypt(fileBytes)
|
||||||
|
}
|
||||||
|
suspend fun FileCredentials.decryptFile(
|
||||||
|
bot: TelegramBot,
|
||||||
|
passportFile: PassportFile
|
||||||
|
): SourceBytes {
|
||||||
|
val pathedFile = bot.execute(GetFile(passportFile.fileId))
|
||||||
|
val bytes = bot.execute(DownloadFile(pathedFile.filePath))
|
||||||
|
return decryptFile(bytes)
|
||||||
|
}
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
package dev.inmo.tgbotapi.utils.passport
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.crypto.SourceBytes
|
||||||
|
import dev.inmo.tgbotapi.types.passport.credentials.EndDataCredentials
|
||||||
|
import java.security.MessageDigest
|
||||||
|
|
||||||
|
fun Pair<SourceBytes, SourceBytes>.createDecryptor(): Decryptor {
|
||||||
|
val secretHash = MessageDigest.getInstance("SHA-512").digest(first + second)
|
||||||
|
val key = secretHash.copyOf(32)
|
||||||
|
val iv = secretHash.copyOfRange(32, 48)
|
||||||
|
|
||||||
|
return AESDecryptor(key, iv)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun EndDataCredentials.createDecryptor() = (secret to hash).createDecryptor()
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
package dev.inmo.tgbotapi.utils.passport
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.types.passport.PassportData
|
||||||
|
import dev.inmo.tgbotapi.types.passport.decrypted.SecureData
|
||||||
|
import java.security.PrivateKey
|
||||||
|
|
||||||
|
inline fun <T> PassportData.doInDecryptionContextWithPKCS8Key(
|
||||||
|
pkcs8Key: PrivateKey,
|
||||||
|
expectedNonce: String? = null,
|
||||||
|
crossinline block: SecureData.() -> T
|
||||||
|
): T {
|
||||||
|
val decryptedCredentials = credentials.decryptWithPKCS8PrivateKey(pkcs8Key)
|
||||||
|
expectedNonce ?.let { require(expectedNonce == decryptedCredentials.nonce) }
|
||||||
|
return decryptedCredentials.secureData.run(block)
|
||||||
|
}
|
||||||
|
inline fun <T> PassportData.doInDecryptionContextWithPKCS8Key(
|
||||||
|
pkcs8Key: String,
|
||||||
|
expectedNonce: String? = null,
|
||||||
|
crossinline block: SecureData.() -> T
|
||||||
|
): T {
|
||||||
|
val decryptedCredentials = credentials.decryptWithPKCS8PrivateKey(pkcs8Key)
|
||||||
|
expectedNonce ?.let { require(expectedNonce == decryptedCredentials.nonce) }
|
||||||
|
return decryptedCredentials.secureData.run(block)
|
||||||
|
}
|
||||||
@@ -23,7 +23,7 @@ internal fun List<Update>.convertWithMediaGroupUpdates(): List<Update> {
|
|||||||
val resultUpdates = mutableListOf<Update>()
|
val resultUpdates = mutableListOf<Update>()
|
||||||
val mediaGroups = mutableMapOf<MediaGroupIdentifier, MutableList<BaseSentMessageUpdate>>()
|
val mediaGroups = mutableMapOf<MediaGroupIdentifier, MutableList<BaseSentMessageUpdate>>()
|
||||||
for (update in this) {
|
for (update in this) {
|
||||||
val data = (update.data as? MediaGroupMessage)
|
val data = (update.data as? MediaGroupMessage<*>)
|
||||||
if (data == null) {
|
if (data == null) {
|
||||||
resultUpdates.add(update)
|
resultUpdates.add(update)
|
||||||
continue
|
continue
|
||||||
|
|||||||
@@ -0,0 +1,45 @@
|
|||||||
|
package dev.inmo.tgbotapi.extensions.api.passport
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.bot.TelegramBot
|
||||||
|
import dev.inmo.tgbotapi.requests.SetPassportDataErrors
|
||||||
|
import dev.inmo.tgbotapi.types.*
|
||||||
|
import dev.inmo.tgbotapi.types.message.PassportMessage
|
||||||
|
import dev.inmo.tgbotapi.types.passport.PassportData
|
||||||
|
import dev.inmo.tgbotapi.types.passport.PassportElementError
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.EncryptedPassportElement
|
||||||
|
import dev.inmo.tgbotapi.utils.passport.Decryptor
|
||||||
|
|
||||||
|
suspend fun TelegramBot.setPassportDataErrors(
|
||||||
|
userId: UserId,
|
||||||
|
errors: List<PassportElementError>
|
||||||
|
) = execute(SetPassportDataErrors(userId, errors))
|
||||||
|
suspend fun TelegramBot.setPassportDataErrors(
|
||||||
|
user: User,
|
||||||
|
errors: List<PassportElementError>
|
||||||
|
) = setPassportDataErrors(user.id, errors)
|
||||||
|
|
||||||
|
suspend fun TelegramBot.setPassportDataErrors(
|
||||||
|
userId: UserId,
|
||||||
|
passportData: PassportData,
|
||||||
|
decryptor: Decryptor,
|
||||||
|
mapper: suspend Decryptor.(EncryptedPassportElement) -> PassportElementError
|
||||||
|
): Boolean = setPassportDataErrors(
|
||||||
|
userId,
|
||||||
|
passportData.data.map { decryptor.mapper(it) }.also {
|
||||||
|
if (it.isEmpty()) {
|
||||||
|
return@setPassportDataErrors false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
suspend fun TelegramBot.setPassportDataErrors(
|
||||||
|
user: User,
|
||||||
|
passportData: PassportData,
|
||||||
|
decryptor: Decryptor,
|
||||||
|
mapper: suspend Decryptor.(EncryptedPassportElement) -> PassportElementError
|
||||||
|
) = setPassportDataErrors(user.id, passportData, decryptor, mapper)
|
||||||
|
|
||||||
|
suspend fun TelegramBot.setPassportDataErrors(
|
||||||
|
passportMessage: PassportMessage,
|
||||||
|
decryptor: Decryptor,
|
||||||
|
mapper: suspend Decryptor.(EncryptedPassportElement) -> PassportElementError
|
||||||
|
) = setPassportDataErrors(passportMessage.user, passportMessage.passportData, decryptor, mapper)
|
||||||
@@ -7,6 +7,8 @@ import dev.inmo.tgbotapi.types.InputMedia.*
|
|||||||
import dev.inmo.tgbotapi.types.MessageIdentifier
|
import dev.inmo.tgbotapi.types.MessageIdentifier
|
||||||
import dev.inmo.tgbotapi.types.chat.abstracts.Chat
|
import dev.inmo.tgbotapi.types.chat.abstracts.Chat
|
||||||
import dev.inmo.tgbotapi.types.message.abstracts.Message
|
import dev.inmo.tgbotapi.types.message.abstracts.Message
|
||||||
|
import dev.inmo.tgbotapi.types.message.content.abstracts.MediaGroupContent
|
||||||
|
import dev.inmo.tgbotapi.types.message.content.abstracts.VisualMediaGroupContent
|
||||||
import dev.inmo.tgbotapi.utils.RiskFeature
|
import dev.inmo.tgbotapi.utils.RiskFeature
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -20,7 +22,7 @@ suspend fun TelegramBot.sendMediaGroup(
|
|||||||
replyToMessageId: MessageIdentifier? = null,
|
replyToMessageId: MessageIdentifier? = null,
|
||||||
allowSendingWithoutReply: Boolean? = null
|
allowSendingWithoutReply: Boolean? = null
|
||||||
) = execute(
|
) = execute(
|
||||||
SendMediaGroup(
|
SendMediaGroup<MediaGroupContent>(
|
||||||
chatId, media, disableNotification, replyToMessageId, allowSendingWithoutReply
|
chatId, media, disableNotification, replyToMessageId, allowSendingWithoutReply
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ fun CoroutineScope.updateHandlerWithMediaGroupsAdaptation(
|
|||||||
launch {
|
launch {
|
||||||
for (update in updatesChannel) {
|
for (update in updatesChannel) {
|
||||||
when (val data = update.data) {
|
when (val data = update.data) {
|
||||||
is MediaGroupMessage -> mediaGroupChannel.send("${data.mediaGroupId}${update::class.simpleName}" to update as BaseMessageUpdate)
|
is MediaGroupMessage<*> -> mediaGroupChannel.send("${data.mediaGroupId}${update::class.simpleName}" to update as BaseMessageUpdate)
|
||||||
else -> output(update)
|
else -> output(update)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +1,11 @@
|
|||||||
package dev.inmo.tgbotapi.extensions.behaviour_builder
|
package dev.inmo.tgbotapi.extensions.behaviour_builder
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.coroutines.subscribeSafelyWithoutExceptions
|
||||||
import dev.inmo.tgbotapi.bot.TelegramBot
|
import dev.inmo.tgbotapi.bot.TelegramBot
|
||||||
|
import dev.inmo.tgbotapi.types.update.abstracts.Update
|
||||||
import dev.inmo.tgbotapi.updateshandlers.FlowsUpdatesFilter
|
import dev.inmo.tgbotapi.updateshandlers.FlowsUpdatesFilter
|
||||||
import dev.inmo.tgbotapi.updateshandlers.UpdatesFilter
|
import kotlinx.coroutines.*
|
||||||
import kotlinx.coroutines.CoroutineScope
|
import kotlinx.coroutines.flow.filter
|
||||||
|
|
||||||
typealias BehaviourContextReceiver<T> = suspend BehaviourContext.() -> T
|
typealias BehaviourContextReceiver<T> = suspend BehaviourContext.() -> T
|
||||||
typealias BehaviourContextAndTypeReceiver<T, I> = suspend BehaviourContext.(I) -> T
|
typealias BehaviourContextAndTypeReceiver<T, I> = suspend BehaviourContext.(I) -> T
|
||||||
@@ -19,4 +21,48 @@ data class BehaviourContext(
|
|||||||
val bot: TelegramBot,
|
val bot: TelegramBot,
|
||||||
val scope: CoroutineScope,
|
val scope: CoroutineScope,
|
||||||
val flowsUpdatesFilter: FlowsUpdatesFilter = FlowsUpdatesFilter()
|
val flowsUpdatesFilter: FlowsUpdatesFilter = FlowsUpdatesFilter()
|
||||||
) : UpdatesFilter by flowsUpdatesFilter, TelegramBot by bot, CoroutineScope by scope
|
) : FlowsUpdatesFilter by flowsUpdatesFilter, TelegramBot by bot, CoroutineScope by scope
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates new one [BehaviourContext], adding subsequent [FlowsUpdatesFilter] in case [newFlowsUpdatesFilterSetUp] is provided and
|
||||||
|
* [CoroutineScope] as new [BehaviourContext.scope]
|
||||||
|
*
|
||||||
|
* @param newFlowsUpdatesFilterSetUp As a parameter receives [FlowsUpdatesFilter] from old [this] [BehaviourContext.flowsUpdatesFilter]
|
||||||
|
*/
|
||||||
|
suspend fun <T> BehaviourContext.doInSubContextWithFlowsUpdatesFilterSetup(
|
||||||
|
newFlowsUpdatesFilterSetUp: BehaviourContextAndTypeReceiver<Unit, FlowsUpdatesFilter>?,
|
||||||
|
behaviourContextReceiver: BehaviourContextReceiver<T>
|
||||||
|
) = copy(
|
||||||
|
flowsUpdatesFilter = FlowsUpdatesFilter(),
|
||||||
|
scope = CoroutineScope(scope.coroutineContext + SupervisorJob())
|
||||||
|
).run {
|
||||||
|
newFlowsUpdatesFilterSetUp ?.let {
|
||||||
|
it.apply { invoke(this@run, this@doInSubContextWithFlowsUpdatesFilterSetup.flowsUpdatesFilter) }
|
||||||
|
}
|
||||||
|
behaviourContextReceiver().also { stop() }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates new one [BehaviourContext], adding subsequent [FlowsUpdatesFilter] in case [updatesFilter] is provided and
|
||||||
|
* [CoroutineScope] as new [BehaviourContext.scope]
|
||||||
|
*/
|
||||||
|
suspend fun <T> BehaviourContext.doInSubContextWithUpdatesFilter(
|
||||||
|
updatesFilter: BehaviourContextAndTypeReceiver<Boolean, Update>?,
|
||||||
|
behaviourContextReceiver: BehaviourContextReceiver<T>
|
||||||
|
) = doInSubContextWithFlowsUpdatesFilterSetup(
|
||||||
|
newFlowsUpdatesFilterSetUp = updatesFilter ?.let {
|
||||||
|
{ oldOne ->
|
||||||
|
oldOne.allUpdatesFlow.filter { updatesFilter(it) }.subscribeSafelyWithoutExceptions(scope, asUpdateReceiver)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
behaviourContextReceiver
|
||||||
|
)
|
||||||
|
|
||||||
|
suspend fun <T> BehaviourContext.doInSubContext(
|
||||||
|
behaviourContextReceiver: BehaviourContextReceiver<T>
|
||||||
|
) = doInSubContextWithFlowsUpdatesFilterSetup(newFlowsUpdatesFilterSetUp = null, behaviourContextReceiver)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method will cancel ALL subsequent contexts, expectations and waiters
|
||||||
|
*/
|
||||||
|
fun BehaviourContext.stop() = scope.cancel()
|
||||||
|
|||||||
@@ -0,0 +1,31 @@
|
|||||||
|
package dev.inmo.tgbotapi.extensions.behaviour_builder
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.coroutines.DeferredAction
|
||||||
|
import dev.inmo.micro_utils.coroutines.invokeFirstOf
|
||||||
|
import kotlinx.coroutines.*
|
||||||
|
|
||||||
|
suspend fun <T> BehaviourContext.parallel(
|
||||||
|
action: BehaviourContextReceiver<T>
|
||||||
|
) = async {
|
||||||
|
action()
|
||||||
|
}
|
||||||
|
|
||||||
|
inline infix fun <T, O> Deferred<T>.withAction(noinline callback: suspend (T) -> O) = DeferredAction(this, callback)
|
||||||
|
|
||||||
|
inline fun <T> Deferred<T>.asAction() = DeferredAction(this) { it }
|
||||||
|
|
||||||
|
suspend fun <O> BehaviourContext.oneOfActions(
|
||||||
|
deferredActions: Iterable<DeferredAction<*, O>>
|
||||||
|
) = deferredActions.invokeFirstOf(scope)
|
||||||
|
|
||||||
|
suspend fun <O> BehaviourContext.oneOfActions(
|
||||||
|
vararg deferredActions: DeferredAction<*, O>
|
||||||
|
) = this@oneOfActions.oneOfActions(deferredActions.toList())
|
||||||
|
|
||||||
|
suspend fun <O> BehaviourContext.oneOf(
|
||||||
|
deferredActions: Iterable<Deferred<O>>
|
||||||
|
) = oneOfActions(deferredActions.map { it.asAction() })
|
||||||
|
|
||||||
|
suspend fun <O> BehaviourContext.oneOf(
|
||||||
|
vararg deferredActions: Deferred<O>
|
||||||
|
) = oneOf(deferredActions.toList())
|
||||||
@@ -24,7 +24,7 @@ private suspend fun <O> BehaviourContext.waitCallbackQueries(
|
|||||||
}.toList().toList()
|
}.toList().toList()
|
||||||
|
|
||||||
|
|
||||||
private suspend inline fun <reified T : CallbackQuery> BehaviourContext.waitEvents(
|
private suspend inline fun <reified T : CallbackQuery> BehaviourContext.waitCallbacks(
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
noinline errorFactory: NullableRequestBuilder<*> = { null },
|
noinline errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
@@ -51,52 +51,52 @@ suspend fun BehaviourContext.waitDataCallbackQuery(
|
|||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
filter: CallbackQueryMapper<DataCallbackQuery>? = null
|
filter: CallbackQueryMapper<DataCallbackQuery>? = null
|
||||||
) = waitEvents(count, initRequest, errorFactory, filter)
|
) = waitCallbacks(count, initRequest, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitGameShortNameCallbackQuery(
|
suspend fun BehaviourContext.waitGameShortNameCallbackQuery(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
filter: CallbackQueryMapper<GameShortNameCallbackQuery>? = null
|
filter: CallbackQueryMapper<GameShortNameCallbackQuery>? = null
|
||||||
) = waitEvents(count, initRequest, errorFactory, filter)
|
) = waitCallbacks(count, initRequest, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitInlineMessageIdCallbackQuery(
|
suspend fun BehaviourContext.waitInlineMessageIdCallbackQuery(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
filter: CallbackQueryMapper<InlineMessageIdCallbackQuery>? = null
|
filter: CallbackQueryMapper<InlineMessageIdCallbackQuery>? = null
|
||||||
) = waitEvents(count, initRequest, errorFactory, filter)
|
) = waitCallbacks(count, initRequest, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitInlineMessageIdDataCallbackQuery(
|
suspend fun BehaviourContext.waitInlineMessageIdDataCallbackQuery(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
filter: CallbackQueryMapper<InlineMessageIdDataCallbackQuery>? = null
|
filter: CallbackQueryMapper<InlineMessageIdDataCallbackQuery>? = null
|
||||||
) = waitEvents(count, initRequest, errorFactory, filter)
|
) = waitCallbacks(count, initRequest, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitInlineMessageIdGameShortNameCallbackQuery(
|
suspend fun BehaviourContext.waitInlineMessageIdGameShortNameCallbackQuery(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
filter: CallbackQueryMapper<InlineMessageIdGameShortNameCallbackQuery>? = null
|
filter: CallbackQueryMapper<InlineMessageIdGameShortNameCallbackQuery>? = null
|
||||||
) = waitEvents(count, initRequest, errorFactory, filter)
|
) = waitCallbacks(count, initRequest, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitMessageCallbackQuery(
|
suspend fun BehaviourContext.waitMessageCallbackQuery(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
filter: CallbackQueryMapper<MessageCallbackQuery>? = null
|
filter: CallbackQueryMapper<MessageCallbackQuery>? = null
|
||||||
) = waitEvents(count, initRequest, errorFactory, filter)
|
) = waitCallbacks(count, initRequest, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitMessageDataCallbackQuery(
|
suspend fun BehaviourContext.waitMessageDataCallbackQuery(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
filter: CallbackQueryMapper<MessageDataCallbackQuery>? = null
|
filter: CallbackQueryMapper<MessageDataCallbackQuery>? = null
|
||||||
) = waitEvents(count, initRequest, errorFactory, filter)
|
) = waitCallbacks(count, initRequest, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitMessageGameShortNameCallbackQuery(
|
suspend fun BehaviourContext.waitMessageGameShortNameCallbackQuery(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
filter: CallbackQueryMapper<MessageGameShortNameCallbackQuery>? = null
|
filter: CallbackQueryMapper<MessageGameShortNameCallbackQuery>? = null
|
||||||
) = waitEvents(count, initRequest, errorFactory, filter)
|
) = waitCallbacks(count, initRequest, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitUnknownCallbackQuery(
|
suspend fun BehaviourContext.waitUnknownCallbackQuery(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
filter: CallbackQueryMapper<UnknownCallbackQueryType>? = null
|
filter: CallbackQueryMapper<UnknownCallbackQueryType>? = null
|
||||||
) = waitEvents(count, initRequest, errorFactory, filter)
|
) = waitCallbacks(count, initRequest, errorFactory, filter)
|
||||||
|
|||||||
@@ -59,6 +59,12 @@ private suspend inline fun <reified T : MessageContent> BehaviourContext.waitCon
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
suspend fun BehaviourContext.waitContentMessage(
|
||||||
|
initRequest: Request<*>? = null,
|
||||||
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
|
count: Int = 1,
|
||||||
|
filter: CommonMessageToContentMapper<MessageContent>? = null
|
||||||
|
) = waitContent(count, initRequest, false, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitContact(
|
suspend fun BehaviourContext.waitContact(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
@@ -101,14 +107,14 @@ suspend fun BehaviourContext.waitVenue(
|
|||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
filter: CommonMessageToContentMapper<VenueContent>? = null
|
filter: CommonMessageToContentMapper<VenueContent>? = null
|
||||||
) = waitContent(count, initRequest, false, errorFactory, filter)
|
) = waitContent(count, initRequest, false, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitAudioMediaGroup(
|
suspend fun BehaviourContext.waitAudioMediaGroupContent(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
includeMediaGroups: Boolean = true,
|
includeMediaGroups: Boolean = true,
|
||||||
filter: CommonMessageToContentMapper<AudioMediaGroupContent>? = null
|
filter: CommonMessageToContentMapper<AudioMediaGroupContent>? = null
|
||||||
) = waitContent(count, initRequest, includeMediaGroups, errorFactory, filter)
|
) = waitContent(count, initRequest, includeMediaGroups, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitDocumentMediaGroup(
|
suspend fun BehaviourContext.waitDocumentMediaGroupContent(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
@@ -119,17 +125,17 @@ suspend fun BehaviourContext.waitMedia(
|
|||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
includeMediaGroups: Boolean = true,
|
includeMediaGroups: Boolean = false,
|
||||||
filter: CommonMessageToContentMapper<MediaContent>? = null
|
filter: CommonMessageToContentMapper<MediaContent>? = null
|
||||||
) = waitContent(count, initRequest, includeMediaGroups, errorFactory, filter)
|
) = waitContent(count, initRequest, includeMediaGroups, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitMediaGroup(
|
suspend fun BehaviourContext.waitAnyMediaGroupContent(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
includeMediaGroups: Boolean = true,
|
includeMediaGroups: Boolean = true,
|
||||||
filter: CommonMessageToContentMapper<MediaGroupContent>? = null
|
filter: CommonMessageToContentMapper<MediaGroupContent>? = null
|
||||||
) = waitContent(count, initRequest, includeMediaGroups, errorFactory, filter)
|
) = waitContent(count, initRequest, includeMediaGroups, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitVisualMediaGroup(
|
suspend fun BehaviourContext.waitVisualMediaGroupContent(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
@@ -146,21 +152,21 @@ suspend fun BehaviourContext.waitAudio(
|
|||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
includeMediaGroups: Boolean = true,
|
includeMediaGroups: Boolean = false,
|
||||||
filter: CommonMessageToContentMapper<AudioContent>? = null
|
filter: CommonMessageToContentMapper<AudioContent>? = null
|
||||||
) = waitContent(count, initRequest, includeMediaGroups, errorFactory, filter)
|
) = waitContent(count, initRequest, includeMediaGroups, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitDocument(
|
suspend fun BehaviourContext.waitDocument(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
includeMediaGroups: Boolean = true,
|
includeMediaGroups: Boolean = false,
|
||||||
filter: CommonMessageToContentMapper<DocumentContent>? = null
|
filter: CommonMessageToContentMapper<DocumentContent>? = null
|
||||||
) = waitContent(count, initRequest, includeMediaGroups, errorFactory, filter)
|
) = waitContent(count, initRequest, includeMediaGroups, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitPhoto(
|
suspend fun BehaviourContext.waitPhoto(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
includeMediaGroups: Boolean = true,
|
includeMediaGroups: Boolean = false,
|
||||||
filter: CommonMessageToContentMapper<PhotoContent>? = null
|
filter: CommonMessageToContentMapper<PhotoContent>? = null
|
||||||
) = waitContent(count, initRequest, includeMediaGroups, errorFactory, filter)
|
) = waitContent(count, initRequest, includeMediaGroups, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitSticker(
|
suspend fun BehaviourContext.waitSticker(
|
||||||
@@ -173,7 +179,7 @@ suspend fun BehaviourContext.waitVideo(
|
|||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
includeMediaGroups: Boolean = true,
|
includeMediaGroups: Boolean = false,
|
||||||
filter: CommonMessageToContentMapper<VideoContent>? = null
|
filter: CommonMessageToContentMapper<VideoContent>? = null
|
||||||
) = waitContent(count, initRequest, includeMediaGroups, errorFactory, filter)
|
) = waitContent(count, initRequest, includeMediaGroups, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitVideoNote(
|
suspend fun BehaviourContext.waitVideoNote(
|
||||||
|
|||||||
@@ -0,0 +1,68 @@
|
|||||||
|
package dev.inmo.tgbotapi.extensions.behaviour_builder.expectations
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.extensions.behaviour_builder.BehaviourContext
|
||||||
|
import dev.inmo.tgbotapi.extensions.utils.asCallbackQueryUpdate
|
||||||
|
import dev.inmo.tgbotapi.extensions.utils.asInlineQueryUpdate
|
||||||
|
import dev.inmo.tgbotapi.requests.abstracts.Request
|
||||||
|
import dev.inmo.tgbotapi.types.CallbackQuery.*
|
||||||
|
import dev.inmo.tgbotapi.types.InlineQueries.abstracts.InlineQuery
|
||||||
|
import dev.inmo.tgbotapi.types.InlineQueries.query.BaseInlineQuery
|
||||||
|
import dev.inmo.tgbotapi.types.InlineQueries.query.LocationInlineQuery
|
||||||
|
import kotlinx.coroutines.flow.toList
|
||||||
|
|
||||||
|
typealias InlineQueryMapper<T> = T.() -> T?
|
||||||
|
|
||||||
|
private suspend fun <O> BehaviourContext.waitInlineQueries(
|
||||||
|
count: Int = 1,
|
||||||
|
initRequest: Request<*>? = null,
|
||||||
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
|
mapper: suspend InlineQuery.() -> O?
|
||||||
|
): List<O> = expectFlow(
|
||||||
|
initRequest,
|
||||||
|
count,
|
||||||
|
errorFactory
|
||||||
|
) {
|
||||||
|
it.asInlineQueryUpdate() ?.data ?.mapper().let(::listOfNotNull)
|
||||||
|
}.toList().toList()
|
||||||
|
|
||||||
|
|
||||||
|
private suspend inline fun <reified T : InlineQuery> BehaviourContext.waitInlines(
|
||||||
|
count: Int = 1,
|
||||||
|
initRequest: Request<*>? = null,
|
||||||
|
noinline errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
|
noinline filter: InlineQueryMapper<T>? = null
|
||||||
|
) : List<T> = waitInlineQueries<T>(
|
||||||
|
count,
|
||||||
|
initRequest,
|
||||||
|
errorFactory
|
||||||
|
) {
|
||||||
|
if (this is T) {
|
||||||
|
if (filter == null) {
|
||||||
|
this
|
||||||
|
} else {
|
||||||
|
filter(this)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
suspend fun BehaviourContext.waitAnyInlineQuery(
|
||||||
|
initRequest: Request<*>? = null,
|
||||||
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
|
count: Int = 1,
|
||||||
|
filter: InlineQueryMapper<InlineQuery>? = null
|
||||||
|
) = waitInlines(count, initRequest, errorFactory, filter)
|
||||||
|
|
||||||
|
suspend fun BehaviourContext.waitBaseInlineQuery(
|
||||||
|
initRequest: Request<*>? = null,
|
||||||
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
|
count: Int = 1,
|
||||||
|
filter: InlineQueryMapper<BaseInlineQuery>? = null
|
||||||
|
) = waitInlines(count, initRequest, errorFactory, filter)
|
||||||
|
suspend fun BehaviourContext.waitLocationInlineQuery(
|
||||||
|
initRequest: Request<*>? = null,
|
||||||
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
|
count: Int = 1,
|
||||||
|
filter: InlineQueryMapper<LocationInlineQuery>? = null
|
||||||
|
) = waitInlines(count, initRequest, errorFactory, filter)
|
||||||
@@ -12,14 +12,14 @@ import kotlinx.coroutines.flow.take
|
|||||||
import kotlinx.coroutines.flow.toList
|
import kotlinx.coroutines.flow.toList
|
||||||
|
|
||||||
@PreviewFeature
|
@PreviewFeature
|
||||||
internal suspend inline fun <reified T : MediaGroupContent> BehaviourContext.onMediaGroup(
|
internal suspend inline fun <reified T : MediaGroupContent> BehaviourContext.buildMediaGroupWaiter(
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
noinline errorFactory: NullableRequestBuilder<*> = { null },
|
noinline errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
noinline filter: (suspend (List<MediaGroupMessage>) -> Boolean)? = null
|
noinline filter: (suspend (List<MediaGroupMessage<T>>) -> Boolean)? = null
|
||||||
) = flowsUpdatesFilter.expectFlow(bot, initRequest, count, errorFactory) { update ->
|
) = flowsUpdatesFilter.expectFlow(bot, initRequest, count, errorFactory) { update ->
|
||||||
update.asSentMediaGroupUpdate() ?.data ?.let { mediaGroup ->
|
update.asSentMediaGroupUpdate() ?.data ?.let { mediaGroup ->
|
||||||
if (mediaGroup.all { message -> message.content is T } && (filter == null || filter(mediaGroup))) {
|
if (mediaGroup.all { message -> message.content is T } && (filter == null || filter(mediaGroup as List<MediaGroupMessage<T>>))) {
|
||||||
listOf(
|
listOf(
|
||||||
mediaGroup.map { it.content as T }
|
mediaGroup.map { it.content as T }
|
||||||
)
|
)
|
||||||
@@ -29,33 +29,39 @@ internal suspend inline fun <reified T : MediaGroupContent> BehaviourContext.onM
|
|||||||
} ?: emptyList()
|
} ?: emptyList()
|
||||||
}.take(count).toList()
|
}.take(count).toList()
|
||||||
|
|
||||||
|
suspend fun BehaviourContext.waitMediaGroup(
|
||||||
|
initRequest: Request<*>? = null,
|
||||||
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
|
count: Int = 1,
|
||||||
|
filter: (suspend (List<MediaGroupMessage<MediaGroupContent>>) -> Boolean)? = null
|
||||||
|
) = buildMediaGroupWaiter(count, initRequest, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitPlaylist(
|
suspend fun BehaviourContext.waitPlaylist(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
filter: (suspend (List<MediaGroupMessage>) -> Boolean)? = null
|
filter: (suspend (List<MediaGroupMessage<AudioMediaGroupContent>>) -> Boolean)? = null
|
||||||
) = onMediaGroup<AudioMediaGroupContent>(count, initRequest, errorFactory, filter)
|
) = buildMediaGroupWaiter(count, initRequest, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitDocumentsGroup(
|
suspend fun BehaviourContext.waitDocumentsGroup(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
filter: (suspend (List<MediaGroupMessage>) -> Boolean)? = null
|
filter: (suspend (List<MediaGroupMessage<DocumentMediaGroupContent>>) -> Boolean)? = null
|
||||||
) = onMediaGroup<DocumentMediaGroupContent>(count, initRequest, errorFactory, filter)
|
) = buildMediaGroupWaiter(count, initRequest, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitVisualGallery(
|
suspend fun BehaviourContext.waitVisualGallery(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
filter: (suspend (List<MediaGroupMessage>) -> Boolean)? = null
|
filter: (suspend (List<MediaGroupMessage<VisualMediaGroupContent>>) -> Boolean)? = null
|
||||||
) = onMediaGroup<VisualMediaGroupContent>(count, initRequest, errorFactory, filter)
|
) = buildMediaGroupWaiter(count, initRequest, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitPhotoGallery(
|
suspend fun BehaviourContext.waitPhotoGallery(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
filter: (suspend (List<MediaGroupMessage>) -> Boolean)? = null
|
filter: (suspend (List<MediaGroupMessage<PhotoContent>>) -> Boolean)? = null
|
||||||
) = onMediaGroup<PhotoContent>(count, initRequest, errorFactory, filter)
|
) = buildMediaGroupWaiter(count, initRequest, errorFactory, filter)
|
||||||
suspend fun BehaviourContext.waitVideoGallery(
|
suspend fun BehaviourContext.waitVideoGallery(
|
||||||
initRequest: Request<*>? = null,
|
initRequest: Request<*>? = null,
|
||||||
errorFactory: NullableRequestBuilder<*> = { null },
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
count: Int = 1,
|
count: Int = 1,
|
||||||
filter: (suspend (List<MediaGroupMessage>) -> Boolean)? = null
|
filter: (suspend (List<MediaGroupMessage<VideoContent>>) -> Boolean)? = null
|
||||||
) = onMediaGroup<VideoContent>(count, initRequest, errorFactory, filter)
|
) = buildMediaGroupWaiter(count, initRequest, errorFactory, filter)
|
||||||
|
|||||||
@@ -0,0 +1,54 @@
|
|||||||
|
package dev.inmo.tgbotapi.extensions.behaviour_builder.expectations
|
||||||
|
|
||||||
|
import dev.inmo.tgbotapi.extensions.behaviour_builder.BehaviourContext
|
||||||
|
import dev.inmo.tgbotapi.extensions.utils.*
|
||||||
|
import dev.inmo.tgbotapi.requests.abstracts.Request
|
||||||
|
import dev.inmo.tgbotapi.types.message.PassportMessage
|
||||||
|
import dev.inmo.tgbotapi.types.passport.PassportData
|
||||||
|
import dev.inmo.tgbotapi.types.passport.encrypted.abstracts.EncryptedPassportElement
|
||||||
|
import dev.inmo.tgbotapi.utils.RiskFeature
|
||||||
|
import kotlinx.coroutines.flow.toList
|
||||||
|
|
||||||
|
typealias PassportMessageMapper = suspend PassportMessage.() -> PassportData
|
||||||
|
|
||||||
|
@RiskFeature("Do not use this message directly, use waitPassportMessagesWith or waitAnyPassportMessages instead")
|
||||||
|
suspend fun <O> BehaviourContext.waitPassportMessages(
|
||||||
|
initRequest: Request<*>? = null,
|
||||||
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
|
count: Int = 1,
|
||||||
|
mapper: suspend PassportMessage.() -> O?
|
||||||
|
): List<O> = expectFlow(
|
||||||
|
initRequest,
|
||||||
|
count,
|
||||||
|
errorFactory
|
||||||
|
) {
|
||||||
|
it.asMessageUpdate() ?.data ?.asPassportMessage() ?.mapper().let(::listOfNotNull)
|
||||||
|
}.toList().toList()
|
||||||
|
|
||||||
|
suspend inline fun <reified T : EncryptedPassportElement> BehaviourContext.waitPassportMessagesWith(
|
||||||
|
count: Int = 1,
|
||||||
|
initRequest: Request<*>? = null,
|
||||||
|
noinline errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
|
noinline filter: PassportMessageMapper? = null
|
||||||
|
) : List<PassportData> = waitPassportMessages(
|
||||||
|
initRequest,
|
||||||
|
errorFactory,
|
||||||
|
count
|
||||||
|
) {
|
||||||
|
if (passportData.data.any { it is T }) {
|
||||||
|
if (filter == null) {
|
||||||
|
passportData
|
||||||
|
} else {
|
||||||
|
filter(this)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
suspend fun BehaviourContext.waitAnyPassportMessages(
|
||||||
|
initRequest: Request<*>? = null,
|
||||||
|
errorFactory: NullableRequestBuilder<*> = { null },
|
||||||
|
count: Int = 1,
|
||||||
|
filter: PassportMessageMapper? = null
|
||||||
|
) = waitPassportMessagesWith<EncryptedPassportElement>(count, initRequest, errorFactory, filter)
|
||||||
@@ -1,18 +1,12 @@
|
|||||||
package dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling
|
package dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling
|
||||||
|
|
||||||
|
|
||||||
import dev.inmo.micro_utils.coroutines.safelyWithoutExceptions
|
|
||||||
import dev.inmo.micro_utils.coroutines.subscribeSafelyWithoutExceptions
|
import dev.inmo.micro_utils.coroutines.subscribeSafelyWithoutExceptions
|
||||||
import dev.inmo.tgbotapi.extensions.behaviour_builder.BehaviourContext
|
import dev.inmo.tgbotapi.extensions.behaviour_builder.*
|
||||||
import dev.inmo.tgbotapi.extensions.behaviour_builder.BehaviourContextAndTypeReceiver
|
|
||||||
import dev.inmo.tgbotapi.extensions.behaviour_builder.expectations.expectFlow
|
import dev.inmo.tgbotapi.extensions.behaviour_builder.expectations.expectFlow
|
||||||
import dev.inmo.tgbotapi.extensions.utils.*
|
import dev.inmo.tgbotapi.extensions.utils.*
|
||||||
import dev.inmo.tgbotapi.extensions.utils.extensions.sourceChat
|
import dev.inmo.tgbotapi.extensions.utils.extensions.sourceChat
|
||||||
import dev.inmo.tgbotapi.types.CallbackQuery.*
|
import dev.inmo.tgbotapi.types.CallbackQuery.*
|
||||||
import dev.inmo.tgbotapi.types.message.ChatEvents.*
|
|
||||||
import dev.inmo.tgbotapi.types.message.ChatEvents.abstracts.*
|
|
||||||
import dev.inmo.tgbotapi.updateshandlers.FlowsUpdatesFilter
|
|
||||||
import kotlinx.coroutines.flow.filter
|
|
||||||
|
|
||||||
internal suspend inline fun <reified T : CallbackQuery> BehaviourContext.onCallbackQuery(
|
internal suspend inline fun <reified T : CallbackQuery> BehaviourContext.onCallbackQuery(
|
||||||
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
@@ -27,19 +21,15 @@ internal suspend inline fun <reified T : CallbackQuery> BehaviourContext.onCallb
|
|||||||
}
|
}
|
||||||
}.let(::listOfNotNull)
|
}.let(::listOfNotNull)
|
||||||
}.subscribeSafelyWithoutExceptions(scope) { triggerQuery ->
|
}.subscribeSafelyWithoutExceptions(scope) { triggerQuery ->
|
||||||
val (jobToCancel, scenario) = if (includeFilterByChatInBehaviourSubContext) {
|
doInSubContextWithUpdatesFilter(
|
||||||
val subFilter = FlowsUpdatesFilter()
|
updatesFilter = if (includeFilterByChatInBehaviourSubContext) {
|
||||||
val subBehaviourContext = copy(flowsUpdatesFilter = subFilter)
|
{ it.sourceChat() ?.id ?.chatId == triggerQuery.user.id.chatId }
|
||||||
|
} else {
|
||||||
flowsUpdatesFilter.allUpdatesFlow.filter {
|
null
|
||||||
val chat = it.sourceChat() ?: return@filter false
|
}
|
||||||
chat.id.chatId == triggerQuery.user.id.chatId
|
) {
|
||||||
}.subscribeSafelyWithoutExceptions(scope, subFilter.asUpdateReceiver) to subBehaviourContext
|
scenarioReceiver(triggerQuery)
|
||||||
} else {
|
|
||||||
null to this
|
|
||||||
}
|
}
|
||||||
safelyWithoutExceptions { scenario.scenarioReceiver(triggerQuery) }
|
|
||||||
jobToCancel ?.cancel()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -26,6 +26,12 @@ suspend fun BehaviourContext.command(
|
|||||||
},
|
},
|
||||||
scenarioReceiver
|
scenarioReceiver
|
||||||
)
|
)
|
||||||
|
suspend fun BehaviourContext.command(
|
||||||
|
command: String,
|
||||||
|
requireOnlyCommandInMessage: Boolean = true,
|
||||||
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
|
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<TextContent>>
|
||||||
|
) = command(command.toRegex(), requireOnlyCommandInMessage, includeFilterByChatInBehaviourSubContext, scenarioReceiver)
|
||||||
|
|
||||||
suspend inline fun BehaviourContext.onCommand(
|
suspend inline fun BehaviourContext.onCommand(
|
||||||
commandRegex: Regex,
|
commandRegex: Regex,
|
||||||
@@ -33,3 +39,10 @@ suspend inline fun BehaviourContext.onCommand(
|
|||||||
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
noinline scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<TextContent>>
|
noinline scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<TextContent>>
|
||||||
): Job = command(commandRegex, requireOnlyCommandInMessage, includeFilterByChatInBehaviourSubContext, scenarioReceiver)
|
): Job = command(commandRegex, requireOnlyCommandInMessage, includeFilterByChatInBehaviourSubContext, scenarioReceiver)
|
||||||
|
|
||||||
|
suspend inline fun BehaviourContext.onCommand(
|
||||||
|
command: String,
|
||||||
|
requireOnlyCommandInMessage: Boolean = true,
|
||||||
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
|
noinline scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<TextContent>>
|
||||||
|
): Job = onCommand(command.toRegex(), requireOnlyCommandInMessage, includeFilterByChatInBehaviourSubContext, scenarioReceiver)
|
||||||
|
|||||||
@@ -2,11 +2,8 @@
|
|||||||
|
|
||||||
package dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling
|
package dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling
|
||||||
|
|
||||||
|
|
||||||
import dev.inmo.micro_utils.coroutines.safelyWithoutExceptions
|
|
||||||
import dev.inmo.micro_utils.coroutines.subscribeSafelyWithoutExceptions
|
import dev.inmo.micro_utils.coroutines.subscribeSafelyWithoutExceptions
|
||||||
import dev.inmo.tgbotapi.extensions.behaviour_builder.BehaviourContext
|
import dev.inmo.tgbotapi.extensions.behaviour_builder.*
|
||||||
import dev.inmo.tgbotapi.extensions.behaviour_builder.BehaviourContextAndTypeReceiver
|
|
||||||
import dev.inmo.tgbotapi.extensions.behaviour_builder.expectations.expectFlow
|
import dev.inmo.tgbotapi.extensions.behaviour_builder.expectations.expectFlow
|
||||||
import dev.inmo.tgbotapi.extensions.utils.*
|
import dev.inmo.tgbotapi.extensions.utils.*
|
||||||
import dev.inmo.tgbotapi.extensions.utils.extensions.sourceChat
|
import dev.inmo.tgbotapi.extensions.utils.extensions.sourceChat
|
||||||
@@ -16,9 +13,7 @@ import dev.inmo.tgbotapi.types.message.content.*
|
|||||||
import dev.inmo.tgbotapi.types.message.content.abstracts.*
|
import dev.inmo.tgbotapi.types.message.content.abstracts.*
|
||||||
import dev.inmo.tgbotapi.types.message.content.media.*
|
import dev.inmo.tgbotapi.types.message.content.media.*
|
||||||
import dev.inmo.tgbotapi.types.message.payments.InvoiceContent
|
import dev.inmo.tgbotapi.types.message.payments.InvoiceContent
|
||||||
import dev.inmo.tgbotapi.updateshandlers.FlowsUpdatesFilter
|
|
||||||
import dev.inmo.tgbotapi.utils.PreviewFeature
|
import dev.inmo.tgbotapi.utils.PreviewFeature
|
||||||
import kotlinx.coroutines.flow.filter
|
|
||||||
|
|
||||||
typealias CommonMessageFilter<T> = (suspend (CommonMessage<T>) -> Boolean)
|
typealias CommonMessageFilter<T> = (suspend (CommonMessage<T>) -> Boolean)
|
||||||
|
|
||||||
@@ -50,21 +45,22 @@ internal suspend inline fun <reified T : MessageContent> BehaviourContext.onCont
|
|||||||
}
|
}
|
||||||
}.let(::listOfNotNull)
|
}.let(::listOfNotNull)
|
||||||
}.subscribeSafelyWithoutExceptions(scope) { triggerMessage ->
|
}.subscribeSafelyWithoutExceptions(scope) { triggerMessage ->
|
||||||
val (jobToCancel, scenario) = if (includeFilterByChatInBehaviourSubContext) {
|
doInSubContextWithUpdatesFilter(
|
||||||
val subFilter = FlowsUpdatesFilter()
|
updatesFilter = if (includeFilterByChatInBehaviourSubContext) {
|
||||||
val subBehaviourContext = copy(flowsUpdatesFilter = subFilter)
|
{ it.sourceChat() ?.id ?.chatId == triggerMessage.chat.id.chatId }
|
||||||
|
} else {
|
||||||
flowsUpdatesFilter.allUpdatesFlow.filter {
|
null
|
||||||
val chat = it.sourceChat() ?: return@filter false
|
}
|
||||||
chat.id.chatId == triggerMessage.chat.id.chatId
|
) {
|
||||||
}.subscribeSafelyWithoutExceptions(scope, subFilter.asUpdateReceiver) to subBehaviourContext
|
scenarioReceiver(triggerMessage)
|
||||||
} else {
|
|
||||||
null to this
|
|
||||||
}
|
}
|
||||||
safelyWithoutExceptions { scenario.scenarioReceiver(triggerMessage) }
|
|
||||||
jobToCancel ?.cancel()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
suspend fun BehaviourContext.onContentMessage(
|
||||||
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
|
additionalFilter: CommonMessageFilter<MessageContent>? = null,
|
||||||
|
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<MessageContent>>
|
||||||
|
) = onContent(includeFilterByChatInBehaviourSubContext, false, additionalFilter, scenarioReceiver)
|
||||||
suspend fun BehaviourContext.onContact(
|
suspend fun BehaviourContext.onContact(
|
||||||
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
additionalFilter: CommonMessageFilter<ContactContent>? = null,
|
additionalFilter: CommonMessageFilter<ContactContent>? = null,
|
||||||
@@ -105,7 +101,7 @@ suspend fun BehaviourContext.onAudioMediaGroup(
|
|||||||
additionalFilter: CommonMessageFilter<AudioMediaGroupContent>? = null,
|
additionalFilter: CommonMessageFilter<AudioMediaGroupContent>? = null,
|
||||||
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<AudioMediaGroupContent>>
|
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<AudioMediaGroupContent>>
|
||||||
) = onContent(includeFilterByChatInBehaviourSubContext, true, additionalFilter, scenarioReceiver)
|
) = onContent(includeFilterByChatInBehaviourSubContext, true, additionalFilter, scenarioReceiver)
|
||||||
suspend fun BehaviourContext.onDocumentMediaGroup(
|
suspend fun BehaviourContext.onDocumentMediaGroupContent(
|
||||||
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
includeMediaGroups: Boolean = true,
|
includeMediaGroups: Boolean = true,
|
||||||
additionalFilter: CommonMessageFilter<DocumentMediaGroupContent>? = null,
|
additionalFilter: CommonMessageFilter<DocumentMediaGroupContent>? = null,
|
||||||
@@ -113,7 +109,7 @@ suspend fun BehaviourContext.onDocumentMediaGroup(
|
|||||||
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
||||||
suspend fun BehaviourContext.onMediaCollection(
|
suspend fun BehaviourContext.onMediaCollection(
|
||||||
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
includeMediaGroups: Boolean = true,
|
includeMediaGroups: Boolean = false,
|
||||||
additionalFilter: (suspend (CommonMessage<MediaCollectionContent<TelegramMediaFile>>) -> Boolean)? = null,
|
additionalFilter: (suspend (CommonMessage<MediaCollectionContent<TelegramMediaFile>>) -> Boolean)? = null,
|
||||||
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<MediaCollectionContent<TelegramMediaFile>>>
|
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<MediaCollectionContent<TelegramMediaFile>>>
|
||||||
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
||||||
@@ -123,18 +119,6 @@ suspend fun BehaviourContext.onMedia(
|
|||||||
additionalFilter: CommonMessageFilter<MediaContent>? = null,
|
additionalFilter: CommonMessageFilter<MediaContent>? = null,
|
||||||
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<MediaContent>>
|
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<MediaContent>>
|
||||||
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
||||||
suspend fun BehaviourContext.onMediaGroup(
|
|
||||||
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
|
||||||
includeMediaGroups: Boolean = true,
|
|
||||||
additionalFilter: CommonMessageFilter<MediaGroupContent>? = null,
|
|
||||||
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<MediaGroupContent>>
|
|
||||||
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
|
||||||
suspend fun BehaviourContext.onVisualMediaGroup(
|
|
||||||
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
|
||||||
includeMediaGroups: Boolean = true,
|
|
||||||
additionalFilter: CommonMessageFilter<VisualMediaGroupContent>? = null,
|
|
||||||
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<VisualMediaGroupContent>>
|
|
||||||
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
|
||||||
suspend fun BehaviourContext.onAnimation(
|
suspend fun BehaviourContext.onAnimation(
|
||||||
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
additionalFilter: CommonMessageFilter<AnimationContent>? = null,
|
additionalFilter: CommonMessageFilter<AnimationContent>? = null,
|
||||||
@@ -142,19 +126,19 @@ suspend fun BehaviourContext.onAnimation(
|
|||||||
) = onContent(includeFilterByChatInBehaviourSubContext, false, additionalFilter, scenarioReceiver)
|
) = onContent(includeFilterByChatInBehaviourSubContext, false, additionalFilter, scenarioReceiver)
|
||||||
suspend fun BehaviourContext.onAudio(
|
suspend fun BehaviourContext.onAudio(
|
||||||
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
includeMediaGroups: Boolean = true,
|
includeMediaGroups: Boolean = false,
|
||||||
additionalFilter: CommonMessageFilter<AudioContent>? = null,
|
additionalFilter: CommonMessageFilter<AudioContent>? = null,
|
||||||
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<AudioContent>>
|
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<AudioContent>>
|
||||||
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
||||||
suspend fun BehaviourContext.onDocument(
|
suspend fun BehaviourContext.onDocument(
|
||||||
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
includeMediaGroups: Boolean = true,
|
includeMediaGroups: Boolean = false,
|
||||||
additionalFilter: CommonMessageFilter<DocumentContent>? = null,
|
additionalFilter: CommonMessageFilter<DocumentContent>? = null,
|
||||||
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<DocumentContent>>
|
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<DocumentContent>>
|
||||||
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
||||||
suspend fun BehaviourContext.onPhoto(
|
suspend fun BehaviourContext.onPhoto(
|
||||||
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
includeMediaGroups: Boolean = true,
|
includeMediaGroups: Boolean = false,
|
||||||
additionalFilter: CommonMessageFilter<PhotoContent>? = null,
|
additionalFilter: CommonMessageFilter<PhotoContent>? = null,
|
||||||
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<PhotoContent>>
|
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<PhotoContent>>
|
||||||
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
||||||
@@ -165,7 +149,7 @@ suspend fun BehaviourContext.onSticker(
|
|||||||
) = onContent(includeFilterByChatInBehaviourSubContext, false, additionalFilter, scenarioReceiver)
|
) = onContent(includeFilterByChatInBehaviourSubContext, false, additionalFilter, scenarioReceiver)
|
||||||
suspend fun BehaviourContext.onVideo(
|
suspend fun BehaviourContext.onVideo(
|
||||||
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
includeMediaGroups: Boolean = true,
|
includeMediaGroups: Boolean = false,
|
||||||
additionalFilter: CommonMessageFilter<VideoContent>? = null,
|
additionalFilter: CommonMessageFilter<VideoContent>? = null,
|
||||||
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<VideoContent>>
|
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, CommonMessage<VideoContent>>
|
||||||
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
) = onContent(includeFilterByChatInBehaviourSubContext, includeMediaGroups, additionalFilter, scenarioReceiver)
|
||||||
|
|||||||
@@ -1,21 +1,14 @@
|
|||||||
package dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling
|
package dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling
|
||||||
|
|
||||||
|
|
||||||
import dev.inmo.micro_utils.coroutines.safelyWithoutExceptions
|
|
||||||
import dev.inmo.micro_utils.coroutines.subscribeSafelyWithoutExceptions
|
import dev.inmo.micro_utils.coroutines.subscribeSafelyWithoutExceptions
|
||||||
import dev.inmo.tgbotapi.extensions.behaviour_builder.BehaviourContext
|
import dev.inmo.tgbotapi.extensions.behaviour_builder.*
|
||||||
import dev.inmo.tgbotapi.extensions.behaviour_builder.BehaviourContextAndTypeReceiver
|
|
||||||
import dev.inmo.tgbotapi.extensions.behaviour_builder.expectations.expectFlow
|
import dev.inmo.tgbotapi.extensions.behaviour_builder.expectations.expectFlow
|
||||||
import dev.inmo.tgbotapi.extensions.utils.*
|
import dev.inmo.tgbotapi.extensions.utils.*
|
||||||
import dev.inmo.tgbotapi.extensions.utils.extensions.sourceChat
|
import dev.inmo.tgbotapi.extensions.utils.extensions.sourceChat
|
||||||
import dev.inmo.tgbotapi.types.message.ChatEvents.*
|
import dev.inmo.tgbotapi.types.message.ChatEvents.*
|
||||||
import dev.inmo.tgbotapi.types.message.ChatEvents.abstracts.*
|
import dev.inmo.tgbotapi.types.message.ChatEvents.abstracts.*
|
||||||
import dev.inmo.tgbotapi.types.message.abstracts.ChatEventMessage
|
import dev.inmo.tgbotapi.types.message.abstracts.ChatEventMessage
|
||||||
import dev.inmo.tgbotapi.types.message.content.*
|
|
||||||
import dev.inmo.tgbotapi.types.message.content.abstracts.*
|
|
||||||
import dev.inmo.tgbotapi.types.message.content.media.*
|
|
||||||
import dev.inmo.tgbotapi.updateshandlers.FlowsUpdatesFilter
|
|
||||||
import kotlinx.coroutines.flow.filter
|
|
||||||
|
|
||||||
internal suspend inline fun <reified T : ChatEvent> BehaviourContext.onEvent(
|
internal suspend inline fun <reified T : ChatEvent> BehaviourContext.onEvent(
|
||||||
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
@@ -31,19 +24,13 @@ internal suspend inline fun <reified T : ChatEvent> BehaviourContext.onEvent(
|
|||||||
}
|
}
|
||||||
}.let(::listOfNotNull)
|
}.let(::listOfNotNull)
|
||||||
}.subscribeSafelyWithoutExceptions(scope) { triggerMessage ->
|
}.subscribeSafelyWithoutExceptions(scope) { triggerMessage ->
|
||||||
val (jobToCancel, scenario) = if (includeFilterByChatInBehaviourSubContext) {
|
doInSubContextWithUpdatesFilter(
|
||||||
val subFilter = FlowsUpdatesFilter()
|
updatesFilter = if (includeFilterByChatInBehaviourSubContext) {
|
||||||
val subBehaviourContext = copy(flowsUpdatesFilter = subFilter)
|
{ it.sourceChat() ?.id ?.chatId == triggerMessage.chat.id.chatId }
|
||||||
|
} else null
|
||||||
flowsUpdatesFilter.allUpdatesFlow.filter {
|
) {
|
||||||
val chat = it.sourceChat() ?: return@filter false
|
scenarioReceiver(triggerMessage)
|
||||||
chat.id.chatId == triggerMessage.chat.id.chatId
|
|
||||||
}.subscribeSafelyWithoutExceptions(scope, subFilter.asUpdateReceiver) to subBehaviourContext
|
|
||||||
} else {
|
|
||||||
null to this
|
|
||||||
}
|
}
|
||||||
safelyWithoutExceptions { scenario.scenarioReceiver(triggerMessage) }
|
|
||||||
jobToCancel ?.cancel()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
suspend fun BehaviourContext.onChannelEvent(
|
suspend fun BehaviourContext.onChannelEvent(
|
||||||
|
|||||||
@@ -0,0 +1,57 @@
|
|||||||
|
package dev.inmo.tgbotapi.extensions.behaviour_builder.triggers_handling
|
||||||
|
|
||||||
|
import dev.inmo.micro_utils.coroutines.subscribeSafelyWithoutExceptions
|
||||||
|
import dev.inmo.tgbotapi.extensions.behaviour_builder.*
|
||||||
|
import dev.inmo.tgbotapi.extensions.behaviour_builder.expectations.expectFlow
|
||||||
|
import dev.inmo.tgbotapi.extensions.utils.asInlineQueryUpdate
|
||||||
|
import dev.inmo.tgbotapi.extensions.utils.extensions.sourceChat
|
||||||
|
import dev.inmo.tgbotapi.types.CallbackQuery.DataCallbackQuery
|
||||||
|
import dev.inmo.tgbotapi.types.CallbackQuery.GameShortNameCallbackQuery
|
||||||
|
import dev.inmo.tgbotapi.types.InlineQueries.abstracts.InlineQuery
|
||||||
|
import dev.inmo.tgbotapi.types.InlineQueries.query.BaseInlineQuery
|
||||||
|
import dev.inmo.tgbotapi.types.InlineQueries.query.LocationInlineQuery
|
||||||
|
|
||||||
|
internal suspend inline fun <reified T : InlineQuery> BehaviourContext.onInlineQuery(
|
||||||
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
|
noinline additionalFilter: (suspend (T) -> Boolean)? = null,
|
||||||
|
noinline scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, T>
|
||||||
|
) = flowsUpdatesFilter.expectFlow(bot) {
|
||||||
|
it.asInlineQueryUpdate() ?.data ?.let { query ->
|
||||||
|
if (query is T) {
|
||||||
|
if (additionalFilter == null || additionalFilter(query)) query else null
|
||||||
|
} else {
|
||||||
|
null
|
||||||
|
}
|
||||||
|
}.let(::listOfNotNull)
|
||||||
|
}.subscribeSafelyWithoutExceptions(scope) { triggerQuery ->
|
||||||
|
doInSubContextWithUpdatesFilter(
|
||||||
|
updatesFilter = if (includeFilterByChatInBehaviourSubContext) {
|
||||||
|
{ it.sourceChat() ?.id ?.chatId == triggerQuery.from.id.chatId }
|
||||||
|
} else {
|
||||||
|
null
|
||||||
|
}
|
||||||
|
) {
|
||||||
|
scenarioReceiver(triggerQuery)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
suspend fun BehaviourContext.onAnyInlineQuery(
|
||||||
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
|
additionalFilter: (suspend (InlineQuery) -> Boolean)? = null,
|
||||||
|
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, InlineQuery>
|
||||||
|
) = onInlineQuery(includeFilterByChatInBehaviourSubContext, additionalFilter, scenarioReceiver)
|
||||||
|
|
||||||
|
|
||||||
|
suspend fun BehaviourContext.onBaseInlineQuery(
|
||||||
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
|
additionalFilter: (suspend (BaseInlineQuery) -> Boolean)? = null,
|
||||||
|
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, BaseInlineQuery>
|
||||||
|
) = onInlineQuery(includeFilterByChatInBehaviourSubContext, additionalFilter, scenarioReceiver)
|
||||||
|
|
||||||
|
|
||||||
|
suspend fun BehaviourContext.onLocationInlineQuery(
|
||||||
|
includeFilterByChatInBehaviourSubContext: Boolean = true,
|
||||||
|
additionalFilter: (suspend (LocationInlineQuery) -> Boolean)? = null,
|
||||||
|
scenarioReceiver: BehaviourContextAndTypeReceiver<Unit, LocationInlineQuery>
|
||||||
|
) = onInlineQuery(includeFilterByChatInBehaviourSubContext, additionalFilter, scenarioReceiver)
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user