Skip to content

Commit 91b7d9a

Browse files
authored
Simplify module structure and introduce alleycats-retry (#132)
* Simplify module structure and introduce alleycats-retry * Move retrying to alleycats-retry * Fix oversights
1 parent 5e955f9 commit 91b7d9a

File tree

14 files changed

+111
-220
lines changed

14 files changed

+111
-220
lines changed

README.md

+1-2
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,7 @@
22

33
A library for retrying actions that can fail.
44

5-
Designed to work with [cats](https://typelevel.org/cats/) and (optionally)
6-
[cats-effect](https://typelevel.org/cats-effect/) or [Monix](https://monix.io/).
5+
Designed to work with [cats](https://typelevel.org/cats/) and [cats-effect](https://typelevel.org/cats-effect/) or [Monix](https://monix.io/).
76

87
Inspired by the [retry Haskell
98
package](https://hackage.haskell.org/package/retry).

build.sbt

+23-30
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,6 @@ val commonSettings = Seq(
3737
)
3838

3939
val moduleSettings = commonSettings ++ Seq(
40-
moduleName := s"cats-retry-${name.value}",
4140
scalacOptions ++= Seq(
4241
"-Xfuture",
4342
"-Ywarn-dead-code",
@@ -58,6 +57,7 @@ val moduleSettings = commonSettings ++ Seq(
5857
)
5958

6059
val catsVersion = "2.0.0"
60+
val catsEffectVersion = "2.0.0"
6161
val scalatestVersion = "3.1.0"
6262
val scalaTestPlusVersion = "3.1.0.0-RC2"
6363
val scalacheckVersion = "1.14.2"
@@ -67,9 +67,13 @@ val core = crossProject(JVMPlatform, JSPlatform)
6767
.in(file("modules/core"))
6868
.settings(moduleSettings)
6969
.settings(
70+
name := "cats-retry",
7071
crossScalaVersions := scalaVersions,
7172
libraryDependencies ++= Seq(
7273
"org.typelevel" %%% "cats-core" % catsVersion,
74+
"org.typelevel" %%% "cats-effect" % catsEffectVersion,
75+
"org.scalatest" %%% "scalatest" % scalatestVersion % Test,
76+
"org.scalacheck" %%% "scalacheck" % scalacheckVersion % Test,
7377
"org.typelevel" %%% "cats-laws" % catsVersion % Test,
7478
"org.scalatest" %%% "scalatest" % scalatestVersion % Test,
7579
"org.scalatestplus" %%% "scalatestplus-scalacheck" % scalaTestPlusVersion % Test,
@@ -80,42 +84,30 @@ val core = crossProject(JVMPlatform, JSPlatform)
8084
val coreJVM = core.jvm
8185
val coreJS = core.js
8286

83-
val catsEffect = crossProject(JVMPlatform, JSPlatform)
84-
.in(file("modules/cats-effect"))
87+
val alleycatsRetry = crossProject(JVMPlatform, JSPlatform)
88+
.in(file("modules/alleycats"))
8589
.jvmConfigure(_.dependsOn(coreJVM))
8690
.jsConfigure(_.dependsOn(coreJS))
8791
.settings(moduleSettings)
8892
.settings(
93+
name := "alleycats-retry",
8994
crossScalaVersions := scalaVersions,
90-
name := "cats-effect",
91-
libraryDependencies ++= Seq(
92-
"org.typelevel" %%% "cats-effect" % "2.0.0",
93-
"org.scalatest" %%% "scalatest" % scalatestVersion % Test,
94-
"org.scalacheck" %%% "scalacheck" % scalacheckVersion % Test
95-
)
96-
)
97-
val catsEffectJVM = catsEffect.jvm
98-
val catsEffectJS = catsEffect.js
99-
100-
val monix = crossProject(JVMPlatform, JSPlatform)
101-
.in(file("modules/monix"))
102-
.jvmConfigure(_.dependsOn(coreJVM))
103-
.jsConfigure(_.dependsOn(coreJS))
104-
.settings(moduleSettings)
105-
.settings(
106-
crossScalaVersions := List(scalaVersion212, scalaVersion211),
10795
libraryDependencies ++= Seq(
108-
"io.monix" %%% "monix" % "3.1.0",
109-
"org.scalatest" %%% "scalatest" % scalatestVersion % Test,
110-
"org.scalacheck" %%% "scalacheck" % scalacheckVersion % Test
96+
"org.scalatest" %%% "scalatest" % scalatestVersion % Test,
97+
"org.scalacheck" %%% "scalacheck" % scalacheckVersion % Test,
98+
"org.typelevel" %%% "cats-laws" % catsVersion % Test,
99+
"org.scalatest" %%% "scalatest" % scalatestVersion % Test,
100+
"org.scalatestplus" %%% "scalatestplus-scalacheck" % scalaTestPlusVersion % Test,
101+
"org.typelevel" %%% "discipline-scalatest" % disciplineVersion % Test,
102+
"org.scalacheck" %%% "scalacheck" % scalacheckVersion % Test
111103
)
112104
)
113-
val monixJVM = monix.jvm
114-
val monixJS = monix.js
105+
val alleycatsJVM = alleycatsRetry.jvm
106+
val alleycatsJS = alleycatsRetry.js
115107

116108
val docs = project
117109
.in(file("modules/docs"))
118-
.dependsOn(coreJVM, catsEffectJVM, monixJVM)
110+
.dependsOn(coreJVM, alleycatsJVM)
119111
.enablePlugins(MicrositesPlugin, BuildInfoPlugin)
120112
.settings(moduleSettings)
121113
.settings(
@@ -125,6 +117,9 @@ val docs = project
125117
addCompilerPlugin(
126118
"org.typelevel" %% "kind-projector" % "0.11.0" cross CrossVersion.full
127119
),
120+
libraryDependencies ++= Seq(
121+
"io.monix" %%% "monix" % "3.1.0"
122+
),
128123
crossScalaVersions := Nil,
129124
buildInfoPackage := "retry",
130125
publishArtifact := false,
@@ -148,10 +143,8 @@ val root = project
148143
.aggregate(
149144
coreJVM,
150145
coreJS,
151-
catsEffectJVM,
152-
catsEffectJS,
153-
monixJVM,
154-
monixJS,
146+
alleycatsJVM,
147+
alleycatsJS,
155148
docs
156149
)
157150
.settings(commonSettings)

modules/core/jvm/src/main/scala/retry/Sleep.scala modules/alleycats/jvm/src/main/scala/retry/alleycats/instances.scala

+3-11
Original file line numberDiff line numberDiff line change
@@ -1,20 +1,12 @@
11
package retry
2-
3-
import java.util.concurrent.Executors
2+
package alleycats
43

54
import cats.{Eval, Id}
6-
75
import scala.concurrent.duration.FiniteDuration
86
import scala.concurrent.{Future, Promise}
9-
import java.util.concurrent.ThreadFactory
10-
11-
trait Sleep[M[_]] {
12-
def sleep(delay: FiniteDuration): M[Unit]
13-
}
14-
15-
object Sleep {
16-
def apply[M[_]](implicit sleep: Sleep[M]): Sleep[M] = sleep
7+
import java.util.concurrent.{ThreadFactory, Executors}
178

9+
object instances {
1810
implicit val threadSleepId: Sleep[Id] = new Sleep[Id] {
1911
def sleep(delay: FiniteDuration): Id[Unit] = Thread.sleep(delay.toMillis)
2012
}

modules/core/jvm/src/test/scala/retry/PackageObjectLazinessSpec.scala modules/alleycats/jvm/src/test/scala/retry/alleycats/PackageObjectLazinessSpec.scala

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,9 @@
1-
package retry
1+
package retry.alleycats
22

33
import cats.instances.future._
44
import org.scalatest.flatspec.AnyFlatSpec
5+
import retry._
6+
import retry.alleycats.instances._
57

68
import scala.collection.mutable.ArrayBuffer
79
import scala.concurrent.duration._
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
package retry
2+
package alleycats
3+
4+
import cats.{Id, Monad}
5+
6+
object syntax {
7+
def retrying[A](
8+
policy: RetryPolicy[Id],
9+
wasSuccessful: A => Boolean,
10+
onFailure: (A, RetryDetails) => Unit
11+
)(
12+
action: => A
13+
)(
14+
implicit
15+
M: Monad[Id],
16+
S: Sleep[Id]
17+
): A =
18+
retryingM[A][Id](policy, wasSuccessful, onFailure)(action)
19+
}

modules/core/js/src/main/scala/retry/Sleep.scala

-11
This file was deleted.

modules/cats-effect/shared/src/main/scala/retry/CatsEffect.scala modules/core/shared/src/main/scala/retry/Sleep.scala

+7-4
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,17 @@
11
package retry
22

3+
import cats.effect.Timer
34
import scala.concurrent.duration.FiniteDuration
45

5-
import cats.effect.Timer
6+
trait Sleep[M[_]] {
7+
def sleep(delay: FiniteDuration): M[Unit]
8+
}
9+
10+
object Sleep {
11+
def apply[M[_]](implicit sleep: Sleep[M]): Sleep[M] = sleep
612

7-
trait CatsEffect {
813
implicit def sleepUsingTimer[F[_]](implicit timer: Timer[F]): Sleep[F] =
914
new Sleep[F] {
1015
def sleep(delay: FiniteDuration): F[Unit] = timer.sleep(delay)
1116
}
1217
}
13-
14-
object CatsEffect extends CatsEffect

modules/core/shared/src/main/scala/retry/package.scala

-13
Original file line numberDiff line numberDiff line change
@@ -5,19 +5,6 @@ import cats.syntax.flatMap._
55
import scala.concurrent.duration.FiniteDuration
66

77
package object retry {
8-
def retrying[A](
9-
policy: RetryPolicy[Id],
10-
wasSuccessful: A => Boolean,
11-
onFailure: (A, RetryDetails) => Unit
12-
)(
13-
action: => A
14-
)(
15-
implicit
16-
M: Monad[Id],
17-
S: Sleep[Id]
18-
): A =
19-
retryingM[A][Id](policy, wasSuccessful, onFailure)(action)
20-
218
def retryingM[A] = new RetryingPartiallyApplied[A]
229

2310
private[retry] class RetryingPartiallyApplied[A] {

modules/docs/src/main/mdoc/docs/combinators.md

+4-51
Original file line numberDiff line numberDiff line change
@@ -8,58 +8,12 @@ title: Combinators
88
The library offers a few slightly different ways to wrap your operations with
99
retries.
1010

11-
## `retrying`
12-
13-
This is useful when you are not working in a monadic context. You have an
14-
operation that simply returns a value of some type `A`, and you want to retry
15-
until it returns a value that you are happy with.
16-
17-
To use `retrying`, you pass in a predicate that decides whether you are
18-
happy with the result or you want to retry.
19-
20-
The API looks like this:
21-
22-
```scala
23-
def retrying[A](policy: RetryPolicy[Id],
24-
wasSuccessful: A => Boolean,
25-
onFailure: (A, RetryDetails) => Unit)
26-
(action: => A): A
27-
```
28-
29-
You need to pass in:
30-
31-
* a retry policy
32-
* a predicate that decides whether the operation was successful
33-
* a failure handler, often used for logging
34-
* the operation that you want to wrap with retries
35-
36-
For example, let's keep rolling a die until we get a six.
37-
38-
```scala mdoc:reset-class
39-
import cats.Id
40-
import retry._
41-
import scala.concurrent.duration._
42-
43-
val policy = RetryPolicies.constantDelay[Id](10.milliseconds)
44-
45-
val predicate = (_: Int) == 6
46-
47-
def onFailure(failedValue: Int, details: RetryDetails): Unit = {
48-
println(s"Rolled a $failedValue, retrying ...")
49-
}
50-
51-
val loadedDie = util.LoadedDie(2, 5, 4, 1, 3, 2, 6)
52-
53-
retrying(policy, predicate, onFailure){
54-
loadedDie.roll()
55-
}
56-
```
57-
5811
## `retryingM`
5912

60-
This is similar to `retrying`, but is useful when you are working in an
61-
arbitrary `Monad` that is not a `MonadError`. Your operation doesn't throw
62-
errors, but you want to retry until it returns a value that you are happy with.
13+
To use `retryingM`, you pass in a predicate that decides whether you are
14+
happy with the result or you want to retry.
15+
It is useful when you are working in an arbitrary `Monad` that is not a `MonadError`.
16+
Your operation doesn't throw errors, but you want to retry until it returns a value that you are happy with.
6317

6418
The API (modulo some type-inference trickery) looks like this:
6519

@@ -84,7 +38,6 @@ import cats.effect.{ContextShift, IO, Timer}
8438
import scala.concurrent.duration._
8539
import scala.concurrent.ExecutionContext.global
8640
import retry._
87-
import retry.CatsEffect._
8841

8942
// We need an implicit cats.effect.Timer
9043
implicit val timer: Timer[IO] = IO.timer(global)

modules/docs/src/main/mdoc/docs/index.md

+1-6
Original file line numberDiff line numberDiff line change
@@ -35,10 +35,7 @@ println(
3535
s"""
3636
|```
3737
|val catsRetryVersion = "${retry.BuildInfo.version.replaceFirst("\\+.*", "")}"
38-
|libraryDependencies ++= Seq(
39-
| "com.github.cb372" %% "cats-retry-core" % catsRetryVersion,
40-
| "com.github.cb372" %% "cats-retry-cats-effect" % catsRetryVersion
41-
|)
38+
|libraryDependencies += "com.github.cb372" %% "cats-retry" % catsRetryVersion,
4239
|```
4340
|""".stripMargin.trim
4441
)
@@ -98,8 +95,6 @@ import cats.effect.Timer
9895
import scala.concurrent.ExecutionContext.global
9996
implicit val timer: Timer[IO] = IO.timer(global)
10097

101-
// This is so we can use that Timer to perform delays between retries
102-
import retry.CatsEffect._
10398

10499
val flakyRequestWithRetry: IO[String] =
105100
retryingOnAllErrors[String](

0 commit comments

Comments
 (0)