graphql.nadel.engine.util.CollectionUtil.kt Maven / Gradle / Ivy
package graphql.nadel.engine.util
import java.util.Collections
import kotlin.collections.component1
import kotlin.collections.component2
import kotlin.collections.set
internal typealias PairList = List>
/**
* Like [singleOrNull] but the single item must be of type [T].
*/
inline fun Collection<*>.singleOfTypeOrNull(predicate: (T) -> Boolean = { true }): T? {
return singleOrNull { it is T && predicate(it) } as T?
}
/**
* Like [singleOrNull] but the single item must be of type [T].
*/
inline fun Collection<*>.singleOfType(predicate: (T) -> Boolean = { true }): T {
return singleOfTypeOrNull(predicate)!!
}
/**
* Like [singleOrNull] but the single item must be of type [T].
*/
inline fun Sequence<*>.singleOfTypeOrNull(predicate: (T) -> Boolean = { true }): T? {
return singleOrNull { it is T && predicate(it) } as T?
}
/**
* Like [singleOrNull] but the single item must be of type [T].
*/
inline fun Sequence<*>.singleOfType(predicate: (T) -> Boolean = { true }): T {
return singleOfTypeOrNull(predicate)!!
}
inline fun Iterable.strictAssociateBy(crossinline keyExtractor: (E) -> K): Map {
return mapFrom(
map {
keyExtractor(it) to it
}
)
}
inline fun Sequence.strictAssociateBy(crossinline keyExtractor: (E) -> K): Map {
val map = mutableMapOf()
var count = 0
forEach {
map[keyExtractor(it)] = it
count++
}
require(map.size == count)
return Collections.unmodifiableMap(map)
}
/**
* Like [mapOf] but takes in a [Collection] instead of vararg. Useful if your input
* into the [Map] is [List.map]ped from another [Collection].
*/
@JvmName("mapFromPairs")
fun mapFrom(entries: Collection>): Map {
val map = HashMap(entries.size)
map.putAll(entries)
require(map.size == entries.size) {
@Suppress("SimpleRedundantLet") // For debugging purposes if you want to visit the values
"Duplicate keys: " + entries.groupBy { it.first }.filterValues { it.size > 1 }.let {
it.keys
}
}
return map
}
/**
* Like [mapOf] but takes in a [Collection] instead of vararg. Useful if your input
* into the [Map] is [List.map]ped from another [Collection].
*/
@JvmName("mapFromPairs")
fun mapFrom(entries: Sequence>): Map {
val map = HashMap()
var count = 0
entries.forEach {
map[it.first] = it.second
count++
}
require(map.size == count) {
@Suppress("SimpleRedundantLet") // For debugging purposes if you want to visit the values
"Duplicate keys: " + entries.groupBy { it.first }.filterValues { it.size > 1 }.let {
it.keys
}
}
return map
}
fun Sequence>.toMapStrictly(): Map {
return mapFrom(entries = this)
}
fun Collection>.toMapStrictly(): Map {
return mapFrom(entries = this)
}
/**
* Like [mapOf] but takes in a [Collection] instead of vararg. Useful if your input
* into the [Map] is [List.map]ped from another [Collection].
*/
@JvmName("mapFromEntries")
fun mapFrom(entries: Collection>): Map {
val map = HashMap(entries.size)
entries.forEach(map::put)
require(map.size == entries.size)
return map
}
/**
* Utility function to set the [Map.Entry.key] to [Map.Entry.value] in the given [MutableMap].
*/
fun MutableMap.put(entry: Map.Entry) {
put(entry.key, entry.value)
}
/**
* Utility function to set the [Map.Entry.key] to [Map.Entry.value] in the given [MutableMap].
*/
fun MutableMap.put(entry: Pair) {
put(entry.first, entry.second)
}
/**
* Inverts the [Map] such that the values are now the keys and the keys are now the values.
*/
fun Map.invert(): Map {
val map = HashMap(this.size)
forEach { (key, value) ->
map[value] = key
}
require(map.size == this.size)
return map
}
/**
* Try to replace the function body with just:
*
* ```kotlin
*putAll(map)
* ```
*
* If it works then yay we don't need this function anymore, but right now I'm getting:
*
* Type mismatch
*
* Required: Map
*
* Found: Map<*, *>
*/
fun AnyMutableMap.hackPutAll(map: AnyMap) {
@Suppress("UNCHECKED_CAST")
(this as MutableMap).putAll(map)
}
/**
* This function permits an empty collection or a [single] object in the collection.
*
* @return null if empty or [single] object
* @see single
*/
fun Iterable.emptyOrSingle(): T? {
return when (this) {
is List -> when (isEmpty()) {
true -> null
else -> single()
}
else -> iterator().emptyOrSingle()
}
}
/**
* See `Iterable`.[emptyOrSingle]
*/
fun Iterator.emptyOrSingle(): T? {
return when (hasNext()) {
true -> next().also {
if (hasNext()) {
// Copied from List.single
throw IllegalArgumentException("List has more than one element.")
}
}
else -> null
}
}
/**
* See `Iterable`.[emptyOrSingle]
*/
fun Sequence.emptyOrSingle(): T? {
return iterator().emptyOrSingle()
}
inline fun Map.filterValuesOfType(): Map {
@Suppress("UNCHECKED_CAST")
return filterValues {
it is T
} as Map
}
fun Sequence.flatten(recursively: Boolean): Sequence {
return flatMap { element ->
when (element) {
is AnyMap -> sequenceOf(element)
is AnyIterable -> element.asSequence().let {
if (recursively) {
it.flatten(recursively = true)
} else {
it
}
}
else -> sequenceOf(element)
}
}
}
/**
* See [List.subList], but if input is out of bounds then null is returned instead.
*/
fun List.subListOrNull(fromIndex: Int, toIndex: Int): List? {
if (fromIndex < 0 || /*toIndex is exclusive, hence minus 1*/ toIndex - 1 > lastIndex) {
return null
}
return subList(fromIndex = fromIndex, toIndex = toIndex)
}
fun Iterable.foldWhileNotNull(initial: R?, operation: (acc: R, T) -> R?): R? {
var accumulator = initial ?: return null
for (element in this) {
accumulator = operation(accumulator, element) ?: return null
}
return accumulator
}
fun listOfNulls(size: Int): List {
return ArrayList(size).also {
for (i in 1..size) {
it.add(null)
}
}
}
fun sequenceOfNulls(size: Int): Sequence {
return sequence {
for (i in 1..size) {
yield(null)
}
}
}
/**
* Similar to [Sequence.all] but it requires at least [min] matching elements to pass.
*/
fun Sequence.all(min: Int, predicate: (T) -> Boolean): Boolean {
var count = 0
for (element in this) {
if (!predicate(element)) return false
count++
}
return count >= min
}
fun Sequence>.filterPairSecondNotNull(): Sequence> {
return mapNotNull { pair ->
if (pair.second == null) {
null
} else {
@Suppress("UNCHECKED_CAST")
pair as Pair
}
}
}
/**
* Like [List.partition] but only returns the count of each partition.
*/
internal fun List.partitionCount(predicate: (E) -> Boolean): Pair {
var first = 0
var second = 0
for (element in this) {
if (predicate(element)) {
first++
} else {
second++
}
}
return first to second
}
/**
* Like [Sequence.zip] but throws an exception when the two sequences do not have the same number
* of items to join.
*/
internal inline fun Sequence.zipOrThrow(
other: Sequence,
crossinline errorFunction: () -> Nothing,
): Sequence> {
return zipOrThrow(
other = object : Iterable {
override fun iterator(): Iterator {
return other.iterator()
}
},
errorFunction = errorFunction,
)
}
/**
* Like [Sequence.zip] but throws an exception when the two sequences do not have the same number
* of items to join.
*/
internal inline fun Sequence.zipOrThrow(
other: Iterable,
crossinline errorFunction: () -> Nothing,
): Sequence> {
val sequenceA = this
return object : Sequence> {
override fun iterator(): Iterator> {
val iteratorA = sequenceA.iterator()
val iteratorB = other.iterator()
return object : Iterator> {
override fun hasNext(): Boolean {
return when {
iteratorA.hasNext() && iteratorB.hasNext() -> true
iteratorA.hasNext() || iteratorB.hasNext() -> errorFunction()
else -> false
}
}
override fun next(): Pair {
return iteratorA.next() to iteratorB.next()
}
}
}
}
}
internal fun List.startsWith(other: List): Boolean {
return if (size >= other.size) {
asSequence()
.zip(other.asSequence())
.all { (a, b) -> a == b }
} else {
false
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy