mirror of
https://github.com/mamoe/mirai.git
synced 2025-01-25 07:30:14 +08:00
147 lines
5.2 KiB
Kotlin
147 lines
5.2 KiB
Kotlin
/*
|
|
* Copyright 2019-2021 Mamoe Technologies and contributors.
|
|
*
|
|
* 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证.
|
|
* Use of this source code is governed by the GNU AGPLv3 license that can be found through the following link.
|
|
*
|
|
* https://github.com/mamoe/mirai/blob/master/LICENSE
|
|
*/
|
|
|
|
@file:JvmMultifileClass
|
|
@file:JvmName("MiraiUtils")
|
|
|
|
@file:Suppress("NOTHING_TO_INLINE")
|
|
|
|
package net.mamoe.mirai.utils
|
|
|
|
import io.ktor.utils.io.charsets.*
|
|
import kotlinx.io.core.*
|
|
import java.io.File
|
|
import kotlin.text.Charsets
|
|
|
|
public val EMPTY_BYTE_ARRAY: ByteArray = ByteArray(0)
|
|
|
|
public val DECRYPTER_16_ZERO: ByteArray = ByteArray(16)
|
|
public val KEY_16_ZEROS: ByteArray = ByteArray(16)
|
|
|
|
@Suppress("INVISIBLE_MEMBER", "INVISIBLE_REFERENCE")
|
|
public inline fun <R> ByteReadPacket.useBytes(
|
|
n: Int = remaining.toInt(),//not that safe but adequate
|
|
block: (data: ByteArray, length: Int) -> R
|
|
): R = ByteArrayPool.useInstance(n) {
|
|
this.readFully(it, 0, n)
|
|
block(it, n)
|
|
}
|
|
|
|
public inline fun ByteReadPacket.readPacketExact(
|
|
n: Int = remaining.toInt()//not that safe but adequate
|
|
): ByteReadPacket = this.readBytes(n).toReadPacket()
|
|
|
|
|
|
public typealias TlvMap = MutableMap<Int, ByteArray>
|
|
|
|
public inline fun TlvMap.getOrFail(tag: Int): ByteArray {
|
|
return this[tag] ?: error("cannot find tlv 0x${tag.toUHexString("")}($tag)")
|
|
}
|
|
|
|
public inline fun TlvMap.getOrFail(tag: Int, lazyMessage: (tag: Int) -> String): ByteArray {
|
|
return this[tag] ?: error(lazyMessage(tag))
|
|
}
|
|
|
|
@Suppress("FunctionName")
|
|
public inline fun Input._readTLVMap(tagSize: Int = 2, suppressDuplication: Boolean = true): TlvMap =
|
|
_readTLVMap(true, tagSize, suppressDuplication)
|
|
|
|
@Suppress("DuplicatedCode", "FunctionName")
|
|
public fun Input._readTLVMap(
|
|
expectingEOF: Boolean = true,
|
|
tagSize: Int,
|
|
suppressDuplication: Boolean = true
|
|
): TlvMap {
|
|
val map = mutableMapOf<Int, ByteArray>()
|
|
var key = 0
|
|
|
|
while (kotlin.run {
|
|
try {
|
|
key = when (tagSize) {
|
|
1 -> readUByte().toInt()
|
|
2 -> readUShort().toInt()
|
|
4 -> readUInt().toInt()
|
|
else -> error("Unsupported tag size: $tagSize")
|
|
}
|
|
} catch (e: Exception) { // java.nio.BufferUnderflowException is not a EOFException...
|
|
if (expectingEOF) {
|
|
return map
|
|
}
|
|
throw e
|
|
}
|
|
key
|
|
}.toUByte() != UByte.MAX_VALUE) {
|
|
|
|
if (map.containsKey(key)) {
|
|
@Suppress("ControlFlowWithEmptyBody")
|
|
if (!suppressDuplication) {
|
|
/*
|
|
@Suppress("DEPRECATION")
|
|
MiraiLogger.error(
|
|
@Suppress("IMPLICIT_CAST_TO_ANY")
|
|
"""
|
|
Error readTLVMap:
|
|
duplicated key ${when (tagSize) {
|
|
1 -> key.toByte()
|
|
2 -> key.toShort()
|
|
4 -> key
|
|
else -> error("unreachable")
|
|
}.contentToString()}
|
|
map=${map.contentToString()}
|
|
duplicating value=${this.readUShortLVByteArray().toUHexString()}
|
|
""".trimIndent()
|
|
)*/
|
|
} else {
|
|
this.discardExact(this.readShort().toInt() and 0xffff)
|
|
}
|
|
} else {
|
|
try {
|
|
map[key] = this.readBytes(readUShort().toInt())
|
|
} catch (e: Exception) { // BufferUnderflowException, java.io.EOFException
|
|
// if (expectingEOF) {
|
|
// return map
|
|
// }
|
|
throw e
|
|
}
|
|
}
|
|
}
|
|
return map
|
|
}
|
|
|
|
public inline fun Input.readString(length: Int, charset: Charset = Charsets.UTF_8): String =
|
|
String(this.readBytes(length), charset = charset)
|
|
|
|
public inline fun Input.readString(length: Long, charset: Charset = Charsets.UTF_8): String =
|
|
String(this.readBytes(length.toInt()), charset = charset)
|
|
|
|
public inline fun Input.readString(length: Short, charset: Charset = Charsets.UTF_8): String =
|
|
String(this.readBytes(length.toInt()), charset = charset)
|
|
|
|
@JvmSynthetic
|
|
public inline fun Input.readString(length: UShort, charset: Charset = Charsets.UTF_8): String =
|
|
String(this.readBytes(length.toInt()), charset = charset)
|
|
|
|
public inline fun Input.readString(length: Byte, charset: Charset = Charsets.UTF_8): String =
|
|
String(this.readBytes(length.toInt()), charset = charset)
|
|
|
|
public fun Input.readUShortLVString(): String = String(this.readUShortLVByteArray())
|
|
public fun Input.readUShortLVByteArray(): ByteArray = this.readBytes(this.readUShort().toInt())
|
|
|
|
public fun File.createFileIfNotExists() {
|
|
if (!this.exists()) {
|
|
this.parentFile.mkdirs()
|
|
this.createNewFile()
|
|
}
|
|
}
|
|
|
|
public fun File.resolveCreateFile(relative: String): File = this.resolve(relative).apply { createFileIfNotExists() }
|
|
public fun File.resolveCreateFile(relative: File): File = this.resolve(relative).apply { createFileIfNotExists() }
|
|
|
|
public fun File.resolveMkdir(relative: String): File = this.resolve(relative).apply { mkdirs() }
|
|
public fun File.resolveMkdir(relative: File): File = this.resolve(relative).apply { mkdirs() } |