aboutsummaryrefslogtreecommitdiff
path: root/src/main/scala/xyz/driver/pdsuicommon/computation
diff options
context:
space:
mode:
authorvlad <vlad@driver.xyz>2017-06-27 17:13:02 -0700
committervlad <vlad@driver.xyz>2017-06-27 17:13:02 -0700
commit5832f63b84d7388441d1200f2442dc1e9de0225c (patch)
tree32f63acdc920c14effc3d0d2822c05c125ad49e4 /src/main/scala/xyz/driver/pdsuicommon/computation
parent9dd50590d4c8f8b9442d7c21ddd1def9dd453d5e (diff)
downloadrest-query-5832f63b84d7388441d1200f2442dc1e9de0225c.tar.gz
rest-query-5832f63b84d7388441d1200f2442dc1e9de0225c.tar.bz2
rest-query-5832f63b84d7388441d1200f2442dc1e9de0225c.zip
All PDS UI domain models, API case classes, service traits and necessary utils moved to pdsui-commonv0.1.11
Diffstat (limited to 'src/main/scala/xyz/driver/pdsuicommon/computation')
-rw-r--r--src/main/scala/xyz/driver/pdsuicommon/computation/Computation.scala110
-rw-r--r--src/main/scala/xyz/driver/pdsuicommon/computation/FutureToComputationOps.scala24
-rw-r--r--src/main/scala/xyz/driver/pdsuicommon/computation/Implicits.scala15
-rw-r--r--src/main/scala/xyz/driver/pdsuicommon/computation/TryToComputationOps.scala15
4 files changed, 164 insertions, 0 deletions
diff --git a/src/main/scala/xyz/driver/pdsuicommon/computation/Computation.scala b/src/main/scala/xyz/driver/pdsuicommon/computation/Computation.scala
new file mode 100644
index 0000000..ad458de
--- /dev/null
+++ b/src/main/scala/xyz/driver/pdsuicommon/computation/Computation.scala
@@ -0,0 +1,110 @@
+package xyz.driver.pdsuicommon.computation
+
+import scala.concurrent.{ExecutionContext, Future}
+
+/**
+ * Takes care of computations
+ *
+ * Success(either) - the computation will be continued.
+ * Failure(error) - the computation was failed with unhandled error.
+ *
+ * Either[Result, T]:
+ * Left(result) is a final and handled result, another computations (map, flatMap) will be ignored.
+ * Right(T) is a current result. Functions in map/flatMap will continue the computation.
+ *
+ * Example:
+ * {{{
+ * import scala.concurrent.ExecutionContext.Implicits.global
+ * import scala.concurrent.{ExecutionContext, Future}
+ * import com.drivergrp.server.com.drivergrp.server.common.utils.Computation
+ *
+ * def successful = for {
+ * x <- Computation.continue(1)
+ * y <- Computation.continue(2)
+ * } yield s"\$x + \$y"
+ *
+ * // Prints "Success(1 + 2)"
+ * successful.join.onComplete(print)
+ *
+ * def failed = for {
+ * x <- Computation.abort("Failed on x")
+ * _ = print("Second step")
+ * y <- Computation.continue(2)
+ * } yield s"\$x + \$y"
+ *
+ * // Prints "Success(Failed on x)"
+ * failed.join.onComplete(print)
+ * }}}
+ *
+ * TODO: Make future private
+ *
+ * @param future The final flow in a future.
+ * @tparam R Type of result for aborted computation.
+ * @tparam T Type of result for continued computation.
+ */
+final case class Computation[+R, +T](future: Future[Either[R, T]]) {
+
+ def flatMap[R2, T2](f: T => Computation[R2, T2])(implicit ec: ExecutionContext, ev: R <:< R2): Computation[R2, T2] = {
+ Computation(future.flatMap {
+ case Left(x) => Future.successful(Left(x))
+ case Right(x) => f(x).future
+ })
+ }
+
+ def processExceptions[R2](f: PartialFunction[Throwable, R2])(implicit ev1: R <:< R2,
+ ec: ExecutionContext): Computation[R2, T] = {
+ val strategy = f.andThen(x => Left(x): Either[R2, T])
+ val castedFuture: Future[Either[R2, T]] = future.map {
+ case Left(x) => Left(x)
+ case Right(x) => Right(x)
+ }
+ Computation(castedFuture.recover(strategy))
+ }
+
+ def map[T2](f: T => T2)(implicit ec: ExecutionContext): Computation[R, T2] = flatMap { a =>
+ Computation.continue(f(a))
+ }
+
+ def andThen(f: T => Any)(implicit ec: ExecutionContext): Computation[R, T] = map { a =>
+ f(a)
+ a
+ }
+
+ def filter(f: T => Boolean)(implicit ec: ExecutionContext): Computation[R, T] = map { a =>
+ if (f(a)) a
+ else throw new NoSuchElementException("When filtering")
+ }
+
+ def withFilter(f: T => Boolean)(implicit ec: ExecutionContext): Computation[R, T] = filter(f)
+
+ def foreach[T2](f: T => T2)(implicit ec: ExecutionContext): Unit = future.foreach {
+ case Right(x) => f(x)
+ case _ =>
+ }
+
+ def toFuture[R2](resultFormatter: T => R2)(implicit ec: ExecutionContext, ev: R <:< R2): Future[R2] = future.map {
+ case Left(x) => x
+ case Right(x) => resultFormatter(x)
+ }
+
+ def toFuture[R2](implicit ec: ExecutionContext, ev1: R <:< R2, ev2: T <:< R2): Future[R2] = future.map {
+ case Left(x) => x
+ case Right(x) => x
+ }
+
+}
+
+object Computation {
+
+ def continue[T](x: T): Computation[Nothing, T] = Computation(Future.successful(Right(x)))
+
+ def abort[R](result: R): Computation[R, Nothing] = Computation(Future.successful(Left(result)))
+
+ def fail(exception: Throwable): Computation[Nothing, Nothing] = Computation(Future.failed(exception))
+
+ def fromFuture[T](input: Future[T])(implicit ec: ExecutionContext): Computation[Nothing, T] = Computation {
+ input.map { x =>
+ Right(x)
+ }
+ }
+}
diff --git a/src/main/scala/xyz/driver/pdsuicommon/computation/FutureToComputationOps.scala b/src/main/scala/xyz/driver/pdsuicommon/computation/FutureToComputationOps.scala
new file mode 100644
index 0000000..c5800dc
--- /dev/null
+++ b/src/main/scala/xyz/driver/pdsuicommon/computation/FutureToComputationOps.scala
@@ -0,0 +1,24 @@
+package xyz.driver.pdsuicommon.computation
+
+import xyz.driver.pdsuicommon.error.DomainError
+
+import scala.concurrent.{ExecutionContext, Future}
+
+final class FutureToComputationOps[T](val self: Future[T]) extends AnyVal {
+
+ def handleDomainError[U, ER](f: PartialFunction[T, U])
+ (implicit unsuitableToErrorsResponse: DomainError => ER,
+ ec: ExecutionContext): Future[Either[ER, U]] = {
+ self.map {
+ case x if f.isDefinedAt(x) => Right(f(x))
+ case x: DomainError => Left(unsuitableToErrorsResponse(x))
+ case x => throw new RuntimeException(s"Can not process $x")
+ }
+ }
+
+ def toComputation[U, ER](f: PartialFunction[T, U])
+ (implicit unsuitableToErrorsResponse: DomainError => ER,
+ ec: ExecutionContext): Computation[ER, U] = {
+ Computation(handleDomainError(f))
+ }
+}
diff --git a/src/main/scala/xyz/driver/pdsuicommon/computation/Implicits.scala b/src/main/scala/xyz/driver/pdsuicommon/computation/Implicits.scala
new file mode 100644
index 0000000..d5acc2d
--- /dev/null
+++ b/src/main/scala/xyz/driver/pdsuicommon/computation/Implicits.scala
@@ -0,0 +1,15 @@
+package xyz.driver.pdsuicommon.computation
+
+import scala.concurrent.Future
+import scala.util.Try
+
+trait Implicits {
+
+ implicit def futureToFutureComputationOps[T](self: Future[T]): FutureToComputationOps[T] = {
+ new FutureToComputationOps[T](self)
+ }
+
+ implicit def tryToTryComputationOps[T](self: Try[T]): TryToComputationOps[T] = {
+ new TryToComputationOps[T](self)
+ }
+}
diff --git a/src/main/scala/xyz/driver/pdsuicommon/computation/TryToComputationOps.scala b/src/main/scala/xyz/driver/pdsuicommon/computation/TryToComputationOps.scala
new file mode 100644
index 0000000..8282bc6
--- /dev/null
+++ b/src/main/scala/xyz/driver/pdsuicommon/computation/TryToComputationOps.scala
@@ -0,0 +1,15 @@
+package xyz.driver.pdsuicommon.computation
+
+import scala.concurrent.ExecutionContext
+import scala.util.control.NonFatal
+import scala.util.{Failure, Success, Try}
+
+final class TryToComputationOps[T](val self: Try[T]) extends AnyVal {
+
+ def toComputation[ER](implicit exceptionToErrorResponse: Throwable => ER,
+ ec: ExecutionContext): Computation[ER, T] = self match {
+ case Success(x) => Computation.continue(x)
+ case Failure(NonFatal(e)) => Computation.abort(exceptionToErrorResponse(e))
+ case Failure(e) => Computation.fail(e)
+ }
+}