commonMain.com.apollographql.apollo3.api.ExecutionContext.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of apollo-api-jvm Show documentation
Show all versions of apollo-api-jvm Show documentation
Apollo GraphQL API classes
package com.apollographql.apollo3.api
import com.apollographql.apollo3.api.ExecutionContext.Element
import com.apollographql.apollo3.api.ExecutionContext.Key
import kotlin.jvm.JvmField
/**
* A context of GraphQL operation execution, represented as a set of [Key] keys and corresponding [Element] values.
*
* It is inspired by the coroutines Context and allows to pass arbitrary data to interceptors.
*/
interface ExecutionContext {
/**
* Returns the element with the given [key] from this context or `null`.
*/
operator fun get(key: Key): E?
/**
* Accumulates entries of this context starting with [initial] value and applying [operation] from left to right to current accumulator
* value and each element of this context.
*/
fun fold(initial: R, operation: (R, Element) -> R): R
/**
* Returns a context containing elements from this context and elements from other [context].
* The elements from this context with the same key as in the other one are dropped.
*/
operator fun plus(context: ExecutionContext): ExecutionContext {
return if (context === EmptyExecutionContext) this else {
context.fold(this) { acc, element ->
val removed = acc.minusKey(element.key)
if (removed === EmptyExecutionContext) element else {
CombinedExecutionContext(removed, element)
}
}
}
}
/**
* Returns a context containing elements from this context, but without an element with the specified [key].
*/
fun minusKey(key: Key<*>): ExecutionContext
/**
* Key for the elements of [ExecutionContext]. [E] is a type of element with this key.
*/
interface Key
/**
* An element of the [ExecutionContext]. An element of the execution context is a singleton context by itself.
*/
interface Element : ExecutionContext {
/**
* A key of this execution context element.
*/
val key: Key<*>
override operator fun get(key: Key): E? {
@Suppress("UNCHECKED_CAST")
return if (this.key == key) this as E else null
}
override fun fold(initial: R, operation: (R, Element) -> R): R {
return operation(initial, this)
}
override fun minusKey(key: Key<*>): ExecutionContext {
return if (this.key == key) EmptyExecutionContext else this
}
}
companion object {
@JvmField
val Empty: ExecutionContext = EmptyExecutionContext
}
}
internal object EmptyExecutionContext : ExecutionContext {
override fun get(key: Key): E? = null
override fun fold(initial: R, operation: (R, Element) -> R): R = initial
override fun plus(context: ExecutionContext): ExecutionContext = context
override fun minusKey(key: Key<*>): ExecutionContext = this
}
internal class CombinedExecutionContext(
private val left: ExecutionContext,
private val element: Element,
) : ExecutionContext {
override fun get(key: Key): E? {
var cur = this
while (true) {
cur.element[key]?.let { return it }
val next = cur.left
if (next is CombinedExecutionContext) {
cur = next
} else {
return next[key]
}
}
}
override fun fold(initial: R, operation: (R, Element) -> R): R =
operation(left.fold(initial, operation), element)
override fun minusKey(key: Key<*>): ExecutionContext {
element[key]?.let { return left }
val newLeft = left.minusKey(key)
return when {
newLeft === left -> this
newLeft === EmptyExecutionContext -> element
else -> CombinedExecutionContext(newLeft, element)
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy