Skip to content

Commit 5dbb217

Browse files
committed
Fix js test
1 parent 34189a2 commit 5dbb217

File tree

12 files changed

+50
-149
lines changed

12 files changed

+50
-149
lines changed

simbot-component-qq-guild-internal-ed25519/build.gradle.kts

+1
Original file line numberDiff line numberDiff line change
@@ -68,6 +68,7 @@ kotlin {
6868

6969
commonTest.dependencies {
7070
implementation(kotlin("test"))
71+
implementation(libs.kotlinx.coroutines.test)
7172
}
7273

7374
jvmMain.dependencies {

simbot-component-qq-guild-internal-ed25519/src/commonMain/kotlin/love/forte/simbot/qguild/ed25519/Ed25519s.kt

+4-2
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,9 @@ public interface Ed25519KeyPairGenerator {
3838
/**
3939
* 根据当前平台情况和classpath环境选择一个合适的 [Ed25519KeyPairGenerator] 实例。
4040
*
41-
* 在非 JVM 平台中,会使用 [libsodium bindings](https://github.com/ionspin/kotlin-multiplatform-libsodium),
41+
* 在非 JVM 平台中,会使用 [libsodium bindings](https://github.com/ionspin/kotlin-multiplatform-libsodium)。
42+
* 由于 libsodium bindings 初始化可能需要挂起,因此 [ed25519KeyPairGenerator] 在首次执行时也可能是挂起的。
43+
*
4244
* 在 JVM 中,会优先检测是否存在 [BouncyCastle](https://www.bouncycastle.org/) 依赖中所需使用的类,
4345
* 例如:
4446
*
@@ -55,7 +57,7 @@ public interface Ed25519KeyPairGenerator {
5557
*
5658
*/
5759
@InternalEd25519Api
58-
public expect fun ed25519KeyPairGenerator(): Ed25519KeyPairGenerator
60+
public expect suspend fun ed25519KeyPairGenerator(): Ed25519KeyPairGenerator
5961

6062
@InternalEd25519Api
6163
public interface Ed25519KeyPair {

simbot-component-qq-guild-internal-ed25519/src/commonTest/kotlin/love/forte/simbot/qguild/ed25519/Ed25519CommonTests.kt

+2-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
package love.forte.simbot.qguild.ed25519
22

3+
import kotlinx.coroutines.test.runTest
34
import love.forte.simbot.qguild.ed25519.annotations.InternalEd25519Api
45
import kotlin.test.Test
56
import kotlin.test.assertContentEquals
@@ -33,7 +34,7 @@ class Ed25519CommonTests {
3334
}
3435

3536
@Test
36-
fun testCommonKeyPairGeneration() {
37+
fun testCommonKeyPairGeneration() = runTest {
3738
val (pri, pub) = ed25519KeyPairGenerator().generate(SEED.encodeToByteArray())
3839
val publicKeyBytes = pub.bytes
3940
val privateKeyCombined = pri.bytes
@@ -56,5 +57,3 @@ class Ed25519CommonTests {
5657
)
5758
}
5859
}
59-
60-
internal fun a() {}

simbot-component-qq-guild-internal-ed25519/src/jsMain/kotlin/love/forte/simbot/qguild/ed25519/Ed25519s.js.kt

+4-2
Original file line numberDiff line numberDiff line change
@@ -26,5 +26,7 @@ import love.forte.simbot.qguild.ed25519.annotations.InternalEd25519Api
2626
*
2727
*/
2828
@InternalEd25519Api
29-
public actual fun ed25519KeyPairGenerator(): Ed25519KeyPairGenerator = LibsodiumEd25519KeyPairGenerator
30-
29+
public actual suspend fun ed25519KeyPairGenerator(): Ed25519KeyPairGenerator {
30+
initialLibsodiumIfNecessary()
31+
return LibsodiumEd25519KeyPairGenerator
32+
}

simbot-component-qq-guild-internal-ed25519/src/jsMain/kotlin/love/forte/simbot/qguild/ed25519/LibsodiumEd25519s.js.kt

+2-24
Original file line numberDiff line numberDiff line change
@@ -24,10 +24,6 @@ import love.forte.simbot.qguild.ed25519.annotations.InternalEd25519Api
2424

2525
@InternalEd25519Api
2626
public object LibsodiumEd25519KeyPairGenerator : Ed25519KeyPairGenerator {
27-
init {
28-
initialLibsodium
29-
}
30-
3127
override fun generate(seed: ByteArray): Ed25519KeyPair {
3228
return LibsodiumEd25519KeyPair(seed)
3329
}
@@ -79,28 +75,10 @@ public class LibsodiumEd25519PublicKey(private val key: UByteArray) : Ed25519Pub
7975
}
8076
}
8177

82-
83-
/**
84-
* Initialed libsodium by lazy.
85-
*/
86-
private val initialLibsodium: Unit by lazy(
87-
mode = LazyThreadSafetyMode.SYNCHRONIZED
88-
) {
78+
internal suspend fun initialLibsodiumIfNecessary() {
8979
if (!LibsodiumInitializer.isInitialized()) {
9080
ed25519sLogger.info("LibsodiumInitializer is not initialed yet, initializing...")
91-
var done = false
92-
93-
LibsodiumInitializer.initializeWithCallback {
94-
ed25519sLogger.info("LibsodiumInitializer initialized in callback")
95-
done = true
96-
}
97-
98-
while (!done) {
99-
// loop
100-
}
101-
81+
LibsodiumInitializer.initialize()
10282
ed25519sLogger.info("LibsodiumInitializer initialized")
10383
}
104-
105-
Unit
10684
}

simbot-component-qq-guild-internal-ed25519/src/jvmMain/kotlin/love/forte/simbot/qguild/ed25519/Ed25519s.jvm.kt

+1-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,7 @@ import love.forte.simbot.qguild.ed25519.annotations.InternalEd25519Api
3636
*
3737
*/
3838
@InternalEd25519Api
39-
public actual fun ed25519KeyPairGenerator(): Ed25519KeyPairGenerator {
39+
public actual suspend fun ed25519KeyPairGenerator(): Ed25519KeyPairGenerator {
4040
return if (bouncyCastleAccessible) {
4141
BouncyCastleEd25519KeyPairGenerator
4242
} else {

simbot-component-qq-guild-internal-ed25519/src/nativeMain/kotlin/love/forte/simbot/qguild/ed25519/Ed25519s.native.kt

+4-1
Original file line numberDiff line numberDiff line change
@@ -26,4 +26,7 @@ import love.forte.simbot.qguild.ed25519.annotations.InternalEd25519Api
2626
*
2727
*/
2828
@InternalEd25519Api
29-
public actual fun ed25519KeyPairGenerator(): Ed25519KeyPairGenerator = LibsodiumEd25519KeyPairGenerator
29+
public actual suspend fun ed25519KeyPairGenerator(): Ed25519KeyPairGenerator {
30+
initialLibsodiumIfNecessary()
31+
return LibsodiumEd25519KeyPairGenerator
32+
}

simbot-component-qq-guild-internal-ed25519/src/nativeMain/kotlin/love/forte/simbot/qguild/ed25519/LibsodiumEd25519s.native.kt

+12-9
Original file line numberDiff line numberDiff line change
@@ -19,16 +19,12 @@ package love.forte.simbot.qguild.ed25519
1919

2020
import com.ionspin.kotlin.crypto.LibsodiumInitializer
2121
import com.ionspin.kotlin.crypto.signature.InvalidSignatureException
22-
import kotlinx.coroutines.runBlocking
22+
import kotlinx.coroutines.CompletableDeferred
2323
import love.forte.simbot.qguild.ed25519.annotations.InternalEd25519Api
2424

2525

2626
@InternalEd25519Api
2727
public object LibsodiumEd25519KeyPairGenerator : Ed25519KeyPairGenerator {
28-
init {
29-
initialLibsodium
30-
}
31-
3228
override fun generate(seed: ByteArray): Ed25519KeyPair {
3329
return LibsodiumEd25519KeyPair(seed)
3430
}
@@ -81,6 +77,13 @@ public class LibsodiumEd25519PublicKey(private val key: UByteArray) : Ed25519Pub
8177
}
8278

8379

80+
internal suspend fun initialLibsodiumIfNecessary() {
81+
initialLibsodium // Init by lazy
82+
return initialDeferred.await()
83+
}
84+
85+
private val initialDeferred = CompletableDeferred(Unit)
86+
8487
/**
8588
* Initialed libsodium by lazy.
8689
*/
@@ -90,10 +93,10 @@ private val initialLibsodium: Unit by lazy(
9093
if (!LibsodiumInitializer.isInitialized()) {
9194
ed25519sLogger.info("LibsodiumInitializer is not initialed yet, initializing...")
9295

93-
runBlocking { LibsodiumInitializer.initialize() }
94-
95-
ed25519sLogger.info("LibsodiumInitializer initialized")
96+
LibsodiumInitializer.initializeWithCallback {
97+
initialDeferred.complete(Unit)
98+
ed25519sLogger.info("LibsodiumInitializer initialized")
99+
}
96100
}
97-
98101
Unit
99102
}

simbot-component-qq-guild-internal-ed25519/src/non-jvmMain/kotlin/love/forte/simbot/qguild/ed25519/Ed25519s.non-jvm.kt

-31
This file was deleted.

simbot-component-qq-guild-internal-ed25519/src/non-jvmMain/kotlin/love/forte/simbot/qguild/ed25519/LibsodiumEd25519s.kt

-60
This file was deleted.

simbot-component-qq-guild-stdlib/src/commonMain/kotlin/love/forte/simbot/qguild/stdlib/internal/BotImpl.kt

+19-15
Original file line numberDiff line numberDiff line change
@@ -49,10 +49,8 @@ import love.forte.simbot.qguild.api.app.AppAccessToken
4949
import love.forte.simbot.qguild.api.app.GetAppAccessTokenApi
5050
import love.forte.simbot.qguild.api.requestData
5151
import love.forte.simbot.qguild.api.user.GetBotInfoApi
52-
import love.forte.simbot.qguild.ed25519.Ed25519KeyPair
53-
import love.forte.simbot.qguild.ed25519.Signature
52+
import love.forte.simbot.qguild.ed25519.*
5453
import love.forte.simbot.qguild.ed25519.annotations.InternalEd25519Api
55-
import love.forte.simbot.qguild.ed25519.paddingEd25519Seed
5654
import love.forte.simbot.qguild.event.*
5755
import love.forte.simbot.qguild.model.User
5856
import love.forte.simbot.qguild.stdlib.*
@@ -109,15 +107,23 @@ internal class BotImpl(
109107
}
110108
}
111109

112-
private val ed25519KeyPair: Ed25519KeyPair by lazy {
113-
genEd25519Keypair(ticket.secret.paddingEd25519Seed().toByteArray())
110+
private lateinit var _ed25519PrivateKey: Ed25519PrivateKey
111+
private lateinit var _ed25519PublicKey: Ed25519PublicKey
112+
113+
private val ed25519KeyPair: Deferred<Ed25519KeyPair> = async(start = CoroutineStart.LAZY) {
114+
genEd25519Keypair(ticket.secret.paddingEd25519Seed().toByteArray()).also { (pri, pub) ->
115+
_ed25519PrivateKey = pri
116+
_ed25519PublicKey = pub
117+
}
114118
}
115119

116-
private val ed25519PrivateKey
117-
get() = ed25519KeyPair.privateKey
120+
private suspend fun ed25519PrivateKey() =
121+
if (::_ed25519PrivateKey.isInitialized) _ed25519PrivateKey
122+
else ed25519KeyPair.await().privateKey
118123

119-
private val ed25519PublicKey
120-
get() = ed25519KeyPair.publicKey
124+
private suspend fun ed25519PublicKey() =
125+
if (::_ed25519PublicKey.isInitialized) _ed25519PublicKey
126+
else ed25519KeyPair.await().publicKey
121127

122128
internal val eventDecoder = Signal.Dispatch.dispatchJson {
123129
isLenient = true
@@ -424,7 +430,7 @@ internal class BotImpl(
424430
// 13 回调地址验证 Receive 开放平台对机器人服务端进行验证
425431
logger.debug("Emit raw event with payload: {}", payload)
426432

427-
fun verifyIfNecessary() {
433+
suspend fun verifyIfNecessary() {
428434
val (signature, signatureTimestamp) = options?.ed25519SignatureVerification ?: return
429435
logger.debug("`ed25519SignatureVerification` exists, verity the payload")
430436

@@ -446,18 +452,16 @@ internal class BotImpl(
446452

447453
val msg = "$signatureTimestamp$payload"
448454

449-
check(ed25519PublicKey.verify(signature = Signature(signatureBytes), data = msg.toByteArray())) {
455+
check(ed25519PublicKey().verify(signature = Signature(signatureBytes), data = msg.toByteArray())) {
450456
"Ed25519 verify failed"
451457
}
452458
}
453459

454-
fun signatureIfNecessary(verify: Signal.CallbackVerify): EmitResult {
460+
suspend fun signatureIfNecessary(verify: Signal.CallbackVerify): EmitResult {
455461
val (plainToken, eventTs) = verify.data
456462
val msg = "$eventTs$plainToken"
457463

458-
val signature = ed25519PrivateKey.sign(msg.toByteArray())
459-
460-
signature.data
464+
val signature = ed25519PrivateKey().sign(msg.toByteArray())
461465

462466
val verified = Signal.CallbackVerify.Verified(
463467
plainToken,

simbot-component-qq-guild-stdlib/src/commonMain/kotlin/love/forte/simbot/qguild/stdlib/internal/Ed25519s.kt

+1-1
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,6 @@ import love.forte.simbot.qguild.ed25519.annotations.InternalEd25519Api
2222
import love.forte.simbot.qguild.ed25519.ed25519KeyPairGenerator
2323

2424
@OptIn(InternalEd25519Api::class)
25-
internal fun genEd25519Keypair(seed: ByteArray): Ed25519KeyPair {
25+
internal suspend fun genEd25519Keypair(seed: ByteArray): Ed25519KeyPair {
2626
return ed25519KeyPairGenerator().generate(seed)
2727
}

0 commit comments

Comments
 (0)