File: ContinuationSerializationTest.kt

package info (click to toggle)
kotlinx-coroutines 1.0.1-2
  • links: PTS, VCS
  • area: main
  • in suites: bookworm, forky, sid, trixie
  • size: 4,628 kB
  • sloc: xml: 418; sh: 322; javascript: 60; makefile: 17; java: 8
file content (94 lines) | stat: -rw-r--r-- 2,622 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
/*
 * Copyright 2016-2018 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license.
 */

package kotlinx.coroutines


import com.esotericsoftware.kryo.*
import com.esotericsoftware.kryo.io.*
import kotlinx.atomicfu.*
import org.junit.Test
import org.objenesis.strategy.*
import java.io.*
import kotlin.coroutines.*
import kotlin.coroutines.intrinsics.*
import kotlin.reflect.*
import kotlin.test.*

@Ignore
class ContinuationSerializationTest : TestBase() {

    companion object {
        @JvmStatic
        var flag = false
    }

//    private val atomicInt = atomic(1)

    private val kryo =
        Kryo().also { it.instantiatorStrategy = Kryo.DefaultInstantiatorStrategy(StdInstantiatorStrategy()) }

    private var storage: ByteArrayOutputStream = ByteArrayOutputStream()

    @Test
    fun testSafeContinuationSerDe() = testSerization(::serialize, {
        it.javaClass.getDeclaredField("result").apply {
            isAccessible = true
            set(it, COROUTINE_SUSPENDED)
        }
    })

    @Test
    fun testUnsafeContinuationSerDe() = testSerization(::serializeUnsafe, {})

//    @Test
//    fun testCancellableContinuationSerDe() = testSerization(::serializeCancellable, {
//        it.javaClass.superclass.getDeclaredField("_decision").apply {
//            isAccessible = true
//            set(it, atomicInt)
//        }
//    })

    private suspend fun serialize() = suspendCoroutine<Unit> { cont ->
        Output(storage).use {
            kryo.writeClassAndObject(it, cont)
        }
    }

    private suspend fun serializeCancellable() = suspendCancellableCoroutine<Unit> { cont ->
        Output(storage).use {
            kryo.writeClassAndObject(it, cont)
        }
    }

    private suspend fun serializeUnsafe() = suspendCoroutineUninterceptedOrReturn<Unit> { cont ->
        Output(storage).use {
            kryo.writeClassAndObject(it, cont)
        }
    }

    private fun testSerization(serialize: KSuspendFunction0<Unit>, patcher: (Continuation<Unit>) -> Unit) =
        runBlocking {
            launch(Unconfined) {
                expect(1)
                serialize()
                flag = true
            }

            val cont = deserialise()
            patcher(cont)
            expect(2)
            cont.resume(Unit)
            finish(3)
            assertTrue(flag)
        }

    @Suppress("UNCHECKED_CAST")
    private fun deserialise(): Continuation<Unit> {
        val input = Input(ByteArrayInputStream(storage.toByteArray()))
        input.use {
            return kryo.readClassAndObject(it) as Continuation<Unit>
        }
    }
}