Skip to content

Commit

Permalink
Merge pull request #612 from kevin-lee/task/608/TimeSource-ce2
Browse files Browse the repository at this point in the history
Close #608 - [`effectie-cats-effect2-time`] Add `TimeSource` with `Clock` from cats-effect 2
  • Loading branch information
kevin-lee authored Jan 9, 2024
2 parents 5f5585f + 3956e45 commit e1b6b79
Show file tree
Hide file tree
Showing 2 changed files with 207 additions and 0 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
package effectie.instances.ce2

import cats.Monad
import cats.effect.Clock
import cats.syntax.all._
import effectie.time.TimeSource

import java.time.Instant
import scala.concurrent.duration.{FiniteDuration, MILLISECONDS, NANOSECONDS, TimeUnit}

/** @author Kevin Lee
* @since 2024-01-09
*/
trait ClockBasedTimeSource[F[*]] extends TimeSource[F] {

def clock: Clock[F]

override val name: String = "ce2.ClockBasedTimeSource"

override def currentTime(): F[Instant] = clock.instantNow

override def realTimeTo(unit: TimeUnit): F[FiniteDuration] =
clock.realTime(unit).map(FiniteDuration(_, unit))

override def monotonicTo(unit: TimeUnit): F[FiniteDuration] =
clock.monotonic(unit).map(FiniteDuration(_, unit))

override def realTime: F[FiniteDuration] = realTimeTo(MILLISECONDS)

override def monotonic: F[FiniteDuration] = monotonicTo(NANOSECONDS)

override val toString: String = name
}
object ClockBasedTimeSource {
def apply[F[*]](implicit clock: Clock[F], monad: Monad[F]): ClockBasedTimeSource[F] =
new ClockBasedTimeSourceF[F](clock)(monad)

private final class ClockBasedTimeSourceF[F[*]](
override val clock: Clock[F]
)(
override implicit protected val M: Monad[F]
) extends ClockBasedTimeSource[F]
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,164 @@
package effectie.instances.ce2

import cats.effect.{IO, Timer}
import cats.syntax.all._
import effectie.time.syntax._
import hedgehog._
import hedgehog.runner._

import java.time.Instant
import scala.concurrent.duration._

/** @author Kevin Lee
* @since 2024-01-09
*/
object ClockBasedTimeSourceSpec extends Properties {

type F[A] = IO[A]
val F = IO

@SuppressWarnings(Array("org.wartremover.warts.GlobalExecutionContext"))
implicit val timer: Timer[F] = F.timer(scala.concurrent.ExecutionContext.global)

override def tests: List[Test] = List(
example("test currentTime", testCurrentTime),
example("test realTime", testRealTime),
example("test monotonic", testMonotonic),
property("test timeSpent", testTimeSpent).withTests(count = 5),
)

def testCurrentTime: Result = {
ClockBasedTimeSource[F]
.currentTime()
.flatMap(actual =>
timer
.clock
.instantNow
.map { expected =>
Result.diff(actual.toEpochMilli.milliseconds, (expected.toEpochMilli.milliseconds +- 500.milliseconds))(
_.isWithIn(_)
)
}
)
.unsafeRunSync()

}

def testRealTime: Result = {

val now = Instant.now()
val expectedNanos = now.getEpochSecond * 1000000000L + now.getNano
val expectedMillis = expectedNanos / 1000000L

val timeSource = ClockBasedTimeSource[F]

List(
timeSource
.realTime
.map(time =>
Result
.diff(time, (expectedMillis.milliseconds +- 1000.milliseconds))(_.isWithIn(_))
.log("timeSource.realTime"),
),
timeSource
.realTimeTo(MILLISECONDS)
.map(time =>
Result
.diff(time, (expectedMillis.milliseconds +- 1000.milliseconds))(_.isWithIn(_))
.log("timeSource.realTimeTo(MILLISECONDS)"),
),
timeSource
.realTimeTo(NANOSECONDS)
.map(time =>
Result
.diff(time, (expectedNanos.nanoseconds +- 1000.milliseconds))(_.isWithIn(_))
.log("timeSource.realTimeTo(NANOSECONDS)"),
),
).sequence
.map(Result.all)
.unsafeRunSync()
}

def testMonotonic: Result = {

val expectedNanos = System.nanoTime()
val expectedMillis = expectedNanos / 1000000L

val timeSource = ClockBasedTimeSource[F]

List(
timeSource
.monotonic
.map(time =>
Result
.diff(time, (expectedNanos.nanoseconds +- 1000.milliseconds))(_.isWithIn(_))
.log("timeSource.monotonic"),
),
timeSource
.monotonicTo(MILLISECONDS)
.map(time =>
Result
.diff(time, (expectedMillis.milliseconds +- 1000.milliseconds))(_.isWithIn(_))
.log("timeSource.monotonicTo(MILLISECONDS)"),
),
timeSource
.monotonicTo(NANOSECONDS)
.map(time =>
Result
.diff(time, (expectedNanos.nanoseconds +- 1000.milliseconds))(_.isWithIn(_))
.log("timeSource.monotonicTo(NANOSECONDS)"),
),
).sequence
.map(Result.all)
.unsafeRunSync()
}

def testTimeSpent: Property = {
for {
waitFor <- Gen.int(Range.linear(200, 700)).map(_.milliseconds).log("waitFor")
diff <- Gen.constant(180.milliseconds).log("diff")
} yield {
val timeSource = ClockBasedTimeSource[F]

for {
resultAndTimeSpent <- timeSource.timeSpent {
F.sleep(waitFor) *>
F.pure("Done")
}
(result, timeSpent) = resultAndTimeSpent
_ <- F.delay(
println(
show""">>> waitFor: $waitFor
|>>> timeSpent: ${timeSpent.timeSpent.toMillis} milliseconds
|>>> diff: ${(timeSpent.timeSpent - waitFor).toMillis} milliseconds
|""".stripMargin
)
)
} yield {
Result.all(
List(
result ==== "Done",
Result
.diffNamed(
s"timeSpent (${timeSpent.timeSpent.toMillis.show} milliseconds) should be " +
s"within ${(waitFor - diff).show} to ${(waitFor + diff).show}.",
timeSpent,
(waitFor +- diff),
)(_.timeSpent.isWithIn(_))
.log(
s"""--- diff test log ---
|> actual: ${timeSpent.timeSpent.toMillis.show} milliseconds
|> expected range: ${(waitFor - diff).show} to ${(waitFor + diff).show}
|> waitFor: ${waitFor.show}
|> expected diff: +- ${diff.show})
|> actual diff: ${(timeSpent.timeSpent - waitFor).toMillis.show} milliseconds
|""".stripMargin
),
)
)
}
}
.unsafeRunSync()
}

}

0 comments on commit e1b6b79

Please sign in to comment.