1 /*
2  * Copyright 2016-2020 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license.
3  */
4 
5 package kotlinx.coroutines.test.internal
6 
7 import kotlinx.coroutines.*
8 import kotlinx.coroutines.internal.*
9 import kotlin.coroutines.*
10 
11 /**
12  * The testable main dispatcher used by kotlinx-coroutines-test.
13  * It is a [MainCoroutineDispatcher] which delegates all actions to a settable delegate.
14  */
15 internal class TestMainDispatcher(private val mainFactory: MainDispatcherFactory) : MainCoroutineDispatcher(), Delay {
16     private var _delegate: CoroutineDispatcher? = null
17     private val delegate: CoroutineDispatcher get() {
<lambda>null18         _delegate?.let { return it }
<lambda>null19         mainFactory.tryCreateDispatcher(emptyList()).let {
20             // If we've failed to create a dispatcher, do no set _delegate
21             if (!isMissing()) {
22                 _delegate = it
23             }
24             return it
25         }
26     }
27 
28     @Suppress("INVISIBLE_MEMBER")
29     private val delay: Delay get() = delegate as? Delay ?: DefaultDelay
30 
31     override val immediate: MainCoroutineDispatcher
32         get() = (delegate as? MainCoroutineDispatcher)?.immediate ?: this
33 
dispatchnull34     override fun dispatch(context: CoroutineContext, block: Runnable) {
35         delegate.dispatch(context, block)
36     }
37 
38     @ExperimentalCoroutinesApi
isDispatchNeedednull39     override fun isDispatchNeeded(context: CoroutineContext): Boolean = delegate.isDispatchNeeded(context)
40 
41     override fun scheduleResumeAfterDelay(timeMillis: Long, continuation: CancellableContinuation<Unit>) {
42         delay.scheduleResumeAfterDelay(timeMillis, continuation)
43     }
44 
delaynull45     override suspend fun delay(time: Long) {
46         delay.delay(time)
47     }
48 
invokeOnTimeoutnull49     override fun invokeOnTimeout(timeMillis: Long, block: Runnable, context: CoroutineContext): DisposableHandle {
50         return delay.invokeOnTimeout(timeMillis, block, context)
51     }
52 
setDispatchernull53     public fun setDispatcher(dispatcher: CoroutineDispatcher) {
54         _delegate = dispatcher
55     }
56 
resetDispatchernull57     public fun resetDispatcher() {
58         _delegate = null
59     }
60 }
61 
62 internal class TestMainDispatcherFactory : MainDispatcherFactory {
63 
createDispatchernull64     override fun createDispatcher(allFactories: List<MainDispatcherFactory>): MainCoroutineDispatcher {
65         val originalFactory = allFactories.asSequence()
66             .filter { it !== this }
67             .maxBy { it.loadPriority } ?: MissingMainCoroutineDispatcherFactory
68         return TestMainDispatcher(originalFactory)
69     }
70 
71     /**
72      * [Int.MAX_VALUE] -- test dispatcher always wins no matter what factories are present in the classpath.
73      * By default all actions are delegated to the second-priority dispatcher, so that it won't be the issue.
74      */
75     override val loadPriority: Int
76         get() = Int.MAX_VALUE
77 }
78