/* * ============LICENSE_START======================================================= * dcaegen2-collectors-veshv * ================================================================================ * Copyright (C) 2018 NOKIA * ================================================================================ * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * ============LICENSE_END========================================================= */ @file:Suppress("TooManyFunctions") package org.onap.dcae.collectors.veshv.utils.arrow import arrow.core.Either import arrow.core.ForOption import arrow.core.Option import arrow.core.Try import arrow.core.extensions.option.monad.monad import arrow.core.fix import arrow.core.identity import arrow.syntax.collections.firstOption import arrow.typeclasses.MonadContinuation import reactor.core.publisher.Flux import reactor.core.publisher.Mono import java.util.concurrent.atomic.AtomicReference /** * @author Piotr Jaszczyk * @since July 2018 */ object OptionUtils { fun binding(c: suspend MonadContinuation.() -> A) : Option = Option.monad().binding(c).fix() } fun Either.flatten() = fold(::identity, ::identity) fun Either.rightOrThrow() = fold({ throw it }, ::identity) fun Either.rightOrThrow(mapper: (A) -> Throwable) = fold({ throw mapper(it) }, ::identity) fun Flux>.throwOnLeft(): Flux = map { it.rightOrThrow() } fun Flux>.throwOnLeft(f: (A) -> Exception): Flux = map { it.rightOrThrow(f) } fun Mono>.throwOnLeft(f: (A) -> Exception): Mono = map { it.rightOrThrow(f) } fun AtomicReference.getOption() = Option.fromNullable(get()) fun Option.Companion.fromNullablesChain(firstValue: A?, vararg nextValues: () -> A?): Option = if (firstValue != null) Option.just(firstValue) else nextValues.asSequence() .map { it() } .filter { it != null } .firstOption() fun Either.doOnLeft(action: () -> Unit): Either = apply { if (isLeft()) action() } fun Option.doOnEmpty(action: () -> Unit): Option = apply { if (isEmpty()) action() } fun Try.doOnFailure(action: (Throwable) -> Unit): Try = apply { if (this is Try.Failure) { action(exception) } } fun A.mapBinding(c: suspend MonadContinuation.(A) -> B) : Option = let { OptionUtils.binding { c(it) } } fun Option.flatFold(ifEmptyOrFalse: () -> T, ifTrue: () -> T) = fold({ ifEmptyOrFalse() }, { if (it) { ifTrue() } else { ifEmptyOrFalse() } })