diff --git a/.travis.yml b/.travis.yml index 19c15f9b..99cfafd1 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,7 +1,7 @@ language: scala scala: -- 2.11.12 -- 2.12.8 +- 2.12.10 +- 2.13.1 jdk: - openjdk8 - openjdk11 diff --git a/build.sbt b/build.sbt index 21edc00a..ca18d4f6 100644 --- a/build.sbt +++ b/build.sbt @@ -16,19 +16,31 @@ lazy val rho = project lazy val `rho-core` = project .in(file("core")) - .settings(rhoPreviousArtifacts(lastVersion = "0.19.0", "core")) - .settings(buildSettings: _*) + .settings(mimaConfiguration) + .settings(buildSettings ++ Seq( + Compile / unmanagedSourceDirectories ++= { + val baseDir = baseDirectory.value + + val mainSrcDir = "src/main/scala" + CrossVersion.partialVersion(scalaVersion.value) match { + case Some((2, minor)) if minor <= 12 => Seq(baseDir / s"$mainSrcDir-2.12-") + case Some((2, minor)) if minor >= 13 => Seq(baseDir / s"$mainSrcDir-2.13+") + case _ => Nil + } + }, + libraryDependencies ++= Seq("org.scala-lang.modules" %% "scala-collection-compat" % "2.0.0") + ): _*) lazy val `rho-hal` = project .in(file("hal")) .settings(buildSettings :+ halDeps: _*) - .settings(rhoPreviousArtifacts(lastVersion = "0.19.0", "hal")) + .settings(mimaConfiguration) .dependsOn(`rho-core`) lazy val `rho-swagger` = project .in(file("swagger")) .settings(buildSettings :+ swaggerDeps: _*) - .settings(rhoPreviousArtifacts(lastVersion = "0.19.0", "swagger")) + .settings(mimaConfiguration) .dependsOn(`rho-core` % "compile->compile;test->test") lazy val docs = project @@ -46,7 +58,7 @@ lazy val docs = project version.value, apiVersion.value ), - scalacOptions in (ScalaUnidoc, unidoc) += "-Ypartial-unification", + scalacOptions in (ScalaUnidoc, unidoc) ++= versionSpecificEnabledFlags(scalaVersion.value), unidocProjectFilter in (ScalaUnidoc, unidoc) := inProjects( `rho-core`, `rho-hal`, @@ -77,7 +89,7 @@ lazy val `rho-examples` = project ): _*) .dependsOn(`rho-swagger`, `rho-hal`) -lazy val compileFlags = Seq( +lazy val compilerFlags = Seq( "-feature", "-deprecation", "-unchecked", @@ -85,10 +97,14 @@ lazy val compileFlags = Seq( "-language:existentials", "-language:implicitConversions", "-Ywarn-unused", - "-Ypartial-unification", "-Xfatal-warnings" ) +def versionSpecificEnabledFlags(version: String) = (CrossVersion.partialVersion(version) match { + case Some((2, 13)) => Seq.empty[String] + case _ => Seq("-Ypartial-unification") +}) + /* Don't publish setting */ lazy val dontPublish = packagedArtifacts := Map.empty @@ -99,9 +115,9 @@ lazy val license = licenses in ThisBuild := Seq( lazy val buildSettings = publishing ++ Seq( - scalaVersion := "2.12.8", - crossScalaVersions := Seq(scalaVersion.value, "2.11.12"), - scalacOptions ++= compileFlags, + scalaVersion := "2.13.1", + crossScalaVersions := Seq(scalaVersion.value, "2.12.10"), + scalacOptions := compilerFlags ++ versionSpecificEnabledFlags(scalaVersion.value), resolvers += Resolver.sonatypeRepo("snapshots"), fork in run := true, organization in ThisBuild := "org.http4s", diff --git a/core/src/main/scala-2.12-/jdk/CollectionConverters.scala b/core/src/main/scala-2.12-/jdk/CollectionConverters.scala new file mode 100644 index 00000000..abd9071a --- /dev/null +++ b/core/src/main/scala-2.12-/jdk/CollectionConverters.scala @@ -0,0 +1,5 @@ +package scala.jdk + +import scala.collection.convert.{DecorateAsJava, DecorateAsScala} + +object CollectionConverters extends DecorateAsJava with DecorateAsScala \ No newline at end of file diff --git a/core/src/main/scala-2.12-/scala/collection/compat/view/MapViewExtensionMethods.scala b/core/src/main/scala-2.12-/scala/collection/compat/view/MapViewExtensionMethods.scala new file mode 100644 index 00000000..0f4591cb --- /dev/null +++ b/core/src/main/scala-2.12-/scala/collection/compat/view/MapViewExtensionMethods.scala @@ -0,0 +1,13 @@ +package scala.collection.compat.view + +import scala.collection.generic.CanBuildFrom + +class MapViewExtensionMethods[K, V, C <: scala.collection.Map[K, V]](private val self: IterableView[(K, V), C]) extends AnyVal { + def mapValues[W, That](f: V => W)(implicit bf: CanBuildFrom[IterableView[(K, V), C], (K, W), That]): That = + self.map[(K, W), That] { case (k, v) => (k, f(v)) } +} + +trait MapViewExtensions { + implicit def toMapViewExtensionMethods[K, V, C <: scala.collection.Map[K, V]](self: IterableView[(K, V), C]): MapViewExtensionMethods[K, V, C] = + new MapViewExtensionMethods[K, V, C](self) +} \ No newline at end of file diff --git a/core/src/main/scala-2.12-/scala/collection/compat/view/package.scala b/core/src/main/scala-2.12-/scala/collection/compat/view/package.scala new file mode 100644 index 00000000..e2302f70 --- /dev/null +++ b/core/src/main/scala-2.12-/scala/collection/compat/view/package.scala @@ -0,0 +1,7 @@ +package scala.collection.compat + +import scala.collection.{IterableView => SCIterableView} + +package object view extends MapViewExtensions { + type IterableView[A, B] = SCIterableView[A, B] +} diff --git a/core/src/main/scala-2.13+/scala/collection/compat/view/package.scala b/core/src/main/scala-2.13+/scala/collection/compat/view/package.scala new file mode 100644 index 00000000..6bb389ec --- /dev/null +++ b/core/src/main/scala-2.13+/scala/collection/compat/view/package.scala @@ -0,0 +1,7 @@ +package scala.collection.compat + +import scala.collection.{IterableOps, View} + +package object view { + type IterableView[A, _] = IterableOps[A, View, View[A]] +} diff --git a/core/src/main/scala/org/http4s/rho/CompileRoutes.scala b/core/src/main/scala/org/http4s/rho/CompileRoutes.scala index 5824a98a..75d0378f 100644 --- a/core/src/main/scala/org/http4s/rho/CompileRoutes.scala +++ b/core/src/main/scala/org/http4s/rho/CompileRoutes.scala @@ -2,11 +2,8 @@ package org.http4s package rho import scala.collection.immutable.Seq - -import cats.Monad -import cats.data.Kleisli +import cats.effect.Sync import shapeless.HList - import org.http4s.rho.RhoRoute.Tpe import org.http4s.rho.bits.PathTree @@ -43,8 +40,8 @@ object CompileRoutes { * @param routes `Seq` of routes to bundle into a service. * @return An `HttpRoutes` */ - def foldRoutes[F[_]: Monad](routes: Seq[RhoRoute.Tpe[F]]): HttpRoutes[F] = { + def foldRoutes[F[_]: Sync](routes: Seq[RhoRoute.Tpe[F]]): HttpRoutes[F] = { val tree = routes.foldLeft(PathTree[F]()){ (t, r) => t.appendRoute(r) } - Kleisli((req: Request[F]) => tree.getResult(req).toResponse) + HttpRoutes((req: Request[F]) => tree.getResult(req).toResponse) } } diff --git a/core/src/main/scala/org/http4s/rho/RhoDslPathExtractors.scala b/core/src/main/scala/org/http4s/rho/RhoDslPathExtractors.scala index fa7fa21c..08a83ed9 100644 --- a/core/src/main/scala/org/http4s/rho/RhoDslPathExtractors.scala +++ b/core/src/main/scala/org/http4s/rho/RhoDslPathExtractors.scala @@ -2,19 +2,27 @@ package org.http4s.rho import org.http4s.rho.bits.PathAST._ import org.http4s.rho.bits._ +import org.http4s.rho.RhoDslPathExtractors._ import shapeless.{::, HNil} import scala.reflect.runtime.universe.TypeTag trait RhoDslPathExtractors[F[_]] { - private val stringTag = implicitly[TypeTag[String]] - implicit def pathMatch(s: String): TypedPath[F, HNil] = TypedPath(PathMatch(s)) - implicit def pathMatch(s: Symbol): TypedPath[F, String :: HNil] = + /** + * Provides 'pathVar syntax for String path variables (Scala 2.12 only) + */ + implicit def pathCapture(s: Symbol): TypedPath[F, String :: HNil] = TypedPath(PathCapture(s.name, None, StringParser.strParser, stringTag)) + /** + * Provides pv"pathVarName" syntax for String path variables as an alternative for 'pathVar (Symbol) syntax which was removed in Scala 2.13. + */ + implicit def pathCapture(sc: StringContext): PathCaptureStringContext[F] = + new PathCaptureStringContext[F](sc) + /** * Defines a path variable of a URI that should be bound to a route definition */ @@ -34,3 +42,13 @@ trait RhoDslPathExtractors[F[_]] { TypedPath(PathCapture[F](id, Some(description), parser, stringTag)) } + +object RhoDslPathExtractors { + + private val stringTag = implicitly[TypeTag[String]] + + class PathCaptureStringContext[F[_]](val sc: StringContext) extends AnyVal { + def pv(): TypedPath[F, String :: HNil] = + TypedPath[F, String :: HNil](PathCapture(sc.parts.mkString, None, StringParser.strParser, stringTag)) + } +} \ No newline at end of file diff --git a/core/src/main/scala/org/http4s/rho/RhoRoutes.scala b/core/src/main/scala/org/http4s/rho/RhoRoutes.scala index 5eeb9f63..982f8164 100644 --- a/core/src/main/scala/org/http4s/rho/RhoRoutes.scala +++ b/core/src/main/scala/org/http4s/rho/RhoRoutes.scala @@ -2,8 +2,7 @@ package org.http4s package rho import scala.collection.immutable.Seq - -import cats.Monad +import cats.effect.Sync import org.http4s.rho.bits.PathAST.TypedPath import org.log4s.getLogger import shapeless.{HList, HNil} @@ -24,7 +23,7 @@ import shapeless.{HList, HNil} * * @param routes Routes to prepend before elements in the constructor. */ -class RhoRoutes[F[_]: Monad](routes: Seq[RhoRoute[F, _ <: HList]] = Vector.empty) +class RhoRoutes[F[_]: Sync](routes: Seq[RhoRoute[F, _ <: HList]] = Vector.empty) extends bits.MethodAliases with bits.ResponseGeneratorInstances[F] with RoutePrependable[F, RhoRoutes[F]] diff --git a/core/src/main/scala/org/http4s/rho/RoutesBuilder.scala b/core/src/main/scala/org/http4s/rho/RoutesBuilder.scala index 59b879ad..051d23bd 100644 --- a/core/src/main/scala/org/http4s/rho/RoutesBuilder.scala +++ b/core/src/main/scala/org/http4s/rho/RoutesBuilder.scala @@ -2,12 +2,14 @@ package org.http4s.rho import scala.collection.immutable.VectorBuilder import scala.collection.immutable.Seq -import cats.Monad +import cats.effect.Sync import shapeless.HList import org.http4s._ +import scala.collection.compat._ + /** CompileRoutes which accumulates routes and can build a `HttpRoutes` */ -final class RoutesBuilder[F[_]: Monad] private(internalRoutes: VectorBuilder[RhoRoute.Tpe[F]]) extends CompileRoutes[F, RhoRoute.Tpe[F]] { +final class RoutesBuilder[F[_]: Sync] private(internalRoutes: VectorBuilder[RhoRoute.Tpe[F]]) extends CompileRoutes[F, RhoRoute.Tpe[F]] { /** Turn the accumulated routes into an `HttpRoutes` * @@ -25,8 +27,8 @@ final class RoutesBuilder[F[_]: Monad] private(internalRoutes: VectorBuilder[Rho * @param routes Routes to accumulate. * @return `this` instance with its internal state mutated. */ - def append(routes: TraversableOnce[RhoRoute.Tpe[F]]): this.type = { - internalRoutes ++= routes + def append(routes: IterableOnce[RhoRoute.Tpe[F]]): this.type = { + internalRoutes ++= routes.iterator.to(Iterable) this } @@ -46,10 +48,10 @@ final class RoutesBuilder[F[_]: Monad] private(internalRoutes: VectorBuilder[Rho object RoutesBuilder { /** Constructor method for new `RoutesBuilder` instances */ - def apply[F[_]: Monad](): RoutesBuilder[F] = apply(Seq.empty) + def apply[F[_]: Sync](): RoutesBuilder[F] = apply(Seq.empty) /** Constructor method for new `RoutesBuilder` instances with existing routes */ - def apply[F[_]: Monad](routes: Seq[RhoRoute.Tpe[F]]): RoutesBuilder[F] = { + def apply[F[_]: Sync](routes: Seq[RhoRoute.Tpe[F]]): RoutesBuilder[F] = { val builder = new VectorBuilder[RhoRoute.Tpe[F]] builder ++= routes diff --git a/core/src/main/scala/org/http4s/rho/bits/HeaderAppendable.scala b/core/src/main/scala/org/http4s/rho/bits/HeaderAppendable.scala index d0dd6bfb..51df8d99 100644 --- a/core/src/main/scala/org/http4s/rho/bits/HeaderAppendable.scala +++ b/core/src/main/scala/org/http4s/rho/bits/HeaderAppendable.scala @@ -1,8 +1,6 @@ package org.http4s package rho.bits -import scala.language.higherKinds - import shapeless.HList import shapeless.ops.hlist.Prepend diff --git a/core/src/main/scala/org/http4s/rho/bits/PathTree.scala b/core/src/main/scala/org/http4s/rho/bits/PathTree.scala index bf622682..cd118da0 100644 --- a/core/src/main/scala/org/http4s/rho/bits/PathTree.scala +++ b/core/src/main/scala/org/http4s/rho/bits/PathTree.scala @@ -258,12 +258,12 @@ private[rho] trait PathTreeOps[F[_]] extends RuleExecutor[F] { if (!result.isEmpty || end.isEmpty || method == Method.OPTIONS) result else FailureResponse.pure[F] { - val ms = end.keys + val ms = end.keySet val allowedMethods = ms.mkString(", ") val msg = s"$method not allowed. Defined methods: $allowedMethods\n" F.map(MethodNotAllowed.pure(msg))( - _.putHeaders(headers.Allow(ms.head, ms.tail.toList:_*))) + _.putHeaders(headers.Allow(ms))) } } } diff --git a/core/src/main/scala/org/http4s/rho/bits/QueryParser.scala b/core/src/main/scala/org/http4s/rho/bits/QueryParser.scala index a803ee00..e88e608b 100644 --- a/core/src/main/scala/org/http4s/rho/bits/QueryParser.scala +++ b/core/src/main/scala/org/http4s/rho/bits/QueryParser.scala @@ -6,7 +6,7 @@ import org.http4s.rho.bits.QueryParser.Params import scala.annotation.tailrec import scala.collection.immutable.Seq -import scala.collection.generic.CanBuildFrom +import scala.collection.compat._ /** Extract a value from the `Request` `Query` * @@ -41,9 +41,9 @@ trait QueryParsers[F[_]] extends FailureResponseOps[F] { * * The elements must have the same name and each be a valid representation of the requisite type. */ - implicit def multipleParse[A, B[_]](implicit F: Monad[F], p: StringParser[F, A], cbf: CanBuildFrom[Seq[_], A, B[A]]) = new QueryParser[F, B[A]] { + implicit def multipleParse[A, B[_]](implicit F: Monad[F], p: StringParser[F, A], cbf: Factory[A, B[A]]) = new QueryParser[F, B[A]] { override def collect(name: String, params: Params, default: Option[B[A]]): ResultResponse[F, B[A]] = { - val b = cbf() + val b = cbf.newBuilder params.get(name) match { case None => SuccessResponse(default.getOrElse(b.result)) case Some(Seq()) => SuccessResponse(default.getOrElse(b.result)) diff --git a/core/src/main/scala/org/http4s/rho/package.scala b/core/src/main/scala/org/http4s/rho/package.scala index a5143674..23eab9d5 100644 --- a/core/src/main/scala/org/http4s/rho/package.scala +++ b/core/src/main/scala/org/http4s/rho/package.scala @@ -6,8 +6,6 @@ import org.http4s.rho.bits._ import org.http4s.rho.bits.PathAST._ import shapeless.{HList, HNil} -import scala.language.implicitConversions - package object rho extends org.http4s.syntax.AllSyntax { type RhoMiddleware[F[_]] = Seq[RhoRoute[F, _ <: HList]] => Seq[RhoRoute[F, _ <: HList]] diff --git a/core/src/test/scala/ApiExamples.scala b/core/src/test/scala/ApiExamples.scala index d6a2b864..30f06ed8 100644 --- a/core/src/test/scala/ApiExamples.scala +++ b/core/src/test/scala/ApiExamples.scala @@ -39,8 +39,8 @@ class ApiExamples extends Specification { GET / "helloworldnumber" / pathVar[Int] / "foo" |>> { i: Int => Ok(s"Received $i") } - // the symbol 'world just says 'capture a String' with variable name "world" - GET / "helloworldstring" / 'world / "foo" |>> { i: String => + // the pv"world" (pv stands for path variable) says 'capture a String' with variable name "world" + GET / "helloworldstring" / pv"world" / "foo" |>> { i: String => Ok(s"Received $i") } // capture dates @@ -118,7 +118,7 @@ class ApiExamples extends Specification { val path2 = "two" / pathVar[Int] val getLength = captureMap(`Content-Length`)(_.length) - val getTag = captureMap(ETag)(_ => -1l) + val getTag = captureMap(ETag)(_ => -1L) GET / (path1 || path2) +? param[String]("foo") >>> (getLength || getTag) |>> { (i: Int, foo: String, v: Long) => Ok(s"Received $i, $foo, $v") @@ -134,7 +134,7 @@ class ApiExamples extends Specification { GET / "request" |>> { _: Request[IO] => Ok("I don't actually need a request...") } - GET / "request" / 'foo |>> { (_: Request[IO], _: String) => + GET / "request" / pv"foo" |>> { (_: Request[IO], _: String) => Ok("I wanted a request") } } diff --git a/core/src/test/scala/org/http4s/rho/ApiTest.scala b/core/src/test/scala/org/http4s/rho/ApiTest.scala index 7dda8a6d..e2279e0f 100644 --- a/core/src/test/scala/org/http4s/rho/ApiTest.scala +++ b/core/src/test/scala/org/http4s/rho/ApiTest.scala @@ -2,9 +2,8 @@ package org.http4s package rho import scala.collection.immutable.Seq -import cats.Monad import cats.data.OptionT -import cats.effect.IO +import cats.effect.{IO, Sync} import org.http4s.headers.{ETag, `Content-Length`} import org.http4s.rho.bits.MethodAliases._ import org.http4s.rho.bits.RequestAST.AndRule @@ -19,7 +18,7 @@ class ApiTest extends Specification { object ruleExecutor extends RuleExecutor[IO] - def runWith[F[_]: Monad, T <: HList, FU](exec: RouteExecutable[F, T])(f: FU)(implicit hltf: HListToFunc[F, T, FU]): Request[F] => OptionT[F, Response[F]] = { + def runWith[F[_]: Sync, T <: HList, FU](exec: RouteExecutable[F, T])(f: FU)(implicit hltf: HListToFunc[F, T, FU]): Request[F] => OptionT[F, Response[F]] = { val srvc = new RhoRoutes[F] { exec |>> f }.toRoutes() srvc.apply(_: Request[F]) } @@ -131,7 +130,7 @@ class ApiTest extends Specification { val path = GET / "hello" >>> paramFoo val req = Request[IO]( - uri = Uri.fromString("/hello?i=32&f=3.2&s=Asdf").right.getOrElse(sys.error("Failed.")), + uri = Uri.fromString("/hello?i=32&f=3.2&s=Asdf").getOrElse(sys.error("Failed.")), headers = Headers.of(headers.`Content-Length`.unsafeFromLong(10), headers.Date(HttpDate.now)) ) @@ -162,7 +161,7 @@ class ApiTest extends Specification { } "Append headers to a Route" in { - val path = POST / "hello" / 'world +? param[Int]("fav") + val path = POST / "hello" / pv"world" +? param[Int]("fav") val validations = existsAnd(headers.`Content-Length`){ h => h.length != 0 } val route = runWith((path >>> validations >>> capture(ETag)).decoding(EntityDecoder.text[IO])) { @@ -171,7 +170,7 @@ class ApiTest extends Specification { .map(_.putHeaders(tag)) } - val req = Request[IO](POST, uri = Uri.fromString("/hello/neptune?fav=23").right.getOrElse(sys.error("Fail"))) + val req = Request[IO](POST, uri = Uri.fromString("/hello/neptune?fav=23").getOrElse(sys.error("Fail"))) .putHeaders(etag) .withEntity("cool") @@ -181,7 +180,7 @@ class ApiTest extends Specification { } "accept compound or sequential header rules" in { - val path = POST / "hello" / 'world + val path = POST / "hello" / pv"world" val lplus1 = captureMap(headers.`Content-Length`)(_.length + 1) val route1 = runWith((path >>> lplus1 >>> capture(ETag)).decoding(EntityDecoder.text)) { @@ -194,7 +193,7 @@ class ApiTest extends Specification { Ok("") } - val req = Request[IO](POST, uri = Uri.fromString("/hello/neptune?fav=23").right.getOrElse(sys.error("Fail"))) + val req = Request[IO](POST, uri = Uri.fromString("/hello/neptune?fav=23").getOrElse(sys.error("Fail"))) .putHeaders(ETag(ETag.EntityTag("foo"))) .withEntity("cool") @@ -203,21 +202,21 @@ class ApiTest extends Specification { } "Run || routes" in { - val p1 = "one" / 'two - val p2 = "three" / 'four + val p1 = "one" / pv"two" + val p2 = "three" / pv"four" val f = runWith(GET / (p1 || p2)) { (s: String) => Ok("").map(_.putHeaders(ETag(ETag.EntityTag(s)))) } - val req1 = Request[IO](uri = Uri.fromString("/one/two").right.getOrElse(sys.error("Failed."))) + val req1 = Request[IO](uri = Uri.fromString("/one/two").getOrElse(sys.error("Failed."))) checkETag(f(req1), "two") - val req2 = Request[IO](uri = Uri.fromString("/three/four").right.getOrElse(sys.error("Failed."))) + val req2 = Request[IO](uri = Uri.fromString("/three/four").getOrElse(sys.error("Failed."))) checkETag(f(req2), "four") } "Execute a complicated route" in { - val path = POST / "hello" / 'world +? param[Int]("fav") + val path = POST / "hello" / pv"world" +? param[Int]("fav") val validations = existsAnd(headers.`Content-Length`){ h => h.length != 0 } && capture(ETag) @@ -228,7 +227,7 @@ class ApiTest extends Specification { .map(_.putHeaders(ETag(ETag.EntityTag("foo")))) } - val req = Request[IO](POST, uri = Uri.fromString("/hello/neptune?fav=23").right.getOrElse(sys.error("Fail"))) + val req = Request[IO](POST, uri = Uri.fromString("/hello/neptune?fav=23").getOrElse(sys.error("Fail"))) .putHeaders( ETag(ETag.EntityTag("foo"))) .withEntity("cool") @@ -237,7 +236,7 @@ class ApiTest extends Specification { "Deal with 'no entity' responses" in { val route = runWith(GET / "foo") { () => SwitchingProtocols.apply } - val req = Request[IO](GET, uri = Uri.fromString("/foo").right.getOrElse(sys.error("Fail"))) + val req = Request[IO](GET, uri = Uri.fromString("/foo").getOrElse(sys.error("Fail"))) val result = route(req).value.unsafeRunSync().getOrElse(Response.notFound) result.headers.size must_== 0 @@ -262,7 +261,7 @@ class ApiTest extends Specification { "traverse a captureless path" in { val stuff = GET / "hello" - val req = Request[IO](uri = Uri.fromString("/hello").right.getOrElse(sys.error("Failed."))) + val req = Request[IO](uri = Uri.fromString("/hello").getOrElse(sys.error("Failed."))) val f = runWith(stuff) { () => Ok("Cool.").map(_.putHeaders(ETag(ETag.EntityTag("foo")))) } checkETag(f(req), "foo") @@ -278,8 +277,8 @@ class ApiTest extends Specification { } "capture a variable" in { - val stuff = GET / 'hello - val req = Request[IO](uri = Uri.fromString("/hello").right.getOrElse(sys.error("Failed."))) + val stuff = GET / pv"hello" + val req = Request[IO](uri = Uri.fromString("/hello").getOrElse(sys.error("Failed."))) val f = runWith(stuff) { str: String => Ok("Cool.").map(_.putHeaders(ETag(ETag.EntityTag(str)))) } checkETag(f(req), "hello") @@ -287,7 +286,7 @@ class ApiTest extends Specification { "work directly" in { val stuff = GET / "hello" - val req = Request[IO](uri = Uri.fromString("/hello").right.getOrElse(sys.error("Failed."))) + val req = Request[IO](uri = Uri.fromString("/hello").getOrElse(sys.error("Failed."))) val f = runWith(stuff) { () => Ok("Cool.").map(_.putHeaders(ETag(ETag.EntityTag("foo")))) @@ -298,7 +297,7 @@ class ApiTest extends Specification { "capture end with nothing" in { val stuff = GET / "hello" / * - val req = Request[IO](uri = Uri.fromString("/hello").right.getOrElse(sys.error("Failed."))) + val req = Request[IO](uri = Uri.fromString("/hello").getOrElse(sys.error("Failed."))) val f = runWith(stuff) { path: List[String] => Ok("Cool.").map(_.putHeaders(ETag(ETag.EntityTag(if (path.isEmpty) "go" else "nogo")))) } @@ -308,7 +307,7 @@ class ApiTest extends Specification { "capture remaining" in { val stuff = GET / "hello" / * - val req = Request[IO](uri = Uri.fromString("/hello/world/foo").right.getOrElse(sys.error("Failed."))) + val req = Request[IO](uri = Uri.fromString("/hello/world/foo").getOrElse(sys.error("Failed."))) val f = runWith(stuff) { path: List[String] => Ok("Cool.").map(_.putHeaders(ETag(ETag.EntityTag(path.mkString)))) } checkETag(f(req), "worldfoo") @@ -318,7 +317,7 @@ class ApiTest extends Specification { "Query validators" should { "get a query string" in { val path = GET / "hello" +? param[Int]("jimbo") - val req = Request[IO](uri = Uri.fromString("/hello?jimbo=32").right.getOrElse(sys.error("Failed."))) + val req = Request[IO](uri = Uri.fromString("/hello?jimbo=32").getOrElse(sys.error("Failed."))) val route = runWith(path) { i: Int => Ok("stuff").map(_.putHeaders(ETag(ETag.EntityTag((i + 1).toString)))) } @@ -348,7 +347,7 @@ class ApiTest extends Specification { val paramFoo = param[Int]("i") & param[Double]("f") & param[String]("s") map Foo.apply _ val path = GET / "hello" +? paramFoo - val req = Request[IO](uri = Uri.fromString("/hello?i=32&f=3.2&s=Asdf").right.getOrElse(sys.error("Failed."))) + val req = Request[IO](uri = Uri.fromString("/hello?i=32&f=3.2&s=Asdf").getOrElse(sys.error("Failed."))) val route = runWith(path) { (f: Foo) => Ok(s"stuff $f") } @@ -365,10 +364,10 @@ class ApiTest extends Specification { val path = POST / "hello" >>> reqHeader - val req1 = Request[IO](POST, uri = Uri.fromString("/hello").right.getOrElse(sys.error("Fail"))) + val req1 = Request[IO](POST, uri = Uri.fromString("/hello").getOrElse(sys.error("Fail"))) .withEntity("foo") - val req2 = Request[IO](POST, uri = Uri.fromString("/hello").right.getOrElse(sys.error("Fail"))) + val req2 = Request[IO](POST, uri = Uri.fromString("/hello").getOrElse(sys.error("Fail"))) .withEntity("0123456789") // length 10 val route = runWith(path.decoding(EntityDecoder.text)) { str: String => @@ -382,7 +381,7 @@ class ApiTest extends Specification { "Allow the infix operator syntax" in { val path = POST / "hello" - val req = Request[IO](POST, uri = Uri.fromString("/hello").right.getOrElse(sys.error("Fail"))) + val req = Request[IO](POST, uri = Uri.fromString("/hello").getOrElse(sys.error("Fail"))) .withEntity("foo") val route = runWith(path ^ EntityDecoder.text) { str: String => diff --git a/core/src/test/scala/org/http4s/rho/AuthedContextSpec.scala b/core/src/test/scala/org/http4s/rho/AuthedContextSpec.scala index 99047272..c22957ab 100644 --- a/core/src/test/scala/org/http4s/rho/AuthedContextSpec.scala +++ b/core/src/test/scala/org/http4s/rho/AuthedContextSpec.scala @@ -13,7 +13,7 @@ case class User(name: String, id: UUID) object Auth { type O[A] = OptionT[IO, A] - val authUser: Kleisli[O, Request[IO], User] = Kleisli({ _ => + val authUser = Kleisli[O, Request[IO], User]({ _ => OptionT.some[IO](User("Test User", UUID.randomUUID())) }) @@ -34,9 +34,9 @@ object MyRoutes extends RhoRoutes[IO] { } } - GET / "public" / 'place |>> { path: String => Ok(s"not authenticated at $path") } + GET / "public" / pv"place" |>> { path: String => Ok(s"not authenticated at $path") } - GET / "private" / 'place |>> { (req: Request[IO], path: String) => + GET / "private" / pv"place" |>> { (req: Request[IO], path: String) => getAuth(req) match { case Some(user) => Ok(s"${user.name} at $path") case None => Forbidden(s"not authenticated at $path") diff --git a/core/src/test/scala/org/http4s/rho/CodecRouterSpec.scala b/core/src/test/scala/org/http4s/rho/CodecRouterSpec.scala index c4adad0f..110aee34 100644 --- a/core/src/test/scala/org/http4s/rho/CodecRouterSpec.scala +++ b/core/src/test/scala/org/http4s/rho/CodecRouterSpec.scala @@ -5,6 +5,8 @@ import cats.effect.IO import fs2.Stream import org.specs2.mutable.Specification +import scala.collection.compat.immutable.ArraySeq + class CodecRouterSpec extends Specification { def bodyAndStatus(resp: Response[IO]): (String, Status) = { @@ -21,7 +23,7 @@ class CodecRouterSpec extends Specification { "Decode a valid body" in { - val b = Stream.emits("hello".getBytes) + val b = Stream.emits(ArraySeq.unsafeWrapArray("hello".getBytes)) val h = Headers.of(headers.`Content-Type`(MediaType.text.plain)) val req = Request[IO](Method.POST, Uri(path = "/foo"), headers = h, body = b) val result = routes(req).value.unsafeRunSync().getOrElse(Response.notFound) @@ -32,7 +34,7 @@ class CodecRouterSpec extends Specification { } "Fail on invalid body" in { - val b = Stream.emits("hello =".getBytes) + val b = Stream.emits(ArraySeq.unsafeWrapArray("hello =".getBytes)) val h = Headers.of(headers.`Content-Type`(MediaType.application.`x-www-form-urlencoded`)) val req = Request[IO](Method.POST, Uri(path = "/form"), headers = h, body = b) diff --git a/core/src/test/scala/org/http4s/rho/ParamDefaultValueSpec.scala b/core/src/test/scala/org/http4s/rho/ParamDefaultValueSpec.scala index cf2cb13c..b39a4220 100644 --- a/core/src/test/scala/org/http4s/rho/ParamDefaultValueSpec.scala +++ b/core/src/test/scala/org/http4s/rho/ParamDefaultValueSpec.scala @@ -12,7 +12,7 @@ class ParamDefaultValueSpec extends Specification { new String(routes(r).value.unsafeRunSync().getOrElse(Response.notFound).body.compile.toVector.unsafeRunSync().foldLeft(Array[Byte]())(_ :+ _)) def requestGet(s: String, h: Header*): Request[IO] = - Request(bits.MethodAliases.GET, Uri.fromString(s).right.getOrElse(sys.error("Failed.")), headers = Headers.of(h: _*)) + Request(bits.MethodAliases.GET, Uri.fromString(s).getOrElse(sys.error("Failed.")), headers = Headers.of(h: _*)) "GET /test1" should { val routes = new RhoRoutes[IO] { @@ -299,13 +299,13 @@ class ParamDefaultValueSpec extends Specification { body(routes, requestGet("/test13")) must be equalTo default } "fail to map parameter with empty value" in { - body(routes, requestGet("/test13?param1=")) must be equalTo "Invalid query parameter: \"param1\" = \"List()\"" + body(routes, requestGet("/test13?param1=")) must be matching """Invalid query parameter: "param1" = "(List|Vector)\(\)"""".r } "fail to map parameter with one invalid value" in { - body(routes, requestGet("/test13?param1=z")) must be equalTo "Invalid query parameter: \"param1\" = \"List(z)\"" + body(routes, requestGet("/test13?param1=z")) must be matching """Invalid query parameter: "param1" = "(List|Vector)\(z\)"""".r } "map parameter with many values and one invalid" in { - body(routes, requestGet("/test13?param1=z¶m1=aa¶m1=bb")) must be equalTo "Invalid query parameter: \"param1\" = \"List(z, aa, bb)\"" + body(routes, requestGet("/test13?param1=z¶m1=aa¶m1=bb")) must be matching """Invalid query parameter: "param1" = "(List|Vector)\(z, aa, bb\)"""".r } "map parameter with many valid values" in { body(routes, requestGet("/test13?param1=c¶m1=d")) must be equalTo "test13:c,d" @@ -329,7 +329,7 @@ class ParamDefaultValueSpec extends Specification { body(routes, requestGet("/test14?param1=")) must be equalTo "Invalid number format: ''" } "fail to map parameter with one invalid numeric value" in { - body(routes, requestGet("/test14?param1=8¶m1=5¶m1=3")) must be equalTo "Invalid query parameter: \"param1\" = \"List(8, 5, 3)\"" + body(routes, requestGet("/test14?param1=8¶m1=5¶m1=3")) must be matching """Invalid query parameter: "param1" = "(List|Vector)\(8, 5, 3\)"""".r } "fail to map parameter with one non-numeric value" in { body(routes, requestGet("/test14?param1=test")) must be equalTo "Invalid number format: 'test'" diff --git a/core/src/test/scala/org/http4s/rho/RhoRoutesSpec.scala b/core/src/test/scala/org/http4s/rho/RhoRoutesSpec.scala index dc2a3210..5cfd1cab 100644 --- a/core/src/test/scala/org/http4s/rho/RhoRoutesSpec.scala +++ b/core/src/test/scala/org/http4s/rho/RhoRoutesSpec.scala @@ -1,8 +1,8 @@ package org.http4s package rho +import scala.collection.compat.immutable.ArraySeq import scala.collection.immutable.Seq - import java.util.concurrent.atomic.AtomicInteger import cats.effect.IO @@ -14,7 +14,7 @@ import org.specs2.mutable.Specification class RhoRoutesSpec extends Specification with RequestRunner { def construct(method: Method, s: String, h: Header*): Request[IO] = - Request(method, Uri.fromString(s).right.getOrElse(sys.error("Failed.")), headers = Headers.of(h: _*)) + Request(method, Uri.fromString(s).getOrElse(sys.error("Failed.")), headers = Headers.of(h: _*)) def Get(s: String, h: Header*): Request[IO] = construct(Method.GET, s, h:_*) def Put(s: String, h: Header*): Request[IO] = construct(Method.PUT, s, h:_*) @@ -26,7 +26,7 @@ class RhoRoutesSpec extends Specification with RequestRunner { GET / "hello" |>> Ok("route1") - GET / 'hello |>> { _: String => Ok("route2") } + GET / pv"hello" |>> { _: String => Ok("route2") } GET / "hello" / "world" |>> Ok("route3") @@ -313,8 +313,8 @@ class RhoRoutesSpec extends Specification with RequestRunner { } } - val body = Stream.emits("foo".getBytes()) - val uri = Uri.fromString("/foo/1?param=myparam").right.getOrElse(sys.error("Failed.")) + val body = Stream.emits(ArraySeq.unsafeWrapArray("foo".getBytes())) + val uri = Uri.fromString("/foo/1?param=myparam").getOrElse(sys.error("Failed.")) val req = Request[IO](method = Method.POST, uri = uri, body = body) .putHeaders(`Content-Type`(MediaType.text.plain), `Content-Length`.unsafeFromLong("foo".length)) @@ -369,7 +369,7 @@ class RhoRoutesSpec extends Specification with RequestRunner { GET / "bar" |>> "bar" } - val routes2: HttpRoutes[IO] = "foo" /: routes1 toRoutes() + val routes2: HttpRoutes[IO] = ("foo" /: routes1).toRoutes() val req1 = Request[IO](uri = Uri(path ="/foo/bar")) getBody(routes2(req1).value.unsafeRunSync().getOrElse(Response.notFound).body) === "bar" diff --git a/core/src/test/scala/org/http4s/rho/RouteAsUriTemplateSpec.scala b/core/src/test/scala/org/http4s/rho/RouteAsUriTemplateSpec.scala index 33ccd451..1d1f134b 100644 --- a/core/src/test/scala/org/http4s/rho/RouteAsUriTemplateSpec.scala +++ b/core/src/test/scala/org/http4s/rho/RouteAsUriTemplateSpec.scala @@ -21,7 +21,7 @@ class RouteAsUriTemplateSpec extends Specification { route.asUriTemplate(request).get must equalTo(UriTemplate(path = List(PathElm("hello"), PathElm("world")))) } "convert to /hello{/world}" in { - val route: PathBuilder[IO, _ <: HList] = GET / "hello" / 'world + val route: PathBuilder[IO, _ <: HList] = GET / "hello" / pv"world" route.asUriTemplate(request).get must equalTo(UriTemplate(path = List(PathElm("hello"), PathExp("world")))) } "convert to /hello/world/next/time" in { @@ -102,7 +102,7 @@ class RouteAsUriTemplateSpec extends Specification { route.asUriTemplate(request).get must equalTo(UriTemplate(path = List(PathElm("hello"), PathElm("world")))) } "convert to /hello{/world}" in { - val route = "hello" / 'world + val route = "hello" / pv"world" route.asUriTemplate(request).get must equalTo(UriTemplate(path = List(PathElm("hello"), PathExp("world")))) } "convert to /hello/world/next/time" in { diff --git a/core/src/test/scala/org/http4s/rho/bits/HListToFuncSpec.scala b/core/src/test/scala/org/http4s/rho/bits/HListToFuncSpec.scala index d4fccdae..b246d089 100644 --- a/core/src/test/scala/org/http4s/rho/bits/HListToFuncSpec.scala +++ b/core/src/test/scala/org/http4s/rho/bits/HListToFuncSpec.scala @@ -13,7 +13,7 @@ class HListToFuncSpec extends Specification { def checkOk(r: Request[IO]): String = getBody(service(r).value.unsafeRunSync().getOrElse(Response.notFound).body) def Get(s: String, h: Header*): Request[IO] = - Request(bits.MethodAliases.GET, Uri.fromString(s).right.getOrElse(sys.error("Failed.")), headers = Headers.of(h:_*)) + Request(bits.MethodAliases.GET, Uri.fromString(s).getOrElse(sys.error("Failed.")), headers = Headers.of(h:_*)) val service = new RhoRoutes[IO] { GET / "route1" |>> { () => Ok("foo") } diff --git a/core/src/test/scala/org/http4s/rho/bits/PathTreeSpec.scala b/core/src/test/scala/org/http4s/rho/bits/PathTreeSpec.scala index ec8f1b6c..c0d498eb 100644 --- a/core/src/test/scala/org/http4s/rho/bits/PathTreeSpec.scala +++ b/core/src/test/scala/org/http4s/rho/bits/PathTreeSpec.scala @@ -2,13 +2,14 @@ package org.http4s package rho package bits + import java.nio.charset.StandardCharsets import cats.effect.IO -import org.http4s.server.Router -import org.http4s.server.middleware.TranslateUri import org.specs2.mutable.Specification import org.http4s.Uri.uri +import org.http4s.server.middleware.TranslateUri +import org.http4s.server.Router class PathTreeSpec extends Specification { import PathTree._ @@ -37,12 +38,12 @@ class PathTreeSpec extends Specification { } "Honor UriTranslations" in { - val svc = TranslateUri("/bar")(Router.define[IO](("/", new RhoRoutes[IO] { + val svc = TranslateUri("/bar")(Router("/" -> new RhoRoutes[IO] { GET / "foo" |>> "foo" - }.toRoutes()))(HttpRoutes.empty[IO])) + }.toRoutes())).orNotFound val req = Request[IO](Method.GET, uri = Uri(path = "/bar/foo")) - val resp = svc(req).value.unsafeRunSync().getOrElse(Response.notFound) + val resp = svc(req).unsafeRunSync() resp.status must_== Status.Ok val b = new String(resp.body.compile.toVector.unsafeRunSync().foldLeft(Array[Byte]())(_ :+ _), StandardCharsets.UTF_8) diff --git a/core/src/test/scala/org/http4s/rho/bits/ResponseGeneratorSpec.scala b/core/src/test/scala/org/http4s/rho/bits/ResponseGeneratorSpec.scala index 192ba00c..ed594ac5 100644 --- a/core/src/test/scala/org/http4s/rho/bits/ResponseGeneratorSpec.scala +++ b/core/src/test/scala/org/http4s/rho/bits/ResponseGeneratorSpec.scala @@ -30,7 +30,7 @@ class ResponseGeneratorSpec extends Specification { } "Build a redirect response" in { - val location = Uri.fromString("/foo").right.getOrElse(sys.error("Fail.")) + val location = Uri.fromString("/foo").getOrElse(sys.error("Fail.")) val result = MovedPermanently(location).unsafeRunSync() val resp = result.resp @@ -43,7 +43,7 @@ class ResponseGeneratorSpec extends Specification { "Build a redirect response with a body" in { val testBody = "Moved!!!" - val location = Uri.fromString("/foo").right.getOrElse(sys.error("Fail.")) + val location = Uri.fromString("/foo").getOrElse(sys.error("Fail.")) val result = MovedPermanently(location, testBody).unsafeRunSync() val resp = result.resp diff --git a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/BusinessLayer.scala b/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/BusinessLayer.scala index 6de8b10d..d79ba17f 100644 --- a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/BusinessLayer.scala +++ b/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/BusinessLayer.scala @@ -11,7 +11,7 @@ import net.sf.uadetector.internal.data.domain.{ BrowserType => UBrowserType, OperatingSystem => UOperatingSystem} -import scala.collection.JavaConverters._ +import scala.jdk.CollectionConverters._ import scala.collection.immutable.Seq // -- diff --git a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/Main.scala b/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/Main.scala index fd4e0089..1fdb60c8 100644 --- a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/Main.scala +++ b/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/Main.scala @@ -1,9 +1,9 @@ package com.http4s.rho.hal.plus.swagger.demo -import cats.implicits._ -import cats.effect.{ExitCode, IO, IOApp} +import cats.effect.{Blocker, ExitCode, IO, IOApp} import net.sf.uadetector.service.UADetectorServiceFactory.ResourceModuleXmlDataStore -import org.http4s.syntax.all._ +import cats.implicits._ +import org.http4s.implicits._ import org.http4s.server.blaze._ import org.log4s.getLogger @@ -16,15 +16,16 @@ object Main extends IOApp { logger.info(s"Starting Hal example on '$port'") - def run(args: List[String]): IO[ExitCode] = { - val businessLayer = new UADetectorDatabase(new ResourceModuleXmlDataStore()) + def run(args: List[String]): IO[ExitCode] = + Blocker[IO].use{ blocker => + val businessLayer = new UADetectorDatabase(new ResourceModuleXmlDataStore()) - val routes = - new Routes(businessLayer) + val routes = + new Routes(businessLayer, blocker) - BlazeServerBuilder[IO] - .withHttpApp((routes.staticContent <+> routes.dynamicContent).orNotFound) - .bindLocal(port) - .serve.compile.drain.as(ExitCode.Success) - } + BlazeServerBuilder[IO] + .withHttpApp((routes.staticContent <+> routes.dynamicContent).orNotFound) + .bindLocal(port) + .serve.compile.drain.as(ExitCode.Success) + } } diff --git a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/RestRoutes.scala b/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/RestRoutes.scala index c3d37492..20008c3d 100644 --- a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/RestRoutes.scala +++ b/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/RestRoutes.scala @@ -1,6 +1,6 @@ package com.http4s.rho.hal.plus.swagger.demo -import cats.Monad +import cats.effect.Sync import org.http4s.rho.RhoRoutes import org.http4s.rho.hal.{ResourceObjectBuilder => ResObjBuilder, _} import org.http4s.{Request, Uri} @@ -8,7 +8,7 @@ import org.http4s.{Request, Uri} import scala.collection.immutable.Seq import scala.collection.mutable.ListBuffer -class RestRoutes[F[+_]: Monad](val businessLayer: BusinessLayer) extends RhoRoutes[F] { +class RestRoutes[F[+_]: Sync](val businessLayer: BusinessLayer) extends RhoRoutes[F] { // # Query Parameters diff --git a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/Routes.scala b/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/Routes.scala index 493db829..ec91c6d7 100644 --- a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/Routes.scala +++ b/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/Routes.scala @@ -1,11 +1,11 @@ package com.http4s.rho.hal.plus.swagger.demo -import cats.effect.{ContextShift, IO, Timer} +import cats.effect.{Blocker, ContextShift, IO, Timer} import org.http4s.HttpRoutes import org.http4s.rho.RhoMiddleware import org.http4s.rho.swagger.syntax.io._ -class Routes(businessLayer: BusinessLayer)(implicit T: Timer[IO], cs: ContextShift[IO]) { +class Routes(businessLayer: BusinessLayer, blocker: Blocker)(implicit T: Timer[IO], cs: ContextShift[IO]) { val middleware: RhoMiddleware[IO] = createRhoMiddleware() @@ -18,6 +18,6 @@ class Routes(businessLayer: BusinessLayer)(implicit T: Timer[IO], cs: ContextShi * but its nice to keep it self contained */ val staticContent: HttpRoutes[IO] = - new StaticContentService[IO](org.http4s.dsl.io) {}.routes + new StaticContentService[IO](org.http4s.dsl.io, blocker).routes } diff --git a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/StaticContentService.scala b/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/StaticContentService.scala index f7ea56cd..e15e8287 100644 --- a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/StaticContentService.scala +++ b/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/StaticContentService.scala @@ -1,22 +1,16 @@ package com.http4s.rho.hal.plus.swagger.demo import cats.data.OptionT -import cats.effect.{ContextShift, Sync, Timer} +import cats.effect.{Blocker, ContextShift, Sync, Timer} import org.http4s.dsl.Http4sDsl import org.http4s.{HttpRoutes, Request, Response, StaticFile} -import scala.concurrent.ExecutionContext.global - -abstract class StaticContentService[F[_]: Sync : Timer : ContextShift](dsl: Http4sDsl[F]) { +class StaticContentService[F[_]: Sync : Timer : ContextShift](dsl: Http4sDsl[F], blocker: Blocker) { import dsl._ private val halUiDir = "/hal-browser" private val swaggerUiDir = "/swagger-ui" - def fetchResource(path: String, req: Request[F]): OptionT[F, Response[F]] = { - StaticFile.fromResource(path, global, Some(req)) - } - /** * Routes for getting static resources. These might be served more efficiently by apache2 or nginx, * but its nice to keep it self contained. @@ -35,4 +29,8 @@ abstract class StaticContentService[F[_]: Sync : Timer : ContextShift](dsl: Http case req @ GET -> Root / "swagger-ui" => fetchResource(swaggerUiDir + "/index.html", req) case req @ GET -> Root / "swagger-ui.js" => fetchResource(swaggerUiDir + "/swagger-ui.min.js", req) } + + private def fetchResource(path: String, req: Request[F]): OptionT[F, Response[F]] = { + StaticFile.fromResource(path, blocker, Some(req)) + } } diff --git a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/package.scala b/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/package.scala index 766fe5bf..d3acefcc 100644 --- a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/package.scala +++ b/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/package.scala @@ -1,6 +1,5 @@ package com.http4s.rho.hal.plus.swagger -import scala.language.implicitConversions import scala.collection.immutable.Seq import scala.util.Failure import scala.util.Success diff --git a/examples/src/main/scala/com/http4s/rho/swagger/demo/JsonEncoder.scala b/examples/src/main/scala/com/http4s/rho/swagger/demo/JsonEncoder.scala index 704c323a..1b00ac20 100644 --- a/examples/src/main/scala/com/http4s/rho/swagger/demo/JsonEncoder.scala +++ b/examples/src/main/scala/com/http4s/rho/swagger/demo/JsonEncoder.scala @@ -5,6 +5,8 @@ import java.nio.charset.StandardCharsets import org.http4s.headers.`Content-Type` import org.http4s.{Entity, EntityEncoder, MediaType} +import scala.collection.compat.immutable.ArraySeq + object JsonEncoder { import fs2.Stream import org.json4s._ @@ -19,6 +21,6 @@ object JsonEncoder { implicit def autoSerializableEntityEncoder[F[_], A <: AutoSerializable]: EntityEncoder[F, A] = EntityEncoder.encodeBy(`Content-Type`(MediaType.application.json))(a => { val bytes = write(a).getBytes(StandardCharsets.UTF_8) - Entity(Stream.emits(bytes), Some(bytes.length)) + Entity(Stream.emits(ArraySeq.unsafeWrapArray(bytes)), Some(bytes.length)) }) } diff --git a/examples/src/main/scala/com/http4s/rho/swagger/demo/Main.scala b/examples/src/main/scala/com/http4s/rho/swagger/demo/Main.scala index ef006c29..2fad0d27 100644 --- a/examples/src/main/scala/com/http4s/rho/swagger/demo/Main.scala +++ b/examples/src/main/scala/com/http4s/rho/swagger/demo/Main.scala @@ -1,32 +1,35 @@ package com.http4s.rho.swagger.demo -import cats.effect.{ExitCode, IO, IOApp} +import cats.effect.{Blocker, ExitCode, IO, IOApp} import cats.implicits._ import org.http4s.HttpRoutes -import org.http4s.syntax.all._ +import org.http4s.implicits._ import org.http4s.rho.swagger.syntax.{io => ioSwagger} import org.http4s.server.blaze.BlazeServerBuilder import org.log4s.getLogger object Main extends IOApp { private val logger = getLogger + import ioSwagger._ - val port: Int = Option(System.getenv("HTTP_PORT")) + private val port: Int = Option(System.getenv("HTTP_PORT")) .map(_.toInt) .getOrElse(8080) logger.info(s"Starting Swagger example on '$port'") - def run(args: List[String]): IO[ExitCode] = { - val middleware = createRhoMiddleware() + def run(args: List[String]): IO[ExitCode] = + Blocker[IO].use { blocker => + + val middleware = createRhoMiddleware() - val myService: HttpRoutes[IO] = - new MyRoutes[IO](ioSwagger) {}.toRoutes(middleware) + val myService: HttpRoutes[IO] = + new MyRoutes[IO](ioSwagger).toRoutes(middleware) - BlazeServerBuilder[IO] - .withHttpApp((StaticContentService.routes <+> myService).orNotFound) - .bindLocal(port) - .serve.compile.drain.as(ExitCode.Success) - } + BlazeServerBuilder[IO] + .withHttpApp((StaticContentService.routes(blocker) <+> myService).orNotFound) + .bindLocal(port) + .serve.compile.drain.as(ExitCode.Success) + } } diff --git a/examples/src/main/scala/com/http4s/rho/swagger/demo/MyRoutes.scala b/examples/src/main/scala/com/http4s/rho/swagger/demo/MyRoutes.scala index bc6abcfb..f7c82b83 100644 --- a/examples/src/main/scala/com/http4s/rho/swagger/demo/MyRoutes.scala +++ b/examples/src/main/scala/com/http4s/rho/swagger/demo/MyRoutes.scala @@ -17,7 +17,7 @@ import shapeless.HNil import scala.reflect.ClassTag import scala.collection.immutable.Seq -abstract class MyRoutes[F[+_] : Effect](swaggerSyntax: SwaggerSyntax[F]) +class MyRoutes[F[+_] : Effect](swaggerSyntax: SwaggerSyntax[F]) extends RhoRoutes[F] { import swaggerSyntax._ @@ -47,7 +47,7 @@ abstract class MyRoutes[F[+_] : Effect](swaggerSyntax: SwaggerSyntax[F]) HEAD / "hello" |>> { Ok("Hello head!") } "Generates some JSON data from a route param, and a query Int" ** - GET / "result" / 'foo +? param[Int]("id") |>> { (name: String, id: Int) => Ok(JsonResult(name, id)) } + GET / "result" / pv"foo" +? param[Int]("id") |>> { (name: String, id: Int) => Ok(JsonResult(name, id)) } "Two different response codes can result from this route based on the number given" ** GET / "differentstatus" / pathVar[Int] |>> { i: Int => @@ -97,9 +97,9 @@ abstract class MyRoutes[F[+_] : Effect](swaggerSyntax: SwaggerSyntax[F]) Ok("Deleted cookies!").map(_.withHeaders(hs)) } - "This route allows your to post stuff" ** - List("post", "stuff") @@ - POST / "post" ^ EntityDecoder.text[F] |>> { body: String => + "This route allows your to post stuff" ** + List("post", "stuff") @@ + POST / "post" ^ EntityDecoder.text[F] |>> { body: String => "You posted: " + body } diff --git a/examples/src/main/scala/com/http4s/rho/swagger/demo/StaticContentService.scala b/examples/src/main/scala/com/http4s/rho/swagger/demo/StaticContentService.scala index a4b9c359..052b3ff0 100644 --- a/examples/src/main/scala/com/http4s/rho/swagger/demo/StaticContentService.scala +++ b/examples/src/main/scala/com/http4s/rho/swagger/demo/StaticContentService.scala @@ -1,28 +1,27 @@ package com.http4s.rho.swagger.demo -import cats.effect.{ContextShift, IO} -import org.http4s.{HttpRoutes, Request, Response, StaticFile} +import cats.effect.{Blocker, ContextShift, IO} import org.http4s.dsl.io._ - -import scala.concurrent.ExecutionContext.global +import org.http4s.rho.RhoRoutes +import org.http4s.{HttpRoutes, Request, Response, StaticFile} object StaticContentService { private val swaggerUiDir = "/swagger-ui" - def fetchResource(path: String, req: Request[IO])(implicit cs: ContextShift[IO]): IO[Response[IO]] = { - StaticFile.fromResource(path, global, Some(req)).getOrElseF(NotFound()) - } - /** - * Routes for getting static resources. These might be served more efficiently by apache2 or nginx, - * but its nice to keep it self contained - */ - def routes(implicit cs: ContextShift[IO]): HttpRoutes[IO] = HttpRoutes.of { + * Routes for getting static resources. These might be served more efficiently by apache2 or nginx, + * but its nice to keep it self contained. + */ + def routes(blocker: Blocker)(implicit cs: ContextShift[IO]): HttpRoutes[IO] = new RhoRoutes[IO] { // Swagger User Interface - case req @ GET -> Root / "css" / _ => fetchResource(swaggerUiDir + req.pathInfo, req) - case req @ GET -> Root / "images" / _ => fetchResource(swaggerUiDir + req.pathInfo, req) - case req @ GET -> Root / "lib" / _ => fetchResource(swaggerUiDir + req.pathInfo, req) - case req @ GET -> Root / "swagger-ui" => fetchResource(swaggerUiDir + "/index.html", req) - case req @ GET -> Root / "swagger-ui.js" => fetchResource(swaggerUiDir + "/swagger-ui.min.js", req) - } + GET / "css" / * |>> { (req: Request[IO], _: List[String]) => fetchResource(swaggerUiDir + req.pathInfo, req, blocker) } + GET / "images" / * |>> { (req: Request[IO], _: List[String]) => fetchResource(swaggerUiDir + req.pathInfo, req, blocker) } + GET / "lib" / * |>> { (req: Request[IO], _: List[String]) => fetchResource(swaggerUiDir + req.pathInfo, req, blocker) } + GET / "swagger-ui" |>> { req: Request[IO] => fetchResource(swaggerUiDir + "/index.html", req, blocker) } + GET / "swagger-ui.js" |>> { req: Request[IO] => fetchResource(swaggerUiDir + "/swagger-ui.min.js", req, blocker) } + }.toRoutes() + + private def fetchResource(path: String, req: Request[IO], blocker: Blocker)(implicit cs: ContextShift[IO]): IO[Response[IO]] = + StaticFile.fromResource(path, blocker, Some(req)).getOrElseF(NotFound()) + } diff --git a/project/Dependencies.scala b/project/Dependencies.scala index 32e36dee..bbc36660 100644 --- a/project/Dependencies.scala +++ b/project/Dependencies.scala @@ -2,7 +2,7 @@ import sbt._ import Keys._ object Dependencies { - lazy val http4sVersion = "0.20.11" + lazy val http4sVersion = "0.21.0-M5" lazy val specs2Version = "4.7.1" lazy val http4sServer = "org.http4s" %% "http4s-server" % http4sVersion diff --git a/project/RhoPlugin.scala b/project/RhoPlugin.scala index fa3dc9bb..23449a93 100644 --- a/project/RhoPlugin.scala +++ b/project/RhoPlugin.scala @@ -1,10 +1,13 @@ import sbt._ -import com.typesafe.tools.mima.plugin.MimaKeys._ +import com.typesafe.tools.mima.plugin.MimaPlugin.autoImport.{mimaFailOnNoPrevious, mimaFailOnProblem, mimaPreviousArtifacts} +import sbt.Keys.{moduleName, organization, scalaBinaryVersion, version} object RhoPlugin extends AutoPlugin { object autoImport { val apiVersion = taskKey[(Int, Int)]("Defines the API compatibility version for the project.") + val http4sMimaVersion = settingKey[Option[String]]("Version to target for MiMa compatibility") } + import autoImport._ override def trigger = allRequirements @@ -42,7 +45,19 @@ object RhoPlugin extends AutoPlugin { def isSnapshot(version: String): Boolean = version.endsWith("-SNAPSHOT") - def rhoPreviousArtifacts(lastVersion: String, projectName: String) = { - mimaPreviousArtifacts := Set("org.http4s" %% s"rho-$projectName" % lastVersion) - } + def mimaConfiguration: Seq[Setting[_]] = Seq( + http4sMimaVersion := { + version.value match { + case VersionNumber(Seq(major, minor, patch), _, _) if patch.toInt > 0 => + Some(s"$major.$minor.${patch.toInt - 1}") + case _ => + None + } + }, + mimaFailOnProblem := http4sMimaVersion.value.isDefined, + mimaFailOnNoPrevious := false, + mimaPreviousArtifacts := (http4sMimaVersion.value.map { + organization.value % s"${moduleName.value}_${scalaBinaryVersion.value}" % _ + }).toSet + ) } diff --git a/swagger/src/main/scala/org/http4s/rho/swagger/SwaggerSupport.scala b/swagger/src/main/scala/org/http4s/rho/swagger/SwaggerSupport.scala index b829753a..66d7ec89 100644 --- a/swagger/src/main/scala/org/http4s/rho/swagger/SwaggerSupport.scala +++ b/swagger/src/main/scala/org/http4s/rho/swagger/SwaggerSupport.scala @@ -3,21 +3,20 @@ package rho package swagger import _root_.io.swagger.util.Json -import cats.Monad +import cats.effect.Sync import org.http4s.headers.`Content-Type` import org.http4s.rho.bits.PathAST.{PathMatch, TypedPath} import org.http4s.rho.swagger.models._ import shapeless._ -import scala.collection.immutable.Seq import scala.reflect.runtime.universe._ import scala.collection.immutable.Seq object SwaggerSupport { - def apply[F[_]: Monad](implicit etag: WeakTypeTag[F[_]]): SwaggerSupport[F] = new SwaggerSupport[F] {} + def apply[F[_]: Sync](implicit etag: WeakTypeTag[F[_]]): SwaggerSupport[F] = new SwaggerSupport[F] {} } -abstract class SwaggerSupport[F[_]](implicit F: Monad[F], etag: WeakTypeTag[F[_]]) extends SwaggerSyntax[F] { +abstract class SwaggerSupport[F[_]](implicit F: Sync[F], etag: WeakTypeTag[F[_]]) extends SwaggerSyntax[F] { /** * Create a RhoMiddleware adding a route to get the Swagger json file diff --git a/swagger/src/main/scala/org/http4s/rho/swagger/TypeBuilder.scala b/swagger/src/main/scala/org/http4s/rho/swagger/TypeBuilder.scala index fe2df30f..f4af59b3 100644 --- a/swagger/src/main/scala/org/http4s/rho/swagger/TypeBuilder.scala +++ b/swagger/src/main/scala/org/http4s/rho/swagger/TypeBuilder.scala @@ -314,7 +314,7 @@ object TypeBuilder { DateTimeTypes.exists(t <:< _) private[this] def isCollection(t: Type): Boolean = - t <:< typeOf[collection.Traversable[_]] || t <:< typeOf[java.util.Collection[_]] + t <:< typeOf[collection.Iterable[_]] || t <:< typeOf[java.util.Collection[_]] } private implicit class WrappedType(val t: Type){ diff --git a/swagger/src/main/scala/org/http4s/rho/swagger/models.scala b/swagger/src/main/scala/org/http4s/rho/swagger/models.scala index aeaceea1..7c0eeb9a 100644 --- a/swagger/src/main/scala/org/http4s/rho/swagger/models.scala +++ b/swagger/src/main/scala/org/http4s/rho/swagger/models.scala @@ -5,8 +5,9 @@ import java.util.ArrayList import io.swagger.{models => jm} import io.swagger.models.utils.PropertyModelConverter +import scala.collection.compat.view._ import scala.collection.immutable.Seq -import scala.collection.JavaConverters._ +import scala.jdk.CollectionConverters._ import scala.collection.immutable.ListMap object models { @@ -39,11 +40,11 @@ object models { s.setSchemes(fromList(schemes.map(_.toJModel))) s.setConsumes(fromList(consumes)) s.setProduces(fromList(produces)) - s.setPaths(fromMap(paths.mapValues(_.toJModel))) + s.setPaths(fromMap(paths.view.mapValues(_.toJModel))) s.setSecurity(fromList(security.map(_.toJModel))) - s.setSecurityDefinitions(fromMap(securityDefinitions.mapValues(_.toJModel))) - s.setDefinitions(fromMap(definitions.mapValues(_.toJModel))) - s.setParameters(fromMap(parameters.mapValues(_.toJModel))) + s.setSecurityDefinitions(fromMap(securityDefinitions.view.mapValues(_.toJModel))) + s.setDefinitions(fromMap(definitions.view.mapValues(_.toJModel))) + s.setParameters(fromMap(parameters.view.mapValues(_.toJModel))) s.setExternalDocs(fromOption(externalDocs.map(_.toJModel))) vendorExtensions.foreach { case (key, value:Map[_,_]) => s.setVendorExtension(key, fromMap(value)) @@ -280,9 +281,9 @@ object models { o.setConsumes(fromList(consumes)) o.setProduces(fromList(produces)) o.setParameters(fromList(parameters.map(_.toJModel))) - o.setResponses(fromMap(responses.mapValues(_.toJModel))) - o.setSecurity(fromList(security.map { m => - m.mapValues(_.asJava).asJava + o.setResponses(fromMap(responses.view.mapValues(_.toJModel).toMap)) + o.setSecurity(fromList(security.map { m : Map[String, List[String]] => + m.view.mapValues(_.asJava).toMap.asJava })) o.setExternalDocs(fromOption(externalDocs.map(_.toJModel))) o.setDeprecated(deprecated) @@ -304,7 +305,7 @@ object models { r.setDescription(description) r.setResponseSchema(fromOption(schema.map(_.toJModel).map(new PropertyModelConverter().propertyToModel))) r.setExamples(fromMap(examples)) - r.setHeaders(fromMap(headers.mapValues(_.toJModel))) + r.setHeaders(fromMap(headers.view.mapValues(_.toJModel).toMap)) r } } @@ -343,7 +344,7 @@ object models { m.setDescription(fromOption(description)) m.setRequired(required.asJava) m.setExample(fromOption(example)) - m.setProperties(fromMap(properties.mapValues(_.toJModel))) + m.setProperties(fromMap(properties.view.mapValues(_.toJModel).toMap)) if (additionalProperties.nonEmpty) m.setAdditionalProperties(fromOption(additionalProperties.map(_.toJModel))) m.setDiscriminator(fromOption(discriminator)) m.setExternalDocs(fromOption(externalDocs.map(_.toJModel))) @@ -367,7 +368,7 @@ object models { val am = new jm.ArrayModel am.setType(fromOption(`type`)) am.setDescription(fromOption(description)) - am.setProperties(fromMap(properties.mapValues(_.toJModel))) + am.setProperties(fromMap(properties.view.mapValues(_.toJModel))) am.setItems(fromOption(items.map(_.toJModel))) am.setExample(fromOption(example)) am.setExternalDocs(fromOption(externalDocs.map(_.toJModel))) @@ -395,8 +396,8 @@ object models { cm.setAllOf(new ArrayList(allOf.map(_.toJModel).asJava)) parent.map(_.toJModel).foreach(p => cm.setParent(p)) child.map(_.toJModel).foreach(c => cm.setChild(c)) - cm.setInterfaces(interfaces.map(_.toJModel.asInstanceOf[jm.RefModel]).asJava) - cm.setProperties(properties.mapValues(_.toJModel).asJava) + cm.setInterfaces(interfaces.map(_.toJModel).asJava) + cm.setProperties(properties.view.mapValues(_.toJModel).toMap.asJava) cm.setExample(fromOption(example)) cm.setExternalDocs(fromOption(externalDocs.map(_.toJModel))) cm @@ -414,10 +415,10 @@ object models { , externalDocs : Option[ExternalDocs] = None ) extends Model { - def toJModel: jm.Model = { + def toJModel: jm.RefModel = { val rm = new jm.RefModel(ref) rm.setDescription(fromOption(description)) - rm.setProperties(fromMap(properties.mapValues(_.toJModel))) + rm.setProperties(fromMap(properties.view.mapValues(_.toJModel))) rm.setExample(fromOption(example)) rm.setExternalDocs(fromOption(externalDocs.map(_.toJModel))) rm @@ -633,7 +634,7 @@ object models { } def get$ref() = $ref - def set$ref($ref: String) { this.$ref = $ref } + def set$ref($ref: String) : Unit = { this.$ref = $ref } } def toJModel: jm.parameters.Parameter = { @@ -708,7 +709,7 @@ object models { } def get$ref() = $ref - def set$ref($ref: String) { this.$ref = $ref } + def set$ref($ref: String) : Unit = { this.$ref = $ref } } def withRequired(required: Boolean): AbstractProperty = @@ -747,7 +748,7 @@ object models { ap.setTitle(fromOption(title)) ap.setDescription(fromOption(description)) ap.setFormat(fromOption(format)) - ap.setProperties(fromMap(properties.mapValues(_.toJModel))) + ap.setProperties(fromMap(properties.view.mapValues(_.toJModel))) ap } } @@ -885,5 +886,8 @@ object models { def fromMap[A, B](m: Map[A, B]): java.util.Map[A, B] = if (m.isEmpty) null else m.asJava + + def fromMap[A, B](m: IterableView[(A, B), Iterable[_]]): java.util.Map[A, B] = + if (m.isEmpty) null else m.toMap.asJava } } diff --git a/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerModelsBuilderSpec.scala b/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerModelsBuilderSpec.scala index bdf40812..9ab5c289 100644 --- a/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerModelsBuilderSpec.scala +++ b/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerModelsBuilderSpec.scala @@ -14,7 +14,7 @@ import org.http4s.rho.io._ import org.http4s.rho.swagger.syntax.io._ import org.specs2.mutable.Specification -import scala.language.existentials +import scala.collection.compat.immutable.ArraySeq import scala.collection.immutable.Seq import scala.reflect._ import scala.reflect.runtime.universe._ @@ -735,7 +735,7 @@ class SwaggerModelsBuilderSpec extends Specification { implicit def entityEncoderCsvFile: EntityEncoder[IO, CsvFile] = EntityEncoder.encodeBy[IO, CsvFile](`Content-Type`(MediaType.text.csv, Some(Charset.`UTF-8`))) { _: CsvFile => val bv = "file content".getBytes(Charset.`UTF-8`.nioCharset) - org.http4s.Entity(Stream.emits(bv), Some(bv.length)) + org.http4s.Entity(Stream.emits(ArraySeq.unsafeWrapArray(bv)), Some(bv.length)) } } } diff --git a/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerSpec.scala b/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerSpec.scala index 56df4147..95c6e886 100644 --- a/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerSpec.scala +++ b/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerSpec.scala @@ -6,7 +6,7 @@ import Arbitraries._ import io.swagger.models.{Response => jResponse} import org.http4s.rho.swagger.models.Swagger -import scala.collection.JavaConverters._ +import scala.jdk.CollectionConverters._ class SwaggerSpec extends Specification with ScalaCheck { "The Swagger model can be translated to a 'Java' Swagger model".p diff --git a/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerSupportSpec.scala b/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerSupportSpec.scala index 649afe1d..cdee3c53 100644 --- a/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerSupportSpec.scala +++ b/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerSupportSpec.scala @@ -4,7 +4,6 @@ package swagger import cats.data.NonEmptyList import cats.effect.IO -import cats.syntax.semigroupk._ import org.http4s.rho.bits.MethodAliases.GET import org.http4s.rho.io._ import org.http4s.rho.swagger.models._ @@ -72,14 +71,12 @@ class SwaggerSupportSpec extends Specification { "Provide a way to aggregate routes from multiple RhoRoutes" in { val aggregateSwagger = createSwagger()(baseRoutes.getRoutes ++ moarRoutes.getRoutes) val swaggerRoutes = createSwaggerRoute(aggregateSwagger) - val httpRoutess = NonEmptyList.of(baseRoutes, moarRoutes, swaggerRoutes).map(_.toRoutes()) - - val allthogetherRoutes = httpRoutess.reduceLeft(_ combineK _) + val httpRoutes = NonEmptyList.of(baseRoutes, moarRoutes, swaggerRoutes).reduceLeft(_ and _) val r = Request[IO](GET, Uri(path = "/swagger.json")) val JObject(List((a, JObject(_)), (b, JObject(_)), (c, JObject(_)), (d, JObject(_)))) = - parseJson(RRunner(allthogetherRoutes).checkOk(r)) \\ "paths" + parseJson(RRunner(httpRoutes.toRoutes()).checkOk(r)) \\ "paths" Set(a, b, c, d) should_== Set("/hello", "/hello/{string}", "/goodbye", "/goodbye/{string}") } @@ -103,14 +100,12 @@ class SwaggerSupportSpec extends Specification { "Provide a way to agregate routes from multiple RhoRoutes, with mixed trailing slashes and non-trailing slashes" in { val aggregateSwagger = createSwagger()(baseRoutes.getRoutes ++ moarRoutes.getRoutes ++ mixedTrailingSlashesRoutes.getRoutes) val swaggerRoutes = createSwaggerRoute(aggregateSwagger) - val httpRoutess = NonEmptyList.of(baseRoutes, moarRoutes, swaggerRoutes).map(_.toRoutes()) - - val allthogetherRoutes = httpRoutess.reduceLeft(_ combineK _) + val httpRoutes = NonEmptyList.of(baseRoutes, moarRoutes, swaggerRoutes).reduceLeft(_ and _) val r = Request[IO](GET, Uri(path = "/swagger.json")) val JObject(List((a, JObject(_)), (b, JObject(_)), (c, JObject(_)), (d, JObject(_)), (e, JObject(_)), (f, JObject(_)), (g, JObject(_)))) = - parseJson(RRunner(allthogetherRoutes).checkOk(r)) \\ "paths" + parseJson(RRunner(httpRoutes.toRoutes()).checkOk(r)) \\ "paths" Set(a, b, c, d, e, f, g) should_== Set("/hello", "/hello/{string}", "/goodbye", "/goodbye/{string}", "/foo/", "/foo", "/bar") } diff --git a/version.sbt b/version.sbt index bc0e414f..9a88c375 100644 --- a/version.sbt +++ b/version.sbt @@ -1,2 +1,2 @@ -version in ThisBuild := "0.19.1-SNAPSHOT" +version in ThisBuild := "0.20.0-M1" apiVersion in ThisBuild := RhoPlugin.extractApiVersion(version.value)