From 439990f3a9d1c8ef053afa4ee6d1fbc865fb598e Mon Sep 17 00:00:00 2001 From: Darren Gibson Date: Sun, 25 Apr 2021 15:37:42 -0500 Subject: [PATCH 1/6] Remove HAL support --- build.sbt | 13 +- .../hal/plus/swagger/demo/BusinessLayer.scala | 187 ---------- .../rho/hal/plus/swagger/demo/Main.scala | 36 -- .../rho/hal/plus/swagger/demo/Message.scala | 7 - .../hal/plus/swagger/demo/RestRoutes.scala | 324 ------------------ .../rho/hal/plus/swagger/demo/Routes.scala | 24 -- .../swagger/demo/StaticContentService.scala | 35 -- .../rho/hal/plus/swagger/demo/package.scala | 64 ---- hal/hal.sbt | 3 - .../scala/org/http4s/rho/hal/LinkObject.scala | 110 ------ .../http4s/rho/hal/LinkObjectSerializer.scala | 35 -- .../org/http4s/rho/hal/ResourceObject.scala | 33 -- .../rho/hal/ResourceObjectBuilder.scala | 99 ------ .../rho/hal/ResourceObjectSerializer.scala | 91 ----- .../scala/org/http4s/rho/hal/package.scala | 16 - .../http4s/rho/hal/HalDocBuilderSpec.scala | 146 -------- .../org/http4s/rho/hal/LinkObjectSpec.scala | 64 ---- .../http4s/rho/hal/ResourceObjectSpec.scala | 225 ------------ 18 files changed, 3 insertions(+), 1509 deletions(-) delete mode 100644 examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/BusinessLayer.scala delete mode 100644 examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/Main.scala delete mode 100644 examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/Message.scala delete mode 100644 examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/RestRoutes.scala delete mode 100644 examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/Routes.scala delete mode 100644 examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/StaticContentService.scala delete mode 100644 examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/package.scala delete mode 100644 hal/hal.sbt delete mode 100644 hal/src/main/scala/org/http4s/rho/hal/LinkObject.scala delete mode 100644 hal/src/main/scala/org/http4s/rho/hal/LinkObjectSerializer.scala delete mode 100644 hal/src/main/scala/org/http4s/rho/hal/ResourceObject.scala delete mode 100644 hal/src/main/scala/org/http4s/rho/hal/ResourceObjectBuilder.scala delete mode 100644 hal/src/main/scala/org/http4s/rho/hal/ResourceObjectSerializer.scala delete mode 100644 hal/src/main/scala/org/http4s/rho/hal/package.scala delete mode 100644 hal/src/test/scala/org/http4s/rho/hal/HalDocBuilderSpec.scala delete mode 100644 hal/src/test/scala/org/http4s/rho/hal/LinkObjectSpec.scala delete mode 100644 hal/src/test/scala/org/http4s/rho/hal/ResourceObjectSpec.scala diff --git a/build.sbt b/build.sbt index b99b563df..b44f22a55 100644 --- a/build.sbt +++ b/build.sbt @@ -12,7 +12,7 @@ lazy val rho = project .in(file(".")) .disablePlugins(MimaPlugin) .settings(buildSettings: _*) - .aggregate(`rho-core`, `rho-hal`, `rho-swagger`, `rho-swagger-ui`, `rho-examples`) + .aggregate(`rho-core`, `rho-swagger`, `rho-swagger-ui`, `rho-examples`) lazy val `rho-core` = project .in(file("core")) @@ -32,12 +32,6 @@ lazy val `rho-core` = project libraryDependencies ++= Seq("org.scala-lang.modules" %% "scala-collection-compat" % "2.4.4") ) -lazy val `rho-hal` = project - .in(file("hal")) - .settings(buildSettings :+ halDeps: _*) - .settings(mimaConfiguration) - .dependsOn(`rho-core`) - lazy val `rho-swagger` = project .in(file("swagger")) .settings(buildSettings :+ swaggerDeps: _*) @@ -74,7 +68,6 @@ lazy val docs = project (ScalaUnidoc / unidoc / scalacOptions) ++= versionSpecificEnabledFlags(scalaVersion.value), (ScalaUnidoc / unidoc / unidocProjectFilter) := inProjects( `rho-core`, - `rho-hal`, `rho-swagger` ), git.remoteRepo := "git@github.com:http4s/rho.git", @@ -87,7 +80,7 @@ lazy val docs = project } yield (f, s"api/$major.$minor/$d") } ) - .dependsOn(`rho-core`, `rho-hal`, `rho-swagger`) + .dependsOn(`rho-core`, `rho-swagger`) lazy val `rho-examples` = project .in(file("examples")) @@ -99,7 +92,7 @@ lazy val `rho-examples` = project libraryDependencies ++= Seq(logbackClassic, http4sXmlInstances), dontPublish ) - .dependsOn(`rho-swagger`, `rho-swagger-ui`, `rho-hal`) + .dependsOn(`rho-swagger`, `rho-swagger-ui`) lazy val compilerFlags = Seq( "-feature", 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 deleted file mode 100644 index 9f06bac9e..000000000 --- a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/BusinessLayer.scala +++ /dev/null @@ -1,187 +0,0 @@ -package com.http4s.rho.hal.plus.swagger.demo - -import java.util.SortedSet - -import net.sf.uadetector.datastore.DataStore -import net.sf.uadetector.internal.data.Data -import net.sf.uadetector.internal.data.domain.{ - BrowserOperatingSystemMapping, - Browser => UBrowser, - BrowserPattern => UBrowserPattern, - BrowserType => UBrowserType, - OperatingSystem => UOperatingSystem -} - -import scala.jdk.CollectionConverters._ -import scala.collection.immutable.Seq - -// -- -// Do not take this implementation too serious. It is only there to have some -// resources to play around through REST with JSON HAL -// -- - -case class Browser(id: Int, name: String, typeId: Int) -case class BrowserPattern(id: Int, position: Int, regex: String) -case class BrowserType(id: Int, name: String) -case class OperatingSystem(id: Int, name: String) - -/** Example of an interface to get access to your business domain */ -trait BusinessLayer { - def countBrowsers: Int - def countBrowsersByBrowserTypeId(id: Int): Int - def countOperatingSystems: Int - def hasBrowsersByOperatingSystemId(id: Int): Boolean - def hasOperatingSystemsByBrowserId(id: Int): Boolean - def findBrowser(id: Int): Option[Browser] - def findBrowserIdByPatternId(id: Int): Option[Int] - def findBrowserPattern(id: Int): Option[BrowserPattern] - def findBrowserPatternsByBrowserId(id: Int): Option[Seq[BrowserPattern]] - def findBrowserPatterns(firstResult: Int, maxResults: Int): Seq[BrowserPattern] - def findBrowserType(id: Int): Option[BrowserType] - def findBrowserTypes: Seq[BrowserType] - def findBrowsers(firstResult: Int, maxResults: Int): Seq[Browser] - def findBrowsersByBrowserTypeId(id: Int, firstResult: Int, maxResults: Int): Seq[Browser] - def findBrowsersByOperatingSystemId(id: Int): Seq[Browser] - def findOperatingSystem(id: Int): Option[OperatingSystem] - def findOperatingSystems(firstResult: Int, maxResults: Int): Seq[OperatingSystem] - def findOperatingSystemsByBrowserId(id: Int): Seq[OperatingSystem] -} - -/** Implementation to get access to your business domain */ -class UADetectorDatabase(val dataStore: DataStore) extends BusinessLayer { - - private def toBrowser(b: UBrowser): Browser = - Browser(b.getId, b.getFamilyName, b.getType.getId) - - private def toBrowserPattern(b: UBrowserPattern): BrowserPattern = - BrowserPattern(b.getId, b.getPosition, b.getPattern.pattern) - - private def toBrowserType(t: UBrowserType): BrowserType = - BrowserType(t.getId, t.getName) - - private def toOperatingSystem(b: UOperatingSystem): OperatingSystem = - OperatingSystem(b.getId, b.getName) - - private def data: Data = - dataStore.getData - - private def browsers: List[Browser] = - data.getBrowsers.asScala.foldLeft(List[Browser]()) { (acc, b) => - toBrowser(b) :: acc - } - - private def browserPatterns: List[BrowserPattern] = - data.getBrowserPatterns.asScala.foldLeft(List[BrowserPattern]()) { (acc, e) => - val ps = e._2.asScala.map { p => - toBrowserPattern(p) - }.toList - ps ::: acc - } - - private def browserTypes = - data.getBrowserTypes.asScala.map { t => - toBrowserType(t._2) - }.toList - - private def operatingSystems: List[OperatingSystem] = - data.getOperatingSystems.asScala.foldLeft(List[OperatingSystem]()) { (acc, o) => - toOperatingSystem(o) :: acc - } - - def countBrowsers = data.getBrowsers.size - - def countBrowsersByBrowserTypeId(id: Int) = - browsers.foldLeft(0) { (acc, b) => - if (b.typeId == id) acc + 1 - else acc - } - - def countOperatingSystems = data.getOperatingSystems.size - - def hasBrowsersByOperatingSystemId(id: Int) = { - val found = data.getBrowserToOperatingSystemMappings.asScala.collectFirst { - case m: BrowserOperatingSystemMapping if m.getOperatingSystemId == id => m - } - found.isDefined - } - - def hasOperatingSystemsByBrowserId(id: Int) = { - val found = data.getBrowserToOperatingSystemMappings.asScala.collectFirst { - case m: BrowserOperatingSystemMapping if m.getBrowserId == id => m - } - found.isDefined - } - - def findBrowser(id: Int) = - browsers.collectFirst { - case b: Browser if b.id == id => b - } - - def findBrowserIdByPatternId(id: Int) = - data.getPatternToBrowserMap.entrySet.asScala.collectFirst { - case e: java.util.Map.Entry[UBrowserPattern, UBrowser] if e.getKey.getId == id => - e.getValue.getId - } - - def findBrowserPattern(id: Int) = - browserPatterns.collectFirst { - case p: BrowserPattern if p.id == id => p - } - - def findBrowserPatternsByBrowserId(id: Int) = - data.getBrowserPatterns.get(id) match { - case ps: SortedSet[UBrowserPattern] => Some(ps.asScala.map(p => toBrowserPattern(p)).toList) - case _ => None - } - - def findBrowserPatterns(firstResult: Int, maxResults: Int) = - browserPatterns.drop(firstResult).take(maxResults) - - def findBrowserType(id: Int) = - browserTypes.collectFirst { - case b: BrowserType if b.id == id => b - } - - def findBrowserTypes = - browserTypes - - def findBrowsers(firstResult: Int, maxResults: Int) = - browsers.drop(firstResult).take(maxResults) - - def findBrowsersByBrowserTypeId(id: Int, firstResult: Int, maxResults: Int) = { - val matchingBrowsers = for { - browser <- browsers - if browser.typeId == id - } yield browser - matchingBrowsers.drop(firstResult).take(maxResults) - } - - def findBrowsersByOperatingSystemId(id: Int) = { - val matchingBrowsers = for { - mapping <- data.getBrowserToOperatingSystemMappings.asScala - if mapping.getOperatingSystemId == id - browser <- browsers - if browser.id == mapping.getBrowserId - } yield browser - matchingBrowsers.toList - } - - def findOperatingSystem(id: Int) = - operatingSystems.collectFirst { - case o: OperatingSystem if o.id == id => o - } - - def findOperatingSystems(firstResult: Int, maxResults: Int) = - operatingSystems.drop(firstResult).take(maxResults) - - def findOperatingSystemsByBrowserId(id: Int) = { - val matchingOperatingSystems = for { - mapping <- data.getBrowserToOperatingSystemMappings.asScala - if mapping.getBrowserId == id - os <- operatingSystems - if os.id == mapping.getOperatingSystemId - } yield os - matchingOperatingSystems.toList - } - -} 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 deleted file mode 100644 index a5959d5d6..000000000 --- a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/Main.scala +++ /dev/null @@ -1,36 +0,0 @@ -package com.http4s.rho.hal.plus.swagger.demo - -import cats.effect.{Blocker, ExitCode, IO, IOApp} -import net.sf.uadetector.service.UADetectorServiceFactory.ResourceModuleXmlDataStore -import cats.implicits._ -import org.http4s.implicits._ -import org.http4s.server.blaze._ -import org.log4s.getLogger - -import scala.concurrent.ExecutionContext.global - -object Main extends IOApp { - private val logger = getLogger - - val port: Int = Option(System.getenv("HTTP_PORT")) - .map(_.toInt) - .getOrElse(8080) - - logger.info(s"Starting Hal example on '$port'") - - def run(args: List[String]): IO[ExitCode] = - Blocker[IO].use { blocker => - val businessLayer = new UADetectorDatabase(new ResourceModuleXmlDataStore()) - - val routes = - new Routes(businessLayer, blocker) - - BlazeServerBuilder[IO](global) - .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/Message.scala b/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/Message.scala deleted file mode 100644 index f577a84c9..000000000 --- a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/Message.scala +++ /dev/null @@ -1,7 +0,0 @@ -package com.http4s.rho.hal.plus.swagger.demo - -sealed trait MessageType -case object Error extends MessageType -case object Info extends MessageType -case object Warning extends MessageType -case class Message(text: String, `type`: MessageType) 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 deleted file mode 100644 index c78b796df..000000000 --- a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/RestRoutes.scala +++ /dev/null @@ -1,324 +0,0 @@ -package com.http4s.rho.hal.plus.swagger.demo - -import cats.effect.Sync -import org.http4s.rho.RhoRoutes -import org.http4s.rho.hal.{ResourceObjectBuilder => ResObjBuilder, _} -import org.http4s.{Request, Uri} - -import scala.collection.immutable.Seq -import scala.collection.mutable.ListBuffer - -class RestRoutes[F[+_]: Sync](val businessLayer: BusinessLayer) extends RhoRoutes[F] { - - // # Query Parameters - - val firstResult = param[Int]("firstResult", 0, (i: Int) => i >= 0) - val maxResults = param[Int]("maxResults", 10, (i: Int) => i >= 1 && i <= 100) - val owner = param[String]("owner", "") - val groups = param[Seq[String]]("groups", Nil) - val searchTerm = param[String]("searchTerm", "") - val sortBy = param[String]("sortBy", "") - val showHidden = param[Int]("showHidden", 0) - - // # Path Variables - - val id = pathVar[Int]("id") - - // # HTTP Routes - - val browsers = "browsers" - GET / browsers +? firstResult & maxResults |>> { (request: Request[F], first: Int, max: Int) => - val configurations = businessLayer.findBrowsers(first, max) - val total = businessLayer.countBrowsers - val hal = browsersAsResource(request, first, max, configurations, total) - - Ok(hal.build()) - } - - val browserById = browsers / id - GET / browserById |>> { (request: Request[F], id: Int) => - val found = for { browser <- businessLayer.findBrowser(id) } yield { - val b = browserAsResourceObject(browser, request) - if (businessLayer.hasOperatingSystemsByBrowserId(browser.id)) - for (tpl <- operatingSystemsByBrowser.asUriTemplate(request)) - b.link("operating-systems", tpl.expandPath("id", browser.id).toUriIfPossible.get) - - Ok(b.build()) - } - - found.getOrElse(NotFound(warning(s"Browser $id not found"))) - } - - val browserPatternsById = browsers / id / "patterns" - GET / browserPatternsById |>> { (request: Request[F], id: Int) => - val found = for { - patterns <- businessLayer.findBrowserPatternsByBrowserId(id) - } yield Ok(browserPatternsAsResource(request, 0, Int.MaxValue, patterns, patterns.size).build()) - - found.getOrElse(NotFound(warning(s"Browser $id not found"))) - } - - val browserPatterns = "browser-patterns" - GET / browserPatterns +? firstResult & maxResults |>> { - (request: Request[F], first: Int, max: Int) => - val patterns = businessLayer.findBrowserPatterns(first, max) - val total = businessLayer.countBrowsers - val hal = browserPatternsAsResource(request, first, max, patterns, total) - - Ok(hal.build()) - } - - val browserPatternById = browserPatterns / id - GET / browserPatternById |>> { (request: Request[F], id: Int) => - val found = for { pattern <- businessLayer.findBrowserPattern(id) } yield { - val b = browserPatternAsResourceObject(pattern, request) - for { - tpl <- browserById.asUriTemplate(request) - browserId <- businessLayer.findBrowserIdByPatternId(pattern.id) - } b.link("browser", tpl.expandPath("id", browserId).toUriIfPossible.get) - - Ok(b.build()) - } - - found.getOrElse(NotFound(warning(s"Browser $id not found"))) - } - - val browserTypes = "browser-types" - GET / browserTypes |>> { (request: Request[F]) => - val types = businessLayer.findBrowserTypes - val hal = browserTypesAsResource(request, types) - - Ok(hal.build()) - } - - val browserTypeById = browserTypes / id - GET / browserTypeById |>> { (request: Request[F], id: Int) => - val found = for { browserType <- businessLayer.findBrowserType(id) } yield { - val b = browserTypeAsResourceObject(browserType, request) - for { - tpl <- browsersByBrowserTypeId.asUriTemplate(request) - } b.link("browsers", tpl.expandPath("id", browserType.id).toUriIfPossible.get) - - Ok(b.build()) - } - - found.getOrElse(NotFound(warning(s"Browser type $id not found"))) - } - - val browsersByBrowserTypeId = browserTypes / id / "browsers" - GET / browsersByBrowserTypeId +? firstResult & maxResults |>> { - (request: Request[F], id: Int, first: Int, max: Int) => - val browsers = businessLayer.findBrowsersByBrowserTypeId(id, first, max) - val total = businessLayer.countBrowsersByBrowserTypeId(id) - - if (browsers.nonEmpty) - Ok(browsersAsResource(request, first, max, browsers, total).build()) - else - NotFound(warning(s"No browsers for type $id found")) - } - - val operatingSystems = "operating-systems" - GET / operatingSystems +? firstResult & maxResults |>> { - (request: Request[F], first: Int, max: Int) => - val configurations = businessLayer.findOperatingSystems(first, max) - val total = businessLayer.countOperatingSystems - val hal = operatingSystemsAsResource(request, first, max, configurations, total) - - Ok(hal.build()) - } - - val operatingSystemById = operatingSystems / id - GET / operatingSystemById |>> { (request: Request[F], id: Int) => - val found = for { operatingSystem <- businessLayer.findOperatingSystem(id) } yield { - val b = operatingSystemAsResourceObject(operatingSystem, request) - if (businessLayer.hasBrowsersByOperatingSystemId(operatingSystem.id)) - for (tpl <- browsersByOperatingSystem.asUriTemplate(request)) - b.link("browsers", tpl.expandPath("id", operatingSystem.id).toUriIfPossible.get) - - Ok(b.build()) - } - - found.getOrElse(NotFound(warning(s"OperatingSystem $id not found"))) - } - - val browsersByOperatingSystem = operatingSystemById / "browsers" - GET / browsersByOperatingSystem |>> { (request: Request[F], id: Int) => - val browsers = businessLayer.findBrowsersByOperatingSystemId(id) - - if (browsers.nonEmpty) - Ok(browsersAsResource(request, 0, Int.MaxValue, browsers, browsers.size).build()) - else - NotFound(warning(s"No Browsers for operating system $id found")) - } - - val operatingSystemsByBrowser = browserById / "operating-systems" - GET / operatingSystemsByBrowser |>> { (request: Request[F], id: Int) => - val operatingSystems = businessLayer.findOperatingSystemsByBrowserId(id) - - if (operatingSystems.nonEmpty) - Ok( - operatingSystemsAsResource( - request, - 0, - Int.MaxValue, - operatingSystems, - operatingSystems.size - ).build() - ) - else - NotFound(warning(s"No operating systems for browser $id found")) - } - - GET / "" |>> { request: Request[F] => - val b = new ResObjBuilder[Nothing, Nothing]() - b.link("self", request.uri) - for (uri <- browsers.asUri(request)) b.link(browsers, uri.toString, "Lists browsers") - for (uri <- browserPatterns.asUri(request)) - b.link(browserPatterns, uri.toString, "Lists browser patterns") - for (uri <- browserTypes.asUri(request)) - b.link(browserTypes, uri.toString, "Lists browser types") - for (uri <- operatingSystems.asUri(request)) - b.link(operatingSystems, uri.toString, "Lists operating systems") - - Ok(b.build()) - } - - // # JSON HAL helpers - - def browsersAsResource( - request: Request[F], - first: Int, - max: Int, - browsers: Seq[Browser], - total: Int): ResObjBuilder[(String, Long), Browser] = { - val self = request.uri - val hal = new ResObjBuilder[(String, Long), Browser]() - hal.link("self", selfWithFirstAndMax(self, first, max)) - hal.content("total", total) - - if (first + max < total) { - hal.link("next", self +? (firstResult, first + max) +? (maxResults, max)) - } - if (first > 0) { - hal.link("prev", self +? (firstResult, Math.max(first - max, 0)) +? (maxResults, max)) - } - val res = ListBuffer[ResourceObject[Browser, Nothing]]() - browsers.foreach { browser => - res.append(browserAsResourceObject(browser, request).build()) - } - hal.resources("browsers", res.toList) - } - - def browserAsResourceObject( - browser: Browser, - request: Request[F]): ResObjBuilder[Browser, Nothing] = { - val b = new ResObjBuilder[Browser, Nothing]() - for (tpl <- browserById.asUriTemplate(request)) - b.link("self", tpl.expandPath("id", browser.id).toUriIfPossible.get) - for (tpl <- browserPatternsById.asUriTemplate(request)) - b.link("patterns", tpl.expandPath("id", browser.id).toUriIfPossible.get) - for (tpl <- browserTypeById.asUriTemplate(request)) - b.link("type", tpl.expandPath("id", browser.typeId).toUriIfPossible.get) - - b.content(browser) - } - - def browserPatternsAsResource( - request: Request[F], - first: Int, - max: Int, - browserPatterns: Seq[BrowserPattern], - total: Int): ResObjBuilder[(String, Long), BrowserPattern] = { - val self = request.uri - val hal = new ResObjBuilder[(String, Long), BrowserPattern]() - hal.link("self", selfWithFirstAndMax(self, first, max)) - hal.content("total", total) - if (first + max < total) { - hal.link("next", self +? (firstResult, first + max) +? (maxResults, max)) - } - if (first > 0) { - hal.link("prev", self +? (firstResult, Math.max(first - max, 0)) +? (maxResults, max)) - } - val res = ListBuffer[ResourceObject[BrowserPattern, Nothing]]() - browserPatterns.foreach { browserPattern => - res.append(browserPatternAsResourceObject(browserPattern, request).build()) - } - hal.resources("browserPatterns", res.toList) - } - - def browserPatternAsResourceObject( - browserPattern: BrowserPattern, - request: Request[F]): ResObjBuilder[BrowserPattern, Nothing] = { - val b = new ResObjBuilder[BrowserPattern, Nothing]() - for (tpl <- browserPatternById.asUriTemplate(request)) - b.link("self", tpl.expandPath("id", browserPattern.id).toUriIfPossible.get) - b.content(browserPattern) - } - - def browserTypeAsResourceObject( - browserType: BrowserType, - request: Request[F]): ResObjBuilder[BrowserType, Nothing] = { - val b = new ResObjBuilder[BrowserType, Nothing]() - for (tpl <- browserTypeById.asUriTemplate(request)) - b.link("self", tpl.expandPath("id", browserType.id).toUriIfPossible.get) - b.content(browserType) - } - - def browserTypesAsResource( - request: Request[F], - browserTypes: Seq[BrowserType]): ResObjBuilder[Nothing, BrowserType] = { - val self = request.uri - val hal = new ResObjBuilder[Nothing, BrowserType]() - hal.link("self", self) - val res = ListBuffer[ResourceObject[BrowserType, Nothing]]() - browserTypes.foreach { browserType => - res.append(browserTypeAsResourceObject(browserType, request).build()) - } - hal.resources("browserTypes", res.toList) - } - - def operatingSystemsAsResource( - request: Request[F], - first: Int, - max: Int, - operatingSystems: Seq[OperatingSystem], - total: Int): ResObjBuilder[(String, Long), OperatingSystem] = { - val self = request.uri - val hal = new ResObjBuilder[(String, Long), OperatingSystem]() - hal.link("self", selfWithFirstAndMax(self, first, max)) - hal.content("total", total) - if (first + max < total) { - hal.link("next", self +? (firstResult, first + max) +? (maxResults, max)) - } - if (first > 0) { - hal.link("prev", self +? (firstResult, Math.max(first - max, 0)) +? (maxResults, max)) - } - val res = ListBuffer[ResourceObject[OperatingSystem, Nothing]]() - operatingSystems.foreach { operatingSystem => - res.append(operatingSystemAsResourceObject(operatingSystem, request).build()) - } - hal.resources("operatingSystems", res.toList) - } - - def operatingSystemAsResourceObject( - operatingSystem: OperatingSystem, - request: Request[F]): ResObjBuilder[OperatingSystem, Nothing] = { - val b = new ResObjBuilder[OperatingSystem, Nothing]() - for (tpl <- operatingSystemById.asUriTemplate(request)) - b.link("self", tpl.expandPath("id", operatingSystem.id).toUriIfPossible.get) - b.content(operatingSystem) - } - - def selfWithFirstAndMax(self: Uri, first: Int, max: Int): Uri = - if (!self.containsQueryParam(firstResult) && !self.containsQueryParam(maxResults)) self - else self +? (firstResult, first) +? (maxResults, max) - - // use JSON messages if a non-successful HTTP status must be send - - def message(text: String, `type`: MessageType): Message = - Message(text, `type`) - def error(text: String): Message = message(text, Error) - def info(text: String): Message = message(text, Info) - def warning(text: String): Message = message(text, Warning) - -} 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 deleted file mode 100644 index 102be7166..000000000 --- a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/Routes.scala +++ /dev/null @@ -1,24 +0,0 @@ -package com.http4s.rho.hal.plus.swagger.demo - -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, blocker: Blocker)(implicit - T: Timer[IO], - cs: ContextShift[IO]) { - - val middleware: RhoMiddleware[IO] = - createRhoMiddleware() - - val dynamicContent: HttpRoutes[IO] = - new RestRoutes[IO](businessLayer).toRoutes(middleware) - - /** Routes for getting static resources. These might be served more efficiently by apache2 or nginx, - * but its nice to keep it self contained - */ - val staticContent: HttpRoutes[IO] = - 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 deleted file mode 100644 index 44d103789..000000000 --- a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/StaticContentService.scala +++ /dev/null @@ -1,35 +0,0 @@ -package com.http4s.rho.hal.plus.swagger.demo - -import cats.data.OptionT -import cats.effect.{Blocker, ContextShift, Sync, Timer} -import org.http4s.dsl.Http4sDsl -import org.http4s.{HttpRoutes, Request, Response, StaticFile} - -class StaticContentService[F[_]: Sync: Timer: ContextShift](dsl: Http4sDsl[F], blocker: Blocker) { - import dsl._ - - private val halUiDir = "/hal-browser" - private val swaggerUiDir = "/swagger-ui" - - /** 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: HttpRoutes[F] = HttpRoutes[F] { - - // JSON HAL User Interface - case req if req.uri.path.startsWith("/js/") => fetchResource(halUiDir + req.pathInfo, req) - case req if req.uri.path.startsWith("/vendor/") => fetchResource(halUiDir + req.pathInfo, req) - case req @ GET -> Root / "hal-ui" => fetchResource(halUiDir + "/browser.html", req) - - // 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) - } - - 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 deleted file mode 100644 index e3d40f17c..000000000 --- a/examples/src/main/scala/com/http4s/rho/hal/plus/swagger/demo/package.scala +++ /dev/null @@ -1,64 +0,0 @@ -package com.http4s.rho.hal.plus.swagger - -import scala.collection.immutable.Seq -import scala.util.Failure -import scala.util.Success -import scala.util.Try -import org.http4s.Charset -import org.http4s.headers.`Content-Type` -import org.http4s.MediaType -import org.http4s.EntityEncoder -import org.http4s.rho.hal.LinkObjectSerializer -import org.http4s.rho.hal.ResourceObject -import org.http4s.rho.hal.ResourceObjectSerializer -import org.http4s.rho.bits.TypedQuery -import org.json4s.DefaultFormats -import org.json4s.Extraction.decompose -import org.json4s.Formats -import org.json4s.JsonAST.JValue -import org.json4s.ext.JodaTimeSerializers -import org.json4s.jackson.JsonMethods.compact -import org.json4s.jackson.JsonMethods.render - -package object demo { - - ///// implicit helper functions ///// - - /** Defines the defaults to render known and unknown types to JSON */ - implicit val jsonFormats: Formats = - DefaultFormats ++ - JodaTimeSerializers.all + - new LinkObjectSerializer + - new ResourceObjectSerializer - - implicit def resourceObjectAsJsonEncoder[F[_], A, B]: EntityEncoder[F, ResourceObject[A, B]] = - EntityEncoder - .stringEncoder[F](Charset.`UTF-8`) - .contramap { r: ResourceObject[A, B] => compact(render(json(r))) } - .withContentType(`Content-Type`(MediaType.application.`vnd.hal+json`, Charset.`UTF-8`)) - - implicit def messageAsJsonEncoder[F[_]]: EntityEncoder[F, Message] = - EntityEncoder - .stringEncoder[F](Charset.`UTF-8`) - .contramap { r: Message => compact(render(json(r))) } - .withContentType(`Content-Type`(MediaType.application.json, Charset.`UTF-8`)) - - /** Extracts the name of the first query parameter as string */ - implicit def paramName[F[_]](q: TypedQuery[F, _]): String = q.names.head - - ///// regular helper functions ///// - - /** Converts a sequence of `Try` into a `Try` of sequences. In case failures - * are in there the first failure will be returned. - */ - def flattenTry[T](xs: Seq[Try[T]]): Try[Seq[T]] = { - val (ss: Seq[Success[T]] @unchecked, fs: Seq[Failure[T]] @unchecked) = xs.partition(_.isSuccess) - if (fs.isEmpty) Success(ss.map(_.get)) - else Failure[Seq[T]](fs(0).exception) // Only keep the first failure - } - - /** Converts a given object into a JSON structure */ - def json[A <: Equals](a: A): JValue = - decompose(a) - -} diff --git a/hal/hal.sbt b/hal/hal.sbt deleted file mode 100644 index 8c6b4b680..000000000 --- a/hal/hal.sbt +++ /dev/null @@ -1,3 +0,0 @@ -name := "rho-hal" - -description := "JSON HAL support for Rho" diff --git a/hal/src/main/scala/org/http4s/rho/hal/LinkObject.scala b/hal/src/main/scala/org/http4s/rho/hal/LinkObject.scala deleted file mode 100644 index cc12314e4..000000000 --- a/hal/src/main/scala/org/http4s/rho/hal/LinkObject.scala +++ /dev/null @@ -1,110 +0,0 @@ -package org.http4s.rho.hal - -import org.http4s.Uri -import org.http4s.UriTemplate - -/** A Link Object represents a hyperlink from the containing resource to a URI. - * - * @param href The "href" property is REQUIRED. - * - * Its value is either a URI [RFC3986] or a URI Template [RFC6570]. - * - * If the value is a URI Template then the Link Object SHOULD have a - * "templated" attribute whose value is `true`. - * - * @param templated The "templated" property is OPTIONAL. - * - * Its value is boolean and SHOULD be true when the Link Object's "href" - * property is a URI Template. - * - * Its value SHOULD be considered false if it is undefined or any other - * value than true. - * - * @param type The "type" property is OPTIONAL. - * - * Its value is a string used as a hint to indicate the media type - * expected when dereferencing the target resource. - * - * @param deprecation The "deprecation" property is OPTIONAL. - * - * Its presence indicates that the link is to be deprecated (i.e. - * removed) at a future date. Its value is a URL that SHOULD provide - * further information about the deprecation. - * - * A client SHOULD provide some notification (for example, by logging a - * warning message) whenever it traverses over a link that has this - * property. The notification SHOULD include the deprecation property's - * value so that a client manitainer can easily find information about - * the deprecation. - * - * @param name The "name" property is OPTIONAL. - * - * Its value MAY be used as a secondary key for selecting Link Objects - * which share the same relation type. - * - * @param profile The "profile" property is OPTIONAL. - * - * Its value is a string which is a URI that hints about the profile (as - * defined by [I-D.wilde-profile-link]) of the target resource. - * - * @param title The "title" property is OPTIONAL. - * - * Its value is a string and is intended for labelling the link with a - * human-readable identifier (as defined by [RFC5988]). - * - * @param hreflang The "hreflang" property is OPTIONAL. - * - * Its value is a string and is intended for indicating the language of - * the target resource (as defined by [RFC5988]). - */ -trait LinkObjectLike { - def href: String - def templated: Option[Boolean] = None - def `type`: Option[String] = None - def deprecation: Option[String] = None - def name: Option[String] = None - def profile: Option[String] = None - def title: Option[String] = None - def hreflang: Option[String] = None -} - -/** Represents the default implementation of a Link Object which all aspects of - * the specification. - */ -case class LinkObject( - href: String, - override val name: Option[String] = None, - override val title: Option[String] = None, - override val templated: Option[Boolean] = None, - override val `type`: Option[String] = None, - override val deprecation: Option[String] = None, - override val profile: Option[String] = None, - override val hreflang: Option[String] = None) - extends LinkObjectLike { - require(href != null, "href must not be null") - require(href.nonEmpty, "href must not be empty") -} - -object LinkObject { - - def fromUri( - href: Uri, - name: Option[String] = None, - title: Option[String] = None, - `type`: Option[String] = None, - deprecation: Option[String] = None, - profile: Option[String] = None, - hreflang: Option[String] = None): LinkObject = - new LinkObject(href.toString, name, title, None, `type`, deprecation, profile, hreflang) - - def fromTemplate( - href: UriTemplate, - name: Option[String] = None, - title: Option[String] = None, - `type`: Option[String] = None, - deprecation: Option[String] = None, - profile: Option[String] = None, - hreflang: Option[String] = None): LinkObject = - new LinkObject(href.toString, name, title, Some(true), `type`, deprecation, profile, hreflang) - -} diff --git a/hal/src/main/scala/org/http4s/rho/hal/LinkObjectSerializer.scala b/hal/src/main/scala/org/http4s/rho/hal/LinkObjectSerializer.scala deleted file mode 100644 index 583103205..000000000 --- a/hal/src/main/scala/org/http4s/rho/hal/LinkObjectSerializer.scala +++ /dev/null @@ -1,35 +0,0 @@ -package org.http4s.rho.hal - -import org.json4s._ - -import scala.collection.mutable.ArrayBuffer - -object LinkObjectSerializer { - val serialize: PartialFunction[Any, JValue] = { case l: LinkObject => - serialize(l) - } - def serialize(l: LinkObject): JObject = { - val b = new ArrayBuffer[JField]() - b.append(JField("href", JString(l.href))) - if (l.templated.isDefined) - b.append(JField("templated", JBool(l.templated.get))) - if (l.`type`.isDefined) - b.append(JField("type", JString(l.`type`.get))) - if (l.deprecation.isDefined) - b.append(JField("deprecation", JString(l.deprecation.get))) - if (l.name.isDefined) - b.append(JField("name", JString(l.name.get))) - if (l.profile.isDefined) - b.append(JField("profile", JString(l.profile.get))) - if (l.title.isDefined) - b.append(JField("title", JString(l.title.get))) - if (l.hreflang.isDefined) - b.append(JField("hreflang", JString(l.hreflang.get))) - JObject(b.toList) - } -} - -class LinkObjectSerializer - extends CustomSerializer[LinkObject](_ => - (PartialFunction.empty, LinkObjectSerializer.serialize) - ) diff --git a/hal/src/main/scala/org/http4s/rho/hal/ResourceObject.scala b/hal/src/main/scala/org/http4s/rho/hal/ResourceObject.scala deleted file mode 100644 index c9f3be6dd..000000000 --- a/hal/src/main/scala/org/http4s/rho/hal/ResourceObject.scala +++ /dev/null @@ -1,33 +0,0 @@ -package org.http4s.rho.hal - -/** A Resource Object represents a resource. - * - * @param links contains links to other resources - * - * The "links" property is OPTIONAL. - * - * It is an object whose property names are link relation types (as - * defined by [RFC5988]) and values are either a Link Object or an array - * of Link Objects. The subject resource of these links is the Resource - * Object of which the containing "_links" object is a property. - * - * @param embedded contains embedded resources - * - * The "embedded" property is OPTIONAL - * - * It is an object whose property names are link relation types (as - * defined by [RFC5988]) and values are either a Resource Object or an - * array of Resource Objects. - * - * Embedded Resources MAY be a full, partial, or inconsistent version of - * the representation served from the target URI. - * - * @param content represents an object that contains all other properties - * - * All properties of the content instance will be rendered as JSON, and - * represent the current state of the resource. - */ -case class ResourceObject[T, E]( - links: Links = Nil, - embedded: Embedded[E] = Nil, - content: Option[T] = None) diff --git a/hal/src/main/scala/org/http4s/rho/hal/ResourceObjectBuilder.scala b/hal/src/main/scala/org/http4s/rho/hal/ResourceObjectBuilder.scala deleted file mode 100644 index 3f69f9309..000000000 --- a/hal/src/main/scala/org/http4s/rho/hal/ResourceObjectBuilder.scala +++ /dev/null @@ -1,99 +0,0 @@ -package org.http4s.rho.hal - -import scala.collection.mutable.LinkedHashMap -import scala.collection.immutable.Seq -import org.http4s.Uri -import org.http4s.UriTemplate - -class ResourceObjectBuilder[T, E] { - private val _links = LinkedHashMap[String, Either[LinkObject, Seq[LinkObject]]]() - private val _embedded = - LinkedHashMap[String, Either[ResourceObject[E, _], Seq[ResourceObject[E, _]]]]() - private var content: Option[T] = None - - def content(c: T): this.type = { - content = if (c == null) None else Some(c) - this - } - - /** Creates a single link object with a given `href` and the specified `name` - * to this document builder. In case the same `name` already exists the link - * object will be overwritten. - */ - def link(name: String, href: String, templated: Option[Boolean] = None): this.type = - link(name, LinkObject(href, templated = templated)) - - /** Creates a single link object with a given `href` and the specified `name` - * to this document builder. In case the same `name` already exists the link - * object will be overwritten. - */ - def link(name: String, href: String, title: String): this.type = - link(name, LinkObject(href, title = Some(title))) - - /** Creates a single link object with a given `Uri` as `href` and the - * specified `name` to this document builder. In case the same `name` already - * exists the link object will be overwritten. - */ - def link(name: String, href: Uri): this.type = - link(name, LinkObject(href.toString)) - - /** Creates a single link object with a given `UriTemplate` as `href` and - * the specified `name` to this document builder. In case the same `name` - * already exists the link object will be overwritten. - */ - def link(name: String, href: UriTemplate): this.type = - link(name, LinkObject(href.toString, templated = Some(true))) - - /** Puts a single link object with the specified `name` to this document - * builder. In case the same `name` already exists the link object will be - * overwritten. - */ - def link(name: String, linkObj: LinkObject): this.type = { - _links.put(name, Left(linkObj)) - this - } - - /** Puts a list of link objects with the specified `name` to this document - * builder. In case the same `name` already exists the link objects will be - * overwritten. - */ - def links(name: String, linkObjs: List[LinkObject]): this.type = { - _links.put(name, Right(linkObjs)) - this - } - - /** Puts an array of link objects with the specified `name` to this document - * builder. In case the same `name` already exists the link objects will be - * overwritten. - */ - def links(name: String, linkObjs: LinkObject*): this.type = - links(name, linkObjs.toList) - - /** Puts a single resource object with the specified `name` to this - * document builder. In case the same `name` already exists the resource - * object will be overwritten. - */ - def resource(name: String, resObj: ResourceObject[E, _]): this.type = { - _embedded.put(name, Left(resObj)) - this - } - - /** Puts a list of resource objects with the specified `name` to this - * document builder. In case the same `name` already exists the resource - * objects will be overwritten. - */ - def resources(name: String, resObjs: List[ResourceObject[E, _]]): this.type = { - _embedded.put(name, Right(resObjs)) - this - } - - /** Puts an array of resource objects with the specified `name` to this - * document builder. In case the same `name` already exists the resource - * objects will be overwritten. - */ - def resources(name: String, resObjs: ResourceObject[E, _]*): this.type = - resources(name, resObjs.toList) - - def build(): ResourceObject[T, E] = ResourceObject(_links.toList, _embedded.toList, content) - -} diff --git a/hal/src/main/scala/org/http4s/rho/hal/ResourceObjectSerializer.scala b/hal/src/main/scala/org/http4s/rho/hal/ResourceObjectSerializer.scala deleted file mode 100644 index 0f7a97c77..000000000 --- a/hal/src/main/scala/org/http4s/rho/hal/ResourceObjectSerializer.scala +++ /dev/null @@ -1,91 +0,0 @@ -package org.http4s.rho -package hal - -import org.json4s._ -import org.json4s.Extraction._ - -import scala.collection.immutable.Seq - -object ResourceObjectSerializer { - - def serialize(r: ResourceObject[_, _])(implicit jsonFormats: Formats): JValue = { - val links = serializeLinks(r.links) - val embedded = serializeEmbedded(r.embedded) - val root: JValue = r.content match { - case Some(v) => - val content = decompose(v) - content match { - case JObject(fields) => - if (links.isDefined && embedded.isDefined) - JObject(links.get :: embedded.get :: fields) - else if (links.isDefined) - JObject(links.get :: fields) - else if (embedded.isDefined) - JObject(embedded.get :: fields) - else JObject(fields) - case v: JValue => v - } - case _ => - if (links.isDefined && embedded.isDefined) - JObject(links.get, embedded.get) - else if (links.isDefined) - JObject(links.get) - else if (embedded.isDefined) - JObject(embedded.get) - else JObject() - } - root - } - - private[hal] def serializeEmbedded(embedded: Embedded[_])(implicit - jsonFormats: Formats): Option[JField] = { - val embeddedAsFields = for { - fieldOption <- embedded.map(serializeEmbeddedDef) - field <- fieldOption - } yield field - if (embeddedAsFields.isEmpty) - None - else - Some(JField("_embedded", JObject(embeddedAsFields.toList))) - } - - private[hal] def serializeEmbeddedDef(embeddedDef: EmbeddedDef[_])(implicit - jsonFormats: Formats): Option[JField] = - serializeSingleOrMany(embeddedDef)(ResourceObjectSerializer.serialize) - - private[hal] def serializeLinkDef(linkDef: LinkObjectDef): Option[JField] = - serializeSingleOrMany(linkDef)(LinkObjectSerializer.serialize) - - private[hal] def serializeLinks(links: Links): Option[JField] = { - val linksAsFields = for { - fieldOption <- links.map(serializeLinkDef) - field <- fieldOption - } yield field - if (linksAsFields.isEmpty) - None - else - Some(JField("_links", JObject(linksAsFields.toList))) - } - - private[hal] def serializeSingleOrMany[T](entry: (String, Either[T, Seq[T]]))( - f: T => JValue): Option[JField] = entry._2 match { - case Left(v) => - Some(JField(entry._1, f(v))) - case Right(vs) => - val xs = vs.map(f) - Some(JField(entry._1, JArray(xs.toList))) - case _ => - None - } - -} - -class ResourceObjectSerializer - extends CustomSerializer[ResourceObject[_, _]](format => - ( - PartialFunction.empty, - { case r: ResourceObject[_, _] => - ResourceObjectSerializer.serialize(r)(format) - } - ) - ) diff --git a/hal/src/main/scala/org/http4s/rho/hal/package.scala b/hal/src/main/scala/org/http4s/rho/hal/package.scala deleted file mode 100644 index b4079c6d7..000000000 --- a/hal/src/main/scala/org/http4s/rho/hal/package.scala +++ /dev/null @@ -1,16 +0,0 @@ -package org.http4s.rho - -import scala.collection.immutable.Seq - -/** Describes Hypertext Application Language types and functions */ -package object hal { - - type EmbeddedDef[T] = (String, Either[ResourceObject[T, _], Seq[ResourceObject[T, _]]]) - - type Embedded[T] = List[EmbeddedDef[T]] - - type LinkObjectDef = (String, Either[LinkObject, Seq[LinkObject]]) - - type Links = List[LinkObjectDef] - -} diff --git a/hal/src/test/scala/org/http4s/rho/hal/HalDocBuilderSpec.scala b/hal/src/test/scala/org/http4s/rho/hal/HalDocBuilderSpec.scala deleted file mode 100644 index 60a0f4900..000000000 --- a/hal/src/test/scala/org/http4s/rho/hal/HalDocBuilderSpec.scala +++ /dev/null @@ -1,146 +0,0 @@ -package org.http4s.rho.hal - -import org.http4s.Uri -import org.http4s.UriTemplate -import org.http4s.UriTemplate._ -import org.specs2.mutable.Specification -import scala.collection.immutable.Seq - -object ResourceObjectBuilderSpec extends Specification { - - "ResourceObjectBuilder" should { - - "create empty ResourceObject" in { - new ResourceObjectBuilder().build() must equalTo(ResourceObject()) - } - - "create LinkObject from href" in { - new ResourceObjectBuilder() - .link("self", "/some/where") - .build() must equalTo( - ResourceObject(List("self" -> Left(LinkObject("/some/where", templated = None)))) - ) - } - - "create LinkObject from href and templated" in { - new ResourceObjectBuilder() - .link("self", "/some/where", Some(true)) - .build() must equalTo( - ResourceObject(List("self" -> Left(LinkObject("/some/where", templated = Some(true))))) - ) - } - - "create LinkObject from href and title" in { - new ResourceObjectBuilder() - .link("self", "/some/where", "A title") - .build() must equalTo( - ResourceObject(List("self" -> Left(LinkObject("/some/where", title = Some("A title"))))) - ) - } - - "create LinkObject from Uri" in { - new ResourceObjectBuilder() - .link("self", Uri(path = "/some/where")) - .build() must equalTo(ResourceObject(List("self" -> Left(LinkObject("/some/where"))))) - } - - "create LinkObject from UriTemplate" in { - new ResourceObjectBuilder() - .link("self", UriTemplate(path = List(PathElm("some"), PathExp("where")))) - .build() must equalTo( - ResourceObject(List("self" -> Left(LinkObject("/some{/where}", templated = Some(true))))) - ) - } - - val document = ResourceObject( - links = List( - "self" -> Left(LinkObject("/orders")), - "curies" -> Right( - Seq( - LinkObject( - name = Some("ea"), - href = "http://example.com/docs/rels/{rel}", - templated = Some(true) - ) - ) - ), - "next" -> Left(LinkObject("/orders?page=2")), - "ea:find" -> Left(LinkObject("/orders{?id}", templated = Some(true))), - "ea:admin" -> Right( - Seq( - LinkObject("/admins/2", title = Some("Fred")), - LinkObject("/admins/5", title = Some("Kate")) - ) - ) - ), - embedded = List( - "ea:order" -> - Right( - Seq( - ResourceObject[Map[String, Any], Nothing]( - List( - "self" -> Left(LinkObject("/orders/123")), - "ea:basket" -> Left(LinkObject("/baskets/98712")), - "ea:customer" -> Left(LinkObject("/customers/7809")) - ), - Nil, - Some(Map("total" -> 30.00, "currency" -> "USD", "status" -> "shipped")) - ), - ResourceObject[Map[String, Any], Nothing]( - List( - "self" -> Left(LinkObject("/orders/124")), - "ea:basket" -> Left(LinkObject("/baskets/97213")), - "ea:customer" -> Left(LinkObject("/customers/12369")) - ), - Nil, - Some(Map("total" -> 20.00, "currency" -> "USD", "status" -> "processing")) - ) - ) - ) - ), - content = Some(Map("currentlyProcessing" -> 14, "shippedToday" -> 20)) - ) - - // our data structure - val documentBuilder = - new ResourceObjectBuilder[Map[String, Int], Map[String, Any]]() - .link("self", "/orders") - .links( - "curies", - LinkObject( - name = Some("ea"), - href = "http://example.com/docs/rels/{rel}", - templated = Some(true) - ) - ) - .link("next", "/orders?page=2") - .link("ea:find", "/orders{?id}", Some(true)) - .links( - "ea:admin", - LinkObject("/admins/2", title = Some("Fred")), - LinkObject("/admins/5", title = Some("Kate")) - ) - .resources( - "ea:order", - new ResourceObjectBuilder[Map[String, Any], Nothing]() - .link("self", LinkObject("/orders/123")) - .link("ea:basket", LinkObject("/baskets/98712")) - .link("ea:customer", LinkObject("/customers/7809")) - .content(Map("total" -> 30.00, "currency" -> "USD", "status" -> "shipped")) - .build(), - new ResourceObjectBuilder[Map[String, Any], Nothing]() - .link("self", LinkObject("/orders/124")) - .link("ea:basket", LinkObject("/baskets/97213")) - .link("ea:customer", LinkObject("/customers/12369")) - .content(Map("total" -> 20.00, "currency" -> "USD", "status" -> "processing")) - .build() - ) - .content(Map("currentlyProcessing" -> 14, "shippedToday" -> 20)) - .build() - - "build a ResourceObject" in { - documentBuilder must be equalTo document - } - } - -} diff --git a/hal/src/test/scala/org/http4s/rho/hal/LinkObjectSpec.scala b/hal/src/test/scala/org/http4s/rho/hal/LinkObjectSpec.scala deleted file mode 100644 index 229141276..000000000 --- a/hal/src/test/scala/org/http4s/rho/hal/LinkObjectSpec.scala +++ /dev/null @@ -1,64 +0,0 @@ -package org.http4s.rho.hal - -import org.specs2.mutable.Specification - -class LinkObjectSpec extends Specification { - - "LinkObject" should { - "have a non-null href" in { - new LinkObject(null) must throwA[IllegalArgumentException] - } - "have a non-empty href" in { - new LinkObject("") must throwA[IllegalArgumentException] - } - "require only a href property" in { - new LinkObject("/link") must be equalTo new LinkObject("/link") - } - "have a templated property optionally" in { - new LinkObject("/link", templated = Some(true)).templated.get must beTrue - } - "have a type property optionally" in { - new LinkObject( - "/link", - `type` = Some("application/json") - ).`type`.get must be equalTo "application/json" - } - "have a deprecation property optionally" in { - new LinkObject( - "/link", - deprecation = Some("http://more/info/about/deprecated") - ).deprecation.get must be equalTo "http://more/info/about/deprecated" - } - "have a name property optionally" in { - new LinkObject("/link", name = Some("Max")).name.get must be equalTo "Max" - } - "have a profile property optionally" in { - new LinkObject("/link", profile = Some("profile1")).profile.get must be equalTo "profile1" - } - "have a title property optionally" in { - new LinkObject( - "/link", - title = Some("The case for hyperlinks in APIs") - ).title.get must be equalTo "The case for hyperlinks in APIs" - } - "have a hreflang property optionally" in { - new LinkObject( - "/link", - hreflang = Some("/href/lang") - ).hreflang.get must be equalTo "/href/lang" - } - "have empty optional properties per default" in { - new LinkObject("/link") must be equalTo new LinkObject( - "/link", - None, - None, - None, - None, - None, - None, - None - ) - } - } - -} diff --git a/hal/src/test/scala/org/http4s/rho/hal/ResourceObjectSpec.scala b/hal/src/test/scala/org/http4s/rho/hal/ResourceObjectSpec.scala deleted file mode 100644 index f8fb0cff8..000000000 --- a/hal/src/test/scala/org/http4s/rho/hal/ResourceObjectSpec.scala +++ /dev/null @@ -1,225 +0,0 @@ -package org.http4s.rho.hal - -import org.json4s._ -import org.json4s.JsonDSL._ -import org.json4s.jackson.JsonMethods._ -import org.json4s.jackson.Serialization._ -import org.specs2.mutable.Specification - -import scala.collection.immutable.Seq - -object ResourceObjectSpec extends Specification { - implicit val jsonFormats: Formats = DefaultFormats - - case class User(name: String, email: String) - val user1 = User("Max", "max@example.com") - - "ResourceObject serialization" should { - "serialize also if empty" in { - val resObj = ResourceObject() - ResourceObjectSerializer.serialize(resObj) must be equalTo - JObject() - } - "with one link only" in { - val resObj = ResourceObject( - List( - "self" -> - Left(LinkObject("/some/path")) - ) - ) - ResourceObjectSerializer.serialize(resObj) must be equalTo - ("_links" -> - ("self" -> - ("href", "/some/path"))) - } - "with two links only" in { - val resObj = ResourceObject( - List( - "self" -> - Right(Seq(LinkObject("/some/path/1"), LinkObject("/some/path/2"))) - ) - ) - ResourceObjectSerializer.serialize(resObj) must be equalTo - ("_links" -> - ("self" -> - List(("href", "/some/path/1"), ("href", "/some/path/2")))) - } - "with one embedded only" in { - val resObj = - ResourceObject(Nil, List("text" -> Left(ResourceObject(content = Some("some content"))))) - ResourceObjectSerializer.serialize(resObj) must be equalTo - ("_embedded" -> - ("text" -> - "some content")) - } - "with two embedded only" in { - val resObj = ResourceObject( - Nil, - List( - "texts" -> - Right( - Seq( - ResourceObject(content = Some("/some/path/1")), - ResourceObject(content = Some("/some/path/2")) - ) - ) - ) - ) - ResourceObjectSerializer.serialize(resObj) must be equalTo - ("_embedded" -> - ("texts" -> - List("/some/path/1", "/some/path/2"))) - } - "with non-primitive content without links and embedded definitions" in { - val resObj = ResourceObject(content = Some(user1)) - ResourceObjectSerializer.serialize(resObj) must be equalTo - (("name" -> "Max") ~ ("email" -> "max@example.com")) - } - "with non-primitive content, links and embedded definitions will be ignored" in { - val resObj = ResourceObject( - content = Some(user1), - links = List( - "self" -> - Left(LinkObject("/users/1")) - ), - embedded = List( - "groups" -> - Right( - Seq( - ResourceObject(content = Some("/groups/1")), - ResourceObject(content = Some("/groups/2")) - ) - ) - ) - ) - ResourceObjectSerializer.serialize(resObj) must be equalTo - (("_links" -> ("self" -> ("href", "/users/1"))) ~ - ("_embedded" -> ("groups" -> List("/groups/1", "/groups/2"))) ~ - ("name" -> "Max") ~ - ("email" -> "max@example.com")) - } - "with primitive content, links and embedded definitions will be ignored" in { - val resObj = ResourceObject( - content = Some("some content"), - links = List( - "self" -> - Left(LinkObject("/some/path")) - ), - embedded = List( - "text" -> - Left(ResourceObject(content = Some("some other content"))) - ) - ) - ResourceObjectSerializer.serialize(resObj) must be equalTo - "some content" - } - } - - "An example from HAL specification" should { - - // an example from http://stateless.co/hal_specification.html - val json = parse(""" -{ - "_links": { - "self": { "href": "/orders" }, - "curies": [{ "href": "http://example.com/docs/rels/{rel}", "templated": true, "name": "ea" }], - "next": { "href": "/orders?page=2" }, - "ea:find": { - "href": "/orders{?id}", - "templated": true - }, - "ea:admin": [{ - "href": "/admins/2", - "title": "Fred" - }, { - "href": "/admins/5", - "title": "Kate" - }] - }, - "_embedded": { - "ea:order": [{ - "_links": { - "self": { "href": "/orders/123" }, - "ea:basket": { "href": "/baskets/98712" }, - "ea:customer": { "href": "/customers/7809" } - }, - "total": 30.00, - "currency": "USD", - "status": "shipped" - }, { - "_links": { - "self": { "href": "/orders/124" }, - "ea:basket": { "href": "/baskets/97213" }, - "ea:customer": { "href": "/customers/12369" } - }, - "total": 20.00, - "currency": "USD", - "status": "processing" - }] - }, - "currentlyProcessing": 14, - "shippedToday": 20 -}""") - - // our data structure - val halDocument = ResourceObject( - links = List( - "self" -> Left(LinkObject("/orders")), - "curies" -> Right( - Seq( - LinkObject( - name = Some("ea"), - href = "http://example.com/docs/rels/{rel}", - templated = Some(true) - ) - ) - ), - "next" -> Left(LinkObject("/orders?page=2")), - "ea:find" -> Left(LinkObject("/orders{?id}", templated = Some(true))), - "ea:admin" -> Right( - Seq( - LinkObject("/admins/2", title = Some("Fred")), - LinkObject("/admins/5", title = Some("Kate")) - ) - ) - ), - embedded = List( - "ea:order" -> - Right( - Seq( - ResourceObject[Map[String, Any], Nothing]( - List( - "self" -> Left(LinkObject("/orders/123")), - "ea:basket" -> Left(LinkObject("/baskets/98712")), - "ea:customer" -> Left(LinkObject("/customers/7809")) - ), - Nil, - Some(Map("total" -> 30.00, "currency" -> "USD", "status" -> "shipped")) - ), - ResourceObject[Map[String, Any], Nothing]( - List( - "self" -> Left(LinkObject("/orders/124")), - "ea:basket" -> Left(LinkObject("/baskets/97213")), - "ea:customer" -> Left(LinkObject("/customers/12369")) - ), - Nil, - Some(Map("total" -> 20.00, "currency" -> "USD", "status" -> "processing")) - ) - ) - ) - ), - content = Some(Map("currentlyProcessing" -> 14, "shippedToday" -> 20)) - ) - - "be equal as pretty-printed JSON string when comparing to our data structure" in { - writePretty(ResourceObjectSerializer.serialize(halDocument)) must be equalTo - writePretty(json) - } - - "be equal as JSON tree when comparing to our serialized data structure" in { - ResourceObjectSerializer.serialize(halDocument) must be equalTo - json - } - } - -} From 0037daf5cde7decaa76093b7a596ce1c6a5a1cf6 Mon Sep 17 00:00:00 2001 From: Darren Gibson Date: Thu, 3 Jun 2021 18:16:17 -0500 Subject: [PATCH 2/6] Initial work on Http4s 0.22 --- .../scala/org/http4s/rho/AuthedContext.scala | 2 +- .../main/scala/org/http4s/rho/Result.scala | 2 +- .../scala/org/http4s/rho/UriConvertible.scala | 10 +- .../scala/org/http4s/rho/bits/PathAST.scala | 6 +- .../scala/org/http4s/rho/bits/PathTree.scala | 67 +++---- .../org/http4s/rho/bits/ResultMatcher.scala | 2 +- .../org/http4s/rho/bits/UriConverter.scala | 4 +- .../test/scala/org/http4s/rho/ApiTest.scala | 7 +- .../org/http4s/rho/AuthedContextSpec.scala | 6 +- .../org/http4s/rho/CodecRouterSpec.scala | 5 +- .../org/http4s/rho/CompileRoutesSpec.scala | 16 +- .../scala/org/http4s/rho/ResultSpec.scala | 2 +- .../scala/org/http4s/rho/RhoRoutesSpec.scala | 43 +++-- .../org/http4s/rho/UriConvertibleSpec.scala | 12 +- .../org/http4s/rho/bits/PathTreeSpec.scala | 169 ++++++++++-------- .../http4s/rho/bits/ResultMatcherSpec.scala | 8 +- .../com/http4s/rho/swagger/demo/Main.scala | 2 +- .../http4s/rho/swagger/demo/MyRoutes.scala | 6 +- project/Dependencies.scala | 4 +- .../rho/swagger/SwaggerModelsBuilder.scala | 12 +- .../swagger/SwaggerModelsBuilderSpec.scala | 4 +- .../rho/swagger/SwaggerSupportSpec.scala | 18 +- 22 files changed, 215 insertions(+), 192 deletions(-) diff --git a/core/src/main/scala/org/http4s/rho/AuthedContext.scala b/core/src/main/scala/org/http4s/rho/AuthedContext.scala index 786f3b822..029dd318d 100644 --- a/core/src/main/scala/org/http4s/rho/AuthedContext.scala +++ b/core/src/main/scala/org/http4s/rho/AuthedContext.scala @@ -5,8 +5,8 @@ import cats.Monad import cats.data.{Kleisli, OptionT} import shapeless.{::, HNil} import org.http4s.rho.bits.{FailureResponseOps, SuccessResponse, TypedHeader} -import _root_.io.chrisdavenport.vault._ import cats.effect._ +import org.typelevel.vault.Key /** The [[AuthedContext]] provides a convenient way to define a RhoRoutes * which works with http4s authentication middleware. diff --git a/core/src/main/scala/org/http4s/rho/Result.scala b/core/src/main/scala/org/http4s/rho/Result.scala index 076fce8b9..6b21f1d3f 100644 --- a/core/src/main/scala/org/http4s/rho/Result.scala +++ b/core/src/main/scala/org/http4s/rho/Result.scala @@ -3,7 +3,7 @@ package rho import cats._ import org.http4s.headers.`Content-Type` -import _root_.io.chrisdavenport.vault._ +import org.typelevel.vault._ /** A helper for capturing the result types and status codes from routes */ sealed case class Result[ diff --git a/core/src/main/scala/org/http4s/rho/UriConvertible.scala b/core/src/main/scala/org/http4s/rho/UriConvertible.scala index 846f73365..001f9bc95 100644 --- a/core/src/main/scala/org/http4s/rho/UriConvertible.scala +++ b/core/src/main/scala/org/http4s/rho/UriConvertible.scala @@ -38,12 +38,8 @@ object UriConvertible { private[rho] def addPathInfo[F[_]](request: Request[F], tpl: UriTemplate): UriTemplate = { val caret = request.attributes.lookup(Request.Keys.PathInfoCaret).getOrElse(0) if (caret == 0) tpl - else if (caret == 1 && request.scriptName == "/") tpl - else tpl.copy(path = UriTemplate.PathElm(removeSlash(request.scriptName)) :: tpl.path) + else if (caret == 1 && request.scriptName.absolute) tpl + else + tpl.copy(path = UriTemplate.PathElm(request.scriptName.toRelative.renderString) :: tpl.path) } - - private[rho] def removeSlash(path: String): String = - if (path.startsWith("/")) path.substring(1) - else path - } diff --git a/core/src/main/scala/org/http4s/rho/bits/PathAST.scala b/core/src/main/scala/org/http4s/rho/bits/PathAST.scala index 0b13c0a19..b0af4762b 100644 --- a/core/src/main/scala/org/http4s/rho/bits/PathAST.scala +++ b/core/src/main/scala/org/http4s/rho/bits/PathAST.scala @@ -105,7 +105,11 @@ object PathAST { case class PathOr(p1: PathRule, p2: PathRule) extends PathRoute - case class PathMatch(s: String) extends PathOperation + case class PathMatch(s: Uri.Path.Segment) extends PathOperation + object PathMatch { + def apply(s: String): PathMatch = PathMatch(Uri.Path.Segment(s)) + val empty: PathMatch = PathMatch(Uri.Path.Segment.empty) + } case class PathCapture[F[_]]( name: String, 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 f9aef41ad..2c1044998 100644 --- a/core/src/main/scala/org/http4s/rho/bits/PathTree.scala +++ b/core/src/main/scala/org/http4s/rho/bits/PathTree.scala @@ -9,37 +9,22 @@ import org.log4s.getLogger import shapeless.{HList, HNil} import scala.annotation.tailrec -import scala.collection.mutable.ListBuffer import scala.util.control.NonFatal object PathTree { def apply[F[_]](): PathTreeOps[F]#PathTree = { val ops = new PathTreeOps[F] {} - new ops.PathTree(ops.MatchNode("")) + new ops.PathTree(ops.MatchNode(Uri.Path.Segment.empty)) } - def splitPath(path: String): List[String] = { - val buff = new ListBuffer[String] - val len = path.length - - @tailrec - def go(i: Int, begin: Int): Unit = - if (i < len) { - if (path.charAt(i) == '/') { - if (i > begin) buff += org.http4s.Uri.decode(path.substring(begin, i)) - go(i + 1, i + 1) - } else go(i + 1, begin) - } else { - buff += org.http4s.Uri.decode(path.substring(begin, i)) - } - - val i = if (path.nonEmpty && path.charAt(0) == '/') 1 else 0 - go(i, i) - - buff.result() + def splitPath(path: Uri.Path): List[Uri.Path.Segment] = { + val fixEmpty = + if (path.isEmpty) List(Uri.Path.Segment.empty) + else path.segments.filterNot(_.isEmpty).toList + if (path.endsWithSlash) fixEmpty :+ Uri.Path.Segment.empty + else fixEmpty } - } private[rho] trait PathTreeOps[F[_]] extends RuleExecutor[F] { @@ -80,9 +65,6 @@ private[rho] trait PathTreeOps[F[_]] extends RuleExecutor[F] { type Action = ResultResponse[F, F[Response[F]]] - /** Generates a list of tokens that represent the path */ - private def keyToPath(key: Request[F]): List[String] = PathTree.splitPath(key.pathInfo) - def makeLeaf[T <: HList](route: RhoRoute[F, T])(implicit F: Monad[F]): Leaf = route.router match { case Router(_, _, rules) => @@ -154,7 +136,7 @@ private[rho] trait PathTreeOps[F[_]] extends RuleExecutor[F] { sealed trait Node[Self <: Node[Self]] extends ResponseGeneratorInstances[F] { - def matches: Map[String, MatchNode] + def matches: Map[Uri.Path.Segment, MatchNode] def captures: List[CaptureNode] @@ -163,7 +145,7 @@ private[rho] trait PathTreeOps[F[_]] extends RuleExecutor[F] { def end: Map[Method, Leaf] def clone( - matches: Map[String, MatchNode], + matches: Map[Uri.Path.Segment, MatchNode], captures: List[CaptureNode], variadic: Map[Method, Leaf], end: Map[Method, Leaf]): Self @@ -184,7 +166,7 @@ private[rho] trait PathTreeOps[F[_]] extends RuleExecutor[F] { case MetaCons(r, _) => append(r :: t, method, action) // discard metadata - case PathMatch("") if !t.isEmpty => + case PathMatch.empty if !t.isEmpty => append(t, method, action) // "" is a NOOP in the middle of a path // the rest of the types need to rewrite a node @@ -222,20 +204,25 @@ private[rho] trait PathTreeOps[F[_]] extends RuleExecutor[F] { */ final def walkTree(method: Method, req: Request[F])(implicit F: Monad[F]): RouteResult[F, F[Response[F]]] = { - val path = keyToPath(req) + val path = PathTree.splitPath(req.pathInfo) walk(method, req, path, HNil) } // This should scan all forward paths. - final protected def walk(method: Method, req: Request[F], path: List[String], stack: HList)( - implicit F: Monad[F]): RouteResult[F, F[Response[F]]] = { + final protected def walk( + method: Method, + req: Request[F], + path: List[Uri.Path.Segment], + stack: HList)(implicit F: Monad[F]): RouteResult[F, F[Response[F]]] = { def tryVariadic(result: RouteResult[F, F[Response[F]]]): RouteResult[F, F[Response[F]]] = variadic.get(method) match { case None => result case Some(l) => l.attempt(req, path :: stack) } - def walkHeadTail(h: String, t: List[String]): RouteResult[F, F[Response[F]]] = { + def walkHeadTail( + h: Uri.Path.Segment, + t: List[Uri.Path.Segment]): RouteResult[F, F[Response[F]]] = { val exact: RouteResult[F, F[Response[F]]] = matches.get(h) match { case Some(n) => n.walk(method, req, t, stack) case None => noMatch @@ -247,7 +234,7 @@ private[rho] trait PathTreeOps[F[_]] extends RuleExecutor[F] { error: RouteResult[F, F[Response[F]]]): RouteResult[F, F[Response[F]]] = children match { case (c @ CaptureNode(p, _, _, _, _)) :: ns => - p.parse(h) match { + p.parse(h.decoded()) match { //TODO: Figure out how to inject char sets etc... case SuccessResponse(r) => val n = c.walk(method, req, t, r :: stack) if (n.isSuccess) n @@ -290,15 +277,15 @@ private[rho] trait PathTreeOps[F[_]] extends RuleExecutor[F] { } case class MatchNode( - name: String, - matches: Map[String, MatchNode] = Map.empty[String, MatchNode], + name: Uri.Path.Segment, + matches: Map[Uri.Path.Segment, MatchNode] = Map.empty[Uri.Path.Segment, MatchNode], captures: List[CaptureNode] = Nil, variadic: Map[Method, Leaf] = Map.empty[Method, Leaf], end: Map[Method, Leaf] = Map.empty[Method, Leaf]) extends Node[MatchNode] { override def clone( - matches: Map[String, MatchNode], + matches: Map[Uri.Path.Segment, MatchNode], captures: List[CaptureNode], variadic: Map[Method, Leaf], end: Map[Method, Leaf]): MatchNode = @@ -321,14 +308,14 @@ private[rho] trait PathTreeOps[F[_]] extends RuleExecutor[F] { case class CaptureNode( parser: StringParser[F, _], - matches: Map[String, MatchNode] = Map.empty[String, MatchNode], + matches: Map[Uri.Path.Segment, MatchNode] = Map.empty[Uri.Path.Segment, MatchNode], captures: List[CaptureNode] = List.empty[CaptureNode], variadic: Map[Method, Leaf] = Map.empty[Method, Leaf], end: Map[Method, Leaf] = Map.empty[Method, Leaf]) extends Node[CaptureNode] { override def clone( - matches: Map[String, MatchNode], + matches: Map[Uri.Path.Segment, MatchNode], captures: List[CaptureNode], variadic: Map[Method, Leaf], end: Map[Method, Leaf]): CaptureNode = @@ -359,8 +346,8 @@ private[rho] trait PathTreeOps[F[_]] extends RuleExecutor[F] { } private def mergeMatches( - m1: Map[String, MatchNode], - m2: Map[String, MatchNode]): Map[String, MatchNode] = + m1: Map[Uri.Path.Segment, MatchNode], + m2: Map[Uri.Path.Segment, MatchNode]): Map[Uri.Path.Segment, MatchNode] = mergeMaps(m1, m2)(_ merge _) private def mergeLeaves(l1: Map[Method, Leaf], l2: Map[Method, Leaf]): Map[Method, Leaf] = diff --git a/core/src/main/scala/org/http4s/rho/bits/ResultMatcher.scala b/core/src/main/scala/org/http4s/rho/bits/ResultMatcher.scala index 02aeef0b9..821dcc6d7 100644 --- a/core/src/main/scala/org/http4s/rho/bits/ResultMatcher.scala +++ b/core/src/main/scala/org/http4s/rho/bits/ResultMatcher.scala @@ -463,7 +463,7 @@ trait ResultMatchers[F[_]] extends ResultMatcherMidPrioInstances[F] { override def conv(req: Request[F], r: Option[R])(implicit F: Monad[F]): F[Response[F]] = r match { case Some(res) => Ok.pure(res) - case None => NotFound.pure(req.uri.path) + case None => NotFound.pure(req.uri.path.renderString) } } diff --git a/core/src/main/scala/org/http4s/rho/bits/UriConverter.scala b/core/src/main/scala/org/http4s/rho/bits/UriConverter.scala index a33ff3448..d7565288c 100644 --- a/core/src/main/scala/org/http4s/rho/bits/UriConverter.scala +++ b/core/src/main/scala/org/http4s/rho/bits/UriConverter.scala @@ -20,8 +20,8 @@ object UriConverter { case Nil => acc case PathAnd(a, b) :: rs => go(a :: b :: rs, acc) case PathOr(a, _) :: rs => go(a :: rs, acc) // we decided to take the first root - case PathMatch("") :: rs => go(rs, acc) - case PathMatch(s) :: rs => go(rs, PathElm(s) :: acc) + case PathMatch.empty :: rs => go(rs, acc) + case PathMatch(s) :: rs => go(rs, PathElm(s.encoded) :: acc) case PathCapture(id, _, _, _) :: rs => go(rs, PathExp(id) :: acc) case CaptureTail :: rs => go(rs, acc) case MetaCons(p, _) :: rs => go(p :: rs, acc) diff --git a/core/src/test/scala/org/http4s/rho/ApiTest.scala b/core/src/test/scala/org/http4s/rho/ApiTest.scala index 646ae1db2..f4d3f4ef8 100644 --- a/core/src/test/scala/org/http4s/rho/ApiTest.scala +++ b/core/src/test/scala/org/http4s/rho/ApiTest.scala @@ -13,6 +13,7 @@ import org.http4s.Uri.uri import org.specs2.matcher.MatchResult import org.specs2.mutable._ import shapeless.{HList, HNil} +import scala.util.control.NoStackTrace class ApiTest extends Specification { @@ -38,7 +39,9 @@ class ApiTest extends Specification { existsAnd(headers.`Content-Length`)(h => h.length != 0) val RequireThrowException = - existsAnd(headers.`Content-Length`)(_ => throw new RuntimeException("this could happen")) + existsAnd(headers.`Content-Length`)(_ => + throw new RuntimeException("this could happen") with NoStackTrace + ) def fetchETag(p: IO[Response[IO]]): ETag = { val resp = p.unsafeRunSync() @@ -181,7 +184,7 @@ class ApiTest extends Specification { .resp .status - val c3 = captureMapR(headers.`Access-Control-Allow-Credentials`, Some(r2))(_ => ???) + val c3 = captureMapR(headers.Accept, Some(r2))(_ => ???) val v2 = ruleExecutor.runRequestRules(c3.rule, req) v2 must beAnInstanceOf[FailureResponse[IO]] v2.asInstanceOf[FailureResponse[IO]].toResponse.unsafeRunSync().status must_== r2 diff --git a/core/src/test/scala/org/http4s/rho/AuthedContextSpec.scala b/core/src/test/scala/org/http4s/rho/AuthedContextSpec.scala index 4bab3380f..cf38f0a5e 100644 --- a/core/src/test/scala/org/http4s/rho/AuthedContextSpec.scala +++ b/core/src/test/scala/org/http4s/rho/AuthedContextSpec.scala @@ -50,7 +50,7 @@ class AuthedContextSpec extends Specification { "AuthedContext execution" should { "Be able to have access to authInfo" in { - val request = Request[IO](Method.GET, Uri(path = "/")) + val request = Request[IO](Method.GET, Uri(path = Uri.Path.fromString("/"))) val resp = routes.run(request).value.unsafeRunSync().getOrElse(Response.notFound) if (resp.status == Status.Ok) { val body = @@ -60,7 +60,7 @@ class AuthedContextSpec extends Specification { } "Does not prevent route from being executed without authentication" in { - val request = Request[IO](Method.GET, Uri(path = "/public/public")) + val request = Request[IO](Method.GET, Uri(path = Uri.Path.fromString("/public/public"))) val resp = routes.run(request).value.unsafeRunSync().getOrElse(Response.notFound) if (resp.status == Status.Ok) { val body = @@ -70,7 +70,7 @@ class AuthedContextSpec extends Specification { } "Does not prevent route from being executed without authentication, but allows to extract it" in { - val request = Request[IO](Method.GET, Uri(path = "/private/private")) + val request = Request[IO](Method.GET, Uri(path = Uri.Path.fromString("/private/private"))) val resp = routes.run(request).value.unsafeRunSync().getOrElse(Response.notFound) if (resp.status == Status.Ok) { val body = diff --git a/core/src/test/scala/org/http4s/rho/CodecRouterSpec.scala b/core/src/test/scala/org/http4s/rho/CodecRouterSpec.scala index bad068c34..c833e4094 100644 --- a/core/src/test/scala/org/http4s/rho/CodecRouterSpec.scala +++ b/core/src/test/scala/org/http4s/rho/CodecRouterSpec.scala @@ -6,6 +6,7 @@ import fs2.Stream import org.specs2.mutable.Specification import scala.collection.compat.immutable.ArraySeq +import org.http4s.Uri.Path class CodecRouterSpec extends Specification { @@ -27,7 +28,7 @@ class CodecRouterSpec extends Specification { 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 req = Request[IO](Method.POST, Uri(path = Path.fromString("/foo")), headers = h, body = b) val result = routes(req).value.unsafeRunSync().getOrElse(Response.notFound) val (bb, s) = bodyAndStatus(result) @@ -38,7 +39,7 @@ class CodecRouterSpec extends Specification { "Fail on invalid body" in { 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) + val req = Request[IO](Method.POST, Uri(path = Path.fromString("/form")), headers = h, body = b) routes(req).value.unsafeRunSync().map(_.status) must be some Status.BadRequest } diff --git a/core/src/test/scala/org/http4s/rho/CompileRoutesSpec.scala b/core/src/test/scala/org/http4s/rho/CompileRoutesSpec.scala index e3ebca939..77ffb1a96 100644 --- a/core/src/test/scala/org/http4s/rho/CompileRoutesSpec.scala +++ b/core/src/test/scala/org/http4s/rho/CompileRoutesSpec.scala @@ -5,6 +5,7 @@ import org.http4s.rho.bits.MethodAliases._ import org.http4s.rho.io._ import org.http4s.{Method, Request, Uri} import org.specs2.mutable.Specification +import org.http4s.Uri.Path class CompileRoutesSpec extends Specification { @@ -19,7 +20,7 @@ class CompileRoutesSpec extends Specification { val c = RoutesBuilder[IO]() getFoo(c) - "GetFoo" === RRunner(c.toRoutes()).checkOk(Request(uri = Uri(path = "/hello"))) + "GetFoo" === RRunner(c.toRoutes()).checkOk(Request(uri=Uri(path=Path.fromString("/hello")))) } "Build multiple routes" in { @@ -27,9 +28,8 @@ class CompileRoutesSpec extends Specification { getFoo(c) putFoo(c) - "GetFoo" === RRunner(c.toRoutes()).checkOk(Request(uri = Uri(path = "/hello"))) - "PutFoo" === RRunner(c.toRoutes()) - .checkOk(Request(method = Method.PUT, uri = Uri(path = "/hello"))) + "GetFoo" === RRunner(c.toRoutes()).checkOk(Request(uri=Uri(path=Path.fromString("/hello")))) + "PutFoo" === RRunner(c.toRoutes()).checkOk(Request(method = Method.PUT, uri=Uri(path=Path.fromString("/hello")))) } "Make routes from a collection of RhoRoutes" in { @@ -39,8 +39,8 @@ class CompileRoutesSpec extends Specification { (PUT / "hello" |>> "PutFoo") :: Nil val srvc = CompileRoutes.foldRoutes[IO](routes) - "GetFoo" === RRunner(srvc).checkOk(Request(uri = Uri(path = "/hello"))) - "PutFoo" === RRunner(srvc).checkOk(Request(method = Method.PUT, uri = Uri(path = "/hello"))) + "GetFoo" === RRunner(srvc).checkOk(Request(uri=Uri(path=Path.fromString("/hello")))) + "PutFoo" === RRunner(srvc).checkOk(Request(method = Method.PUT, uri=Uri(path=Path.fromString("/hello")))) } "Concatenate correctly" in { @@ -48,8 +48,8 @@ class CompileRoutesSpec extends Specification { val c2 = RoutesBuilder[IO](); putFoo(c2) val srvc = c1.append(c2.routes()).toRoutes() - "GetFoo" === RRunner(srvc).checkOk(Request(uri = Uri(path = "/hello"))) - "PutFoo" === RRunner(srvc).checkOk(Request(method = Method.PUT, uri = Uri(path = "/hello"))) + "GetFoo" === RRunner(srvc).checkOk(Request(uri=Uri(path=Path.fromString("/hello")))) + "PutFoo" === RRunner(srvc).checkOk(Request(method = Method.PUT, uri=Uri(path=Path.fromString("/hello")))) } } diff --git a/core/src/test/scala/org/http4s/rho/ResultSpec.scala b/core/src/test/scala/org/http4s/rho/ResultSpec.scala index 1d3e47f36..1e22862ae 100644 --- a/core/src/test/scala/org/http4s/rho/ResultSpec.scala +++ b/core/src/test/scala/org/http4s/rho/ResultSpec.scala @@ -1,6 +1,6 @@ package org.http4s.rho -import _root_.io.chrisdavenport.vault._ +import org.typelevel.vault._ import cats.effect._ import org.http4s.headers._ import org.http4s.rho.io._ diff --git a/core/src/test/scala/org/http4s/rho/RhoRoutesSpec.scala b/core/src/test/scala/org/http4s/rho/RhoRoutesSpec.scala index 60c85c085..9e293c12a 100644 --- a/core/src/test/scala/org/http4s/rho/RhoRoutesSpec.scala +++ b/core/src/test/scala/org/http4s/rho/RhoRoutesSpec.scala @@ -11,6 +11,9 @@ import org.http4s.headers.{`Content-Length`, `Content-Type`} import org.http4s.rho.io._ import org.http4s.Uri.uri import org.specs2.mutable.Specification +import org.http4s.Uri.Path +import org.typelevel.ci.CIString +import scala.util.control.NoStackTrace class RhoRoutesSpec extends Specification with RequestRunner { def construct(method: Method, s: String, h: Header*): Request[IO] = @@ -95,7 +98,7 @@ class RhoRoutesSpec extends Specification with RequestRunner { "RhoRoutes execution" should { "Handle definition without a path, which points to '/'" in { - val request = Request[IO](Method.GET, Uri(path = "/")) + val request = Request[IO](Method.GET, Uri(path = Path.fromString("/"))) checkOk(request) should_== "just root with parameter 'foo=bar'" } @@ -109,7 +112,7 @@ class RhoRoutesSpec extends Specification with RequestRunner { val resp = httpRoutes(request).value.unsafeRunSync().getOrElse(Response.notFound) resp.status must_== Status.MethodNotAllowed - resp.headers.get("Allow".ci) must beSome(Header.Raw("Allow".ci, "GET")) + resp.headers.get(CIString("Allow")) must beSome(Header.Raw(CIString("Allow"), "GET")) } "Yield `MethodNotAllowed` when invalid method used" in { @@ -124,10 +127,10 @@ class RhoRoutesSpec extends Specification with RequestRunner { GET / "" / "foo" |>> Ok("bar") }.toRoutes() - val req1 = Request[IO](Method.GET, Uri(path = "/foo")) + val req1 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo"))) getBody(service(req1).value.unsafeRunSync().getOrElse(Response.notFound).body) should_== "bar" - val req2 = Request[IO](Method.GET, Uri(path = "//foo")) + val req2 = Request[IO](Method.GET, Uri(path = Path.fromString("//foo"))) getBody(service(req2).value.unsafeRunSync().getOrElse(Response.notFound).body) should_== "bar" } @@ -253,21 +256,21 @@ class RhoRoutesSpec extends Specification with RequestRunner { } "Level one path definition to /some" in { - val req1 = Request[IO](Method.GET, Uri(path = "/some")) + val req1 = Request[IO](Method.GET, Uri(path = Path.fromString("/some"))) checkOk(req1) should_== "root to some" } "Execute a directly provided Task every invocation" in { - val req = Request[IO](Method.GET, Uri(path = "directTask")) + val req = Request[IO](Method.GET, Uri(path = Path.fromString("directTask"))) checkOk(req) should_== "0" checkOk(req) should_== "1" } "Interpret uris ending in '/' differently than those without" in { - val req1 = Request[IO](Method.GET, Uri(path = "terminal/")) + val req1 = Request[IO](Method.GET, Uri(path = Path.fromString("terminal/"))) checkOk(req1) should_== "terminal/" - val req2 = Request[IO](Method.GET, Uri(path = "terminal")) + val req2 = Request[IO](Method.GET, Uri(path = Path.fromString("terminal"))) checkOk(req2) should_== "terminal" } @@ -279,17 +282,17 @@ class RhoRoutesSpec extends Specification with RequestRunner { GET / "foo" |>> Ok("none") }.toRoutes() - val req1 = Request[IO](Method.GET, Uri(path = "/foo").+?("bar", "0")) + val req1 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo")).+?("bar", "0")) getBody( service(req1).value.unsafeRunSync().getOrElse(Response.notFound).body ) must_== "Int: 0" - val req2 = Request[IO](Method.GET, Uri(path = "/foo").+?("bar", "s")) + val req2 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo")).+?("bar", "s")) getBody( service(req2).value.unsafeRunSync().getOrElse(Response.notFound).body ) must_== "String: s" - val req3 = Request[IO](Method.GET, Uri(path = "/foo")) + val req3 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo"))) getBody(service(req3).value.unsafeRunSync().getOrElse(Response.notFound).body) must_== "none" } @@ -299,12 +302,12 @@ class RhoRoutesSpec extends Specification with RequestRunner { GET / "foo" +? param[Int]("bar") |>> { i: Int => Ok(s"Int: $i") } }.toRoutes() - val req1 = Request[IO](Method.GET, Uri(path = "/foo").+?("bar", "0")) + val req1 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo")).+?("bar", "0")) getBody( service(req1).value.unsafeRunSync().getOrElse(Response.notFound).body ) must_== "String: 0" - val req2 = Request[IO](Method.GET, Uri(path = "/foo").+?("bar", "s")) + val req2 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo")).+?("bar", "s")) getBody( service(req2).value.unsafeRunSync().getOrElse(Response.notFound).body ) must_== "String: s" @@ -319,12 +322,12 @@ class RhoRoutesSpec extends Specification with RequestRunner { GET / "foo" |>> Ok(s"failure") }.toRoutes() - val req1 = Request[IO](Method.GET, Uri(path = "/foo").+?("bar", "s")) + val req1 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo")).+?("bar", "s")) getBody( service(req1).value.unsafeRunSync().getOrElse(Response.notFound).body ) must_== "String: s" - val req2 = Request[IO](Method.GET, Uri(path = "/foo")) + val req2 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo"))) getBody(service(req2).value.unsafeRunSync().getOrElse(Response.notFound).body) must_== "none" } @@ -354,9 +357,11 @@ class RhoRoutesSpec extends Specification with RequestRunner { //////////////////////////////////////////////////// "Handle errors in the route actions" in { val service = new RhoRoutes[IO] { - GET / "error" |>> { () => throw new Error("an error"); Ok("Wont get here...") } + GET / "error" |>> { () => + throw new Error("an error") with NoStackTrace; Ok("Wont get here...") + } }.toRoutes() - val req = Request[IO](Method.GET, Uri(path = "/error")) + val req = Request[IO](Method.GET, Uri(path = Path.fromString("/error"))) service(req).value.unsafeRunSync().getOrElse(Response.notFound).status must equalTo( Status.InternalServerError @@ -365,7 +370,7 @@ class RhoRoutesSpec extends Specification with RequestRunner { "give a None for missing route" in { val service = new RhoRoutes[IO] {}.toRoutes() - val req = Request[IO](Method.GET, Uri(path = "/missing")) + val req = Request[IO](Method.GET, Uri(path = Path.fromString("/missing"))) service(req).value.unsafeRunSync().getOrElse(Response.notFound).status must_== Status.NotFound } } @@ -399,7 +404,7 @@ class RhoRoutesSpec extends Specification with RequestRunner { val routes2: HttpRoutes[IO] = ("foo" /: routes1).toRoutes() - val req1 = Request[IO](uri = Uri(path = "/foo/bar")) + val req1 = Request[IO](uri = Uri(path = Path.fromString("/foo/bar"))) getBody(routes2(req1).value.unsafeRunSync().getOrElse(Response.notFound).body) === "bar" } } diff --git a/core/src/test/scala/org/http4s/rho/UriConvertibleSpec.scala b/core/src/test/scala/org/http4s/rho/UriConvertibleSpec.scala index 42a90eb5e..38274c455 100644 --- a/core/src/test/scala/org/http4s/rho/UriConvertibleSpec.scala +++ b/core/src/test/scala/org/http4s/rho/UriConvertibleSpec.scala @@ -1,7 +1,7 @@ package org.http4s package rho -import _root_.io.chrisdavenport.vault._ +import org.typelevel.vault._ import org.specs2.mutable.Specification import UriTemplate._ import cats.effect.IO @@ -14,7 +14,7 @@ object UriConvertibleSpec extends Specification { "UriConvertible.respectPathInfo" should { "respect if URI template is available" in { val request = Request[IO]( - uri = Uri(path = "/some"), + uri = Uri(path = Uri.Path.fromString("/some")), attributes = Vault.empty.insert(Request.Keys.PathInfoCaret, 5) ) val path = List(PathElm("here")) @@ -35,7 +35,7 @@ object UriConvertibleSpec extends Specification { "UriConvertible.addPathInfo" should { "keep the path if PathInfoCaret is not available" in { - val request = Request[IO](uri = Uri(path = "/some")) + val request = Request[IO](uri = Uri(path = Uri.Path.fromString("/some"))) val path = List(PathElm("here")) val query = List(ParamVarExp("ref", "path")) val tpl = UriTemplate(path = path, query = query) @@ -44,7 +44,7 @@ object UriConvertibleSpec extends Specification { } "keep the path if PathInfoCaret is 0" in { val request = Request[IO]( - uri = Uri(path = "/some"), + uri = Uri(path = Uri.Path.fromString("/some")), attributes = Vault.empty.insert(Request.Keys.PathInfoCaret, 0) ) val path = List(PathElm("here")) @@ -55,7 +55,7 @@ object UriConvertibleSpec extends Specification { } "keep the path if PathInfoCaret is 1" in { val request = Request[IO]( - uri = Uri(path = "/some"), + uri = Uri(path = Uri.Path.fromString("/some")), attributes = Vault.empty.insert(Request.Keys.PathInfoCaret, 1) ) val path = List(PathElm("here")) @@ -66,7 +66,7 @@ object UriConvertibleSpec extends Specification { } "manipulate the path if PathInfoCaret greater than 1" in { val request = Request[IO]( - uri = Uri(path = "/some"), + uri = Uri(path = Uri.Path.fromString("/some")), attributes = Vault.empty.insert(Request.Keys.PathInfoCaret, 5) ) val path = List(PathElm("here")) 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 60de9b673..547c97b66 100644 --- a/core/src/test/scala/org/http4s/rho/bits/PathTreeSpec.scala +++ b/core/src/test/scala/org/http4s/rho/bits/PathTreeSpec.scala @@ -9,39 +9,44 @@ import org.specs2.mutable.Specification import org.http4s.Uri.uri import org.http4s.server.middleware.TranslateUri import org.http4s.server.Router +import org.http4s.Uri.Path class PathTreeSpec extends Specification { - import PathTree._ object pathTree extends PathTreeOps[IO] import pathTree._ - "splitPath" should { - "handle an empty string" in { - splitPath("") must_== List("") - } - "handle '/' only" in { - splitPath("/") must_== List("") - } - "handle /test/path" in { - splitPath("/test/path") must_== List("test", "path") - } - "Interpret '//' as '/'" in { - splitPath("/test//path") must_== List("test", "path") - } - "Not swallow a trailing '/'" in { - splitPath("/test/") must_== List("test", "") - } - "Url Decode Path segments" in { - splitPath("/test/%23%24%25!%40/foobar") must_== List("test", "#$%!@", "foobar") - } - } + def matchNodeFromString( + name: String, + matches: Map[String, MatchNode] = Map.empty[String, MatchNode], + captures: List[CaptureNode] = Nil, + variadic: Map[Method, Leaf] = Map.empty[Method, Leaf], + end: Map[Method, Leaf] = Map.empty[Method, Leaf]) = MatchNode( + Uri.Path.Segment(name), + matches.map { case (k, v) => Uri.Path.Segment(k) -> v }, + captures, + variadic, + end + ) + def captureNodeFromString( + parser: StringParser[IO, _], + matches: Map[String, MatchNode] = Map.empty[String, MatchNode], + captures: List[CaptureNode] = List.empty[CaptureNode], + variadic: Map[Method, Leaf] = Map.empty[Method, Leaf], + end: Map[Method, Leaf] = Map.empty[Method, Leaf] + ) = CaptureNode( + parser = parser, + matches = matches.map { case (k, v) => Uri.Path.Segment(k) -> v }, + captures = captures, + variadic = variadic, + end = end + ) "Honor UriTranslations" in { val svc = TranslateUri("/bar")(Router("/" -> new RhoRoutes[IO] { GET / "foo" |>> "foo" }.toRoutes())).orNotFound - val req = Request[IO](Method.GET, uri = Uri(path = "/bar/foo")) + val req = Request[IO](Method.GET, uri = Uri(path = Path.fromString("/bar/foo"))) val resp = svc(req).unsafeRunSync() resp.status must_== Status.Ok @@ -83,30 +88,37 @@ class PathTreeSpec extends Specification { "MatchNodes" should { "Merge empty nodes" in { - val n = MatchNode("") + val n = matchNodeFromString("") n.merge(n) must_== n } "Merge non-empty nodes with the same method" in { - val n1 = MatchNode("foo", end = Map(Method.GET -> l)) - val n2 = MatchNode("foo", end = Map(Method.GET -> l)) + val n1 = matchNodeFromString("foo", end = Map(Method.GET -> l)) + val n2 = matchNodeFromString("foo", end = Map(Method.GET -> l)) n1.merge(n2) must_== n1.copy(end = Map(Method.GET -> (l ++ l))) } "Merge non-empty nodes with different defined methods" in { - val n1 = MatchNode("foo", end = Map(Method.GET -> l)) - val n2 = MatchNode("foo", end = Map(Method.POST -> l)) + val n1 = matchNodeFromString("foo", end = Map(Method.GET -> l)) + val n2 = matchNodeFromString("foo", end = Map(Method.POST -> l)) n1.merge(n2) must_== n1.copy(end = n1.end ++ n2.end) } "Merge non-empty intermediate nodes with matching paths" in { val n1 = - MatchNode("foo", matches = Map("bar" -> MatchNode("bar", end = Map(Method.GET -> l)))) + matchNodeFromString( + "foo", + matches = Map("bar" -> matchNodeFromString("bar", end = Map(Method.GET -> l))) + ) val n2 = - MatchNode("foo", matches = Map("bar" -> MatchNode("bar", end = Map(Method.POST -> l)))) - val r = MatchNode( + matchNodeFromString( + "foo", + matches = Map("bar" -> matchNodeFromString("bar", end = Map(Method.POST -> l))) + ) + val r = matchNodeFromString( "foo", - matches = Map("bar" -> MatchNode("bar", end = Map(Method.POST -> l, Method.GET -> l))) + matches = + Map("bar" -> matchNodeFromString("bar", end = Map(Method.POST -> l, Method.GET -> l))) ) n1.merge(n2) must_== r @@ -114,52 +126,63 @@ class PathTreeSpec extends Specification { "Merge non-empty intermediate nodes with non matching paths" in { val endm: Map[Method, Leaf] = Map(Method.GET -> l) - val bar = MatchNode("bar", end = endm) - val bizz = MatchNode("bizz", end = endm) - val n1 = MatchNode("foo", matches = Map("bar" -> bar)) - val n2 = MatchNode("foo", matches = Map("bizz" -> bizz)) + val bar = matchNodeFromString("bar", end = endm) + val bizz = matchNodeFromString("bizz", end = endm) + val n1 = matchNodeFromString("foo", matches = Map("bar" -> bar)) + val n2 = matchNodeFromString("foo", matches = Map("bizz" -> bizz)) - n1.merge(n2) must_== MatchNode("foo", matches = Map("bar" -> bar, "bizz" -> bizz)) + n1.merge(n2) must_== matchNodeFromString("foo", matches = Map("bar" -> bar, "bizz" -> bizz)) } "Merge non-empty intermediate nodes with mixed matching paths" in { val endm: Map[Method, Leaf] = Map(Method.GET -> l) - val bar = MatchNode("bar", end = endm) - val bizz = CaptureNode(StringParser.booleanParser[IO], end = endm) - val n1 = MatchNode("foo", matches = Map("bar" -> bar)) - val n2 = MatchNode("foo", captures = List(bizz)) + val bar = matchNodeFromString("bar", end = endm) + val bizz = captureNodeFromString(StringParser.booleanParser[IO], end = endm) + val n1 = matchNodeFromString("foo", matches = Map("bar" -> bar)) + val n2 = matchNodeFromString("foo", captures = List(bizz)) - n1.merge(n2) must_== MatchNode("foo", matches = Map("bar" -> bar), captures = List(bizz)) + n1.merge(n2) must_== matchNodeFromString( + "foo", + matches = Map("bar" -> bar), + captures = List(bizz) + ) } } "CapturesNodes" should { val p = StringParser.booleanParser[IO] "Merge empty CaptureNodes" in { - val n = CaptureNode(p) + val n = captureNodeFromString(p) n.merge(n) must_== n } "Merge non-empty nodes with the same method" in { - val n1 = CaptureNode(p, end = Map(Method.GET -> l)) - val n2 = CaptureNode(p, end = Map(Method.GET -> l)) + val n1 = captureNodeFromString(p, end = Map(Method.GET -> l)) + val n2 = captureNodeFromString(p, end = Map(Method.GET -> l)) n1.merge(n2) must_== n1.copy(end = Map(Method.GET -> (l ++ l))) } "Merge non-empty nodes with different defined methods" in { - val n1 = CaptureNode(p, end = Map(Method.GET -> l)) - val n2 = CaptureNode(p, end = Map(Method.POST -> l)) + val n1 = captureNodeFromString(p, end = Map(Method.GET -> l)) + val n2 = captureNodeFromString(p, end = Map(Method.POST -> l)) n1.merge(n2) must_== n1.copy(end = n1.end ++ n2.end) } "Merge non-empty intermediate nodes with matching paths" in { val n1 = - CaptureNode(p, matches = Map("bar" -> MatchNode("bar", end = Map(Method.GET -> l)))) + captureNodeFromString( + p, + matches = Map("bar" -> matchNodeFromString("bar", end = Map(Method.GET -> l))) + ) val n2 = - CaptureNode(p, matches = Map("bar" -> MatchNode("bar", end = Map(Method.POST -> l)))) - val r = CaptureNode( + captureNodeFromString( + p, + matches = Map("bar" -> matchNodeFromString("bar", end = Map(Method.POST -> l))) + ) + val r = captureNodeFromString( p, - matches = Map("bar" -> MatchNode("bar", end = Map(Method.POST -> l, Method.GET -> l))) + matches = + Map("bar" -> matchNodeFromString("bar", end = Map(Method.POST -> l, Method.GET -> l))) ) n1.merge(n2) must_== r @@ -167,46 +190,50 @@ class PathTreeSpec extends Specification { "Merge non-empty intermediate nodes with non matching paths" in { val endm: Map[Method, Leaf] = Map(Method.GET -> l) - val bar = MatchNode("bar", end = endm) - val bizz = MatchNode("bizz", end = endm) - val n1 = CaptureNode(p, matches = Map("bar" -> bar)) - val n2 = CaptureNode(p, matches = Map("bizz" -> bizz)) + val bar = matchNodeFromString("bar", end = endm) + val bizz = matchNodeFromString("bizz", end = endm) + val n1 = captureNodeFromString(p, matches = Map("bar" -> bar)) + val n2 = captureNodeFromString(p, matches = Map("bizz" -> bizz)) - n1.merge(n2) must_== CaptureNode(p, matches = Map("bar" -> bar, "bizz" -> bizz)) + n1.merge(n2) must_== captureNodeFromString(p, matches = Map("bar" -> bar, "bizz" -> bizz)) } "Merge non-empty intermediate nodes with mixed matching paths" in { val endm: Map[Method, Leaf] = Map(Method.GET -> l) - val bar = MatchNode("bar", end = endm) - val bizz = CaptureNode(StringParser.booleanParser[IO], end = endm) - val n1 = CaptureNode(p, matches = Map("bar" -> bar)) - val n2 = CaptureNode(p, captures = List(bizz)) + val bar = matchNodeFromString("bar", end = endm) + val bizz = captureNodeFromString(StringParser.booleanParser[IO], end = endm) + val n1 = captureNodeFromString(p, matches = Map("bar" -> bar)) + val n2 = captureNodeFromString(p, captures = List(bizz)) - n1.merge(n2) must_== CaptureNode(p, matches = Map("bar" -> bar), captures = List(bizz)) + n1.merge(n2) must_== captureNodeFromString( + p, + matches = Map("bar" -> bar), + captures = List(bizz) + ) } "Merging should preserve order" in { val endm: Map[Method, Leaf] = Map(Method.GET -> l) - val bar = CaptureNode(StringParser.intParser[IO], end = endm) - val bizz = CaptureNode(StringParser.booleanParser[IO], end = endm) - val n1 = CaptureNode(p, captures = List(bar)) - val n2 = CaptureNode(p, captures = List(bizz)) + val bar = captureNodeFromString(StringParser.intParser[IO], end = endm) + val bizz = captureNodeFromString(StringParser.booleanParser[IO], end = endm) + val n1 = captureNodeFromString(p, captures = List(bar)) + val n2 = captureNodeFromString(p, captures = List(bizz)) - n1.merge(n2) must_== CaptureNode(p, captures = List(bar, bizz)) + n1.merge(n2) must_== captureNodeFromString(p, captures = List(bar, bizz)) } "Merging should promote order of the same nodes" in { val end1: Map[Method, Leaf] = Map(Method.GET -> l) val end2: Map[Method, Leaf] = Map(Method.POST -> l) - val foo = CaptureNode(StringParser.shortParser[IO], end = end1) - val bar = CaptureNode(StringParser.intParser[IO], end = end1) - val bizz = CaptureNode(StringParser.booleanParser[IO], end = end1) + val foo = captureNodeFromString(StringParser.shortParser[IO], end = end1) + val bar = captureNodeFromString(StringParser.intParser[IO], end = end1) + val bizz = captureNodeFromString(StringParser.booleanParser[IO], end = end1) - val n1 = CaptureNode(p, captures = List(foo, bar)) - val n2 = CaptureNode(p, captures = List(bizz, foo.copy(end = end2))) + val n1 = captureNodeFromString(p, captures = List(foo, bar)) + val n2 = captureNodeFromString(p, captures = List(bizz, foo.copy(end = end2))) - n1.merge(n2) must_== CaptureNode( + n1.merge(n2) must_== captureNodeFromString( p, captures = List(foo.copy(end = end1 ++ end2), bar, bizz) ) diff --git a/core/src/test/scala/org/http4s/rho/bits/ResultMatcherSpec.scala b/core/src/test/scala/org/http4s/rho/bits/ResultMatcherSpec.scala index e4c424408..63abe9dfd 100644 --- a/core/src/test/scala/org/http4s/rho/bits/ResultMatcherSpec.scala +++ b/core/src/test/scala/org/http4s/rho/bits/ResultMatcherSpec.scala @@ -115,10 +115,10 @@ class ResultMatcherSpec extends Specification { case class ModelB(name: String, id: Long) implicit def w1[F[_]: Applicative]: EntityEncoder[F, ModelA] = - EntityEncoder.simple[F, ModelA]()(_ => Chunk.bytes("A".getBytes)) + EntityEncoder.simple[F, ModelA]()(_ => Chunk.array("A".getBytes)) implicit def w2[F[_]: Applicative]: EntityEncoder[F, ModelB] = - EntityEncoder.simple[F, ModelB]()(_ => Chunk.bytes("B".getBytes)) + EntityEncoder.simple[F, ModelB]()(_ => Chunk.array("B".getBytes)) val srvc = new TRhoRoutes[IO] { GET / "foo" |>> { () => @@ -157,8 +157,8 @@ object Foo { case class FooB(name: String, id: Long) implicit def w1[F[_]: Applicative]: EntityEncoder[F, FooA] = - EntityEncoder.simple[F, FooA]()(_ => Chunk.bytes("A".getBytes)) + EntityEncoder.simple[F, FooA]()(_ => Chunk.array("A".getBytes)) implicit def w2[F[_]: Applicative]: EntityEncoder[F, FooB] = - EntityEncoder.simple[F, FooB]()(_ => Chunk.bytes("B".getBytes)) + EntityEncoder.simple[F, FooB]()(_ => Chunk.array("B".getBytes)) } 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 dde83f950..e68d16b02 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 @@ -6,7 +6,7 @@ import org.http4s.implicits._ import org.http4s.rho.swagger.SwaggerMetadata import org.http4s.rho.swagger.models.{Info, Tag} import org.http4s.rho.swagger.syntax.{io => ioSwagger} -import org.http4s.server.blaze.BlazeServerBuilder +import org.http4s.blaze.server.BlazeServerBuilder import org.log4s.getLogger import scala.concurrent.ExecutionContext.global 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 347283469..3f0e65f89 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 @@ -1,7 +1,7 @@ package com.http4s.rho.swagger.demo import cats.Monad -import cats.effect.Effect +import cats.effect._ import cats.implicits._ import com.http4s.rho.swagger.demo.JsonEncoder.{AutoSerializable, _} import com.http4s.rho.swagger.demo.MyRoutes._ @@ -21,12 +21,12 @@ class MyRoutes[F[+_]: Effect](swaggerSyntax: SwaggerSyntax[F]) extends RhoRoutes case Some(_) => // Cookie found, good to go None case None => // Didn't find cookie - Some(TemporaryRedirect(Uri(path = "/addcookie")).widen) + Some(TemporaryRedirect(Uri(path = Uri.Path.fromString("/addcookie"))).widen) } } "We don't want to have a real 'root' route anyway... " ** - GET |>> TemporaryRedirect(Uri(path = "/swagger-ui")) + GET |>> TemporaryRedirect(Uri(path = Uri.Path.fromString("/swagger-ui"))) // We want to define this chunk of the service as abstract for reuse below val hello = "hello" @@ GET / "hello" diff --git a/project/Dependencies.scala b/project/Dependencies.scala index 9c435ffd6..573fabf7c 100644 --- a/project/Dependencies.scala +++ b/project/Dependencies.scala @@ -3,7 +3,7 @@ import Keys._ // format: off object Dependencies { - lazy val http4sVersion = "0.21.23" + lazy val http4sVersion = "0.22.0-RC1" lazy val specs2Version = "4.10.6" val scala_213 = "2.13.4" @@ -22,7 +22,7 @@ object Dependencies { lazy val logbackClassic = "ch.qos.logback" % "logback-classic" % "1.2.3" lazy val uadetector = "net.sf.uadetector" % "uadetector-resources" % "2014.10" lazy val shapeless = "com.chuusai" %% "shapeless" % "2.3.3" - lazy val scalaXml = "org.scala-lang.modules" %% "scala-xml" % "1.3.0" + lazy val scalaXml = "org.scala-lang.modules" %% "scala-xml" % "2.0.0" lazy val swaggerUi = "org.webjars" % "swagger-ui" % "3.46.0" lazy val specs2 = Seq("org.specs2" %% "specs2-core" % specs2Version % "test", diff --git a/swagger/src/main/scala/org/http4s/rho/swagger/SwaggerModelsBuilder.scala b/swagger/src/main/scala/org/http4s/rho/swagger/SwaggerModelsBuilder.scala index f56539c93..b933009f6 100644 --- a/swagger/src/main/scala/org/http4s/rho/swagger/SwaggerModelsBuilder.scala +++ b/swagger/src/main/scala/org/http4s/rho/swagger/SwaggerModelsBuilder.scala @@ -99,7 +99,7 @@ private[swagger] class SwaggerModelsBuilder[F[_]](formats: SwaggerFormats)(impli def go(stack: List[PathOperation], pps: List[PathParameter]): List[PathParameter] = stack match { case Nil => pps - case PathMatch("") :: xs => go(xs, pps) + case PathMatch.empty :: xs => go(xs, pps) case PathMatch(_) :: xs => go(xs, pps) case MetaCons(_, _) :: xs => go(xs, pps) case PathCapture(id, desc, p, _) :: xs => @@ -124,7 +124,7 @@ private[swagger] class SwaggerModelsBuilder[F[_]](formats: SwaggerFormats)(impli def go(stack: List[PathOperation], summary: Option[String]): Option[String] = stack match { - case PathMatch("") :: Nil => go(Nil, summary) + case PathMatch.empty :: Nil => go(Nil, summary) case PathMatch(_) :: xs => go(xs, summary) case PathCapture(_, _, _, _) :: xs => go(xs, summary) case CaptureTail :: _ => summary @@ -145,10 +145,10 @@ private[swagger] class SwaggerModelsBuilder[F[_]](formats: SwaggerFormats)(impli def go(stack: List[PathOperation], tags: List[String]): List[String] = stack match { - case PathMatch("") :: xs => go(xs, tags) + case PathMatch.empty :: xs => go(xs, tags) case PathMatch(segment) :: xs => tags match { - case Nil => go(xs, segment :: Nil) + case Nil => go(xs, segment.decoded() :: Nil) case ts => go(xs, ts) } case PathCapture(id, _, _, _) :: xs => @@ -495,8 +495,8 @@ private[swagger] class SwaggerModelsBuilder[F[_]](formats: SwaggerFormats)(impli def go(stack: List[PathOperation], pathStr: String): String = stack match { case Nil => if (pathStr.isEmpty) "/" else pathStr - case PathMatch("") :: Nil => pathStr + "/" - case PathMatch("") :: xs => go(xs, pathStr) + case PathMatch.empty :: Nil => pathStr + "/" + case PathMatch.empty :: xs => go(xs, pathStr) case PathMatch(s) :: xs => go(xs, pathStr + "/" + s) case MetaCons(_, _) :: xs => go(xs, pathStr) case PathCapture(id, _, _, _) :: xs => go(xs, s"$pathStr/{$id}") 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 0ca6adf89..d01212807 100644 --- a/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerModelsBuilderSpec.scala +++ b/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerModelsBuilderSpec.scala @@ -891,10 +891,10 @@ class SwaggerModelsBuilderSpec extends Specification { .withContentType(`Content-Type`(MediaType.application.json, Charset.`UTF-8`)) implicit def listEntityEncoder[F[_]: Applicative, A]: EntityEncoder[F, List[A]] = - EntityEncoder.simple[F, List[A]]()(_ => Chunk.bytes("A".getBytes)) + EntityEncoder.simple[F, List[A]]()(_ => Chunk.array("A".getBytes)) implicit def mapEntityEncoder[F[_]: Applicative, A, B]: EntityEncoder[F, Map[A, B]] = - EntityEncoder.simple[F, Map[A, B]]()(_ => Chunk.bytes("A".getBytes)) + EntityEncoder.simple[F, Map[A, B]]()(_ => Chunk.array("A".getBytes)) case class CsvFile() 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 5b127f2ca..c7007f919 100644 --- a/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerSupportSpec.scala +++ b/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerSupportSpec.scala @@ -44,7 +44,7 @@ class SwaggerSupportSpec extends Specification { "Expose an API listing" in { val service = baseRoutes.toRoutes(createRhoMiddleware(swaggerRoutesInSwagger = true)) - val r = Request[IO](GET, Uri(path = "/swagger.json")) + val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) val JObject(List((a, JObject(_)), (b, JObject(_)), (c, JObject(_)), (d, JObject(_)))) = parseJson(RRunner(service).checkOk(r)) \\ "paths" @@ -55,7 +55,7 @@ class SwaggerSupportSpec extends Specification { "Support prefixed routes" in { val service = ("foo" /: baseRoutes).toRoutes(createRhoMiddleware(swaggerRoutesInSwagger = true)) - val r = Request[IO](GET, Uri(path = "/swagger.json")) + val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) val JObject(List((a, JObject(_)), (b, JObject(_)), (c, JObject(_)), (d, JObject(_)))) = parseJson(RRunner(service).checkOk(r)) \\ "paths" @@ -79,7 +79,7 @@ class SwaggerSupportSpec extends Specification { val swaggerRoutes = createSwaggerRoute(aggregateSwagger) val httpRoutes = NonEmptyList.of(baseRoutes, moarRoutes, swaggerRoutes).reduceLeft(_ and _) - val r = Request[IO](GET, Uri(path = "/swagger.json")) + val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) val JObject(List((a, JObject(_)), (b, JObject(_)), (c, JObject(_)), (d, JObject(_)))) = parseJson(RRunner(httpRoutes.toRoutes()).checkOk(r)) \\ "paths" @@ -89,7 +89,7 @@ class SwaggerSupportSpec extends Specification { "Support endpoints which end in a slash" in { val service = trailingSlashRoutes.toRoutes(createRhoMiddleware()) - val r = Request[IO](GET, Uri(path = "/swagger.json")) + val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) val JObject(List((a, JObject(_)))) = parseJson(RRunner(service).checkOk(r)) \\ "paths" a should_== "/foo/" @@ -97,7 +97,7 @@ class SwaggerSupportSpec extends Specification { "Support endpoints which end in a slash being mixed with normal endpoints" in { val service = mixedTrailingSlashesRoutes.toRoutes(createRhoMiddleware()) - val r = Request[IO](GET, Uri(path = "/swagger.json")) + val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) val JObject(List((a, JObject(_)), (b, JObject(_)), (c, JObject(_)))) = parseJson(RRunner(service).checkOk(r)) \\ "paths" @@ -111,7 +111,7 @@ class SwaggerSupportSpec extends Specification { val swaggerRoutes = createSwaggerRoute(aggregateSwagger) val httpRoutes = NonEmptyList.of(baseRoutes, moarRoutes, swaggerRoutes).reduceLeft(_ and _) - val r = Request[IO](GET, Uri(path = "/swagger.json")) + val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) val JObject( List( @@ -140,7 +140,7 @@ class SwaggerSupportSpec extends Specification { "Check metadata in API listing" in { val service = metaDataRoutes.toRoutes(createRhoMiddleware(swaggerRoutesInSwagger = true)) - val r = Request[IO](GET, Uri(path = "/swagger.json")) + val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) val json = parseJson(RRunner(service).checkOk(r)) @@ -205,7 +205,7 @@ class SwaggerSupportSpec extends Specification { ) ) - val r = Request[IO](GET, Uri(path = "/swagger-test.json")) + val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger-test.json"))) val json = parseJson(RRunner(service).checkOk(r)) val JString(icn) = json \ "info" \ "contact" \ "name" @@ -239,7 +239,7 @@ class SwaggerSupportSpec extends Specification { "Check metadata in API listing" in { val service = metaDataRoutes.toRoutes(createRhoMiddleware(swaggerRoutesInSwagger = true)) - val r = Request[IO](GET, Uri(path = "/swagger.json")) + val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) val json = parseJson(RRunner(service).checkOk(r)) From f897aeb17dd37a8a75c097c9be0d99f992b06518 Mon Sep 17 00:00:00 2001 From: Darren Gibson Date: Thu, 3 Jun 2021 18:57:14 -0500 Subject: [PATCH 3/6] Refactor to use new Http4s header format --- build.sbt | 2 +- .../main/scala/org/http4s/rho/Result.scala | 18 +- .../http4s/rho/RhoDslHeaderExtractors.scala | 265 ++++++++++-------- .../scala/org/http4s/rho/bits/Metadata.scala | 3 +- .../org/http4s/rho/bits/ResultResponse.scala | 7 +- core/src/test/scala/ApiExamples.scala | 10 +- .../test/scala/org/http4s/rho/ApiTest.scala | 57 ++-- .../org/http4s/rho/AuthedContextSpec.scala | 6 +- .../org/http4s/rho/CodecRouterSpec.scala | 10 +- .../org/http4s/rho/CompileRoutesSpec.scala | 17 +- .../http4s/rho/ParamDefaultValueSpec.scala | 4 +- .../scala/org/http4s/rho/ResultSpec.scala | 4 +- .../scala/org/http4s/rho/RhoRoutesSpec.scala | 48 ++-- .../org/http4s/rho/UriConvertibleSpec.scala | 10 +- .../org/http4s/rho/bits/HListToFuncSpec.scala | 4 +- .../org/http4s/rho/bits/PathTreeSpec.scala | 5 +- .../rho/bits/ResponseGeneratorSpec.scala | 28 +- .../http4s/rho/swagger/demo/JsonEncoder.scala | 26 -- .../http4s/rho/swagger/demo/MyRoutes.scala | 21 +- project/Dependencies.scala | 31 +- .../rho/swagger/SwaggerModelsBuilder.scala | 5 +- .../http4s/rho/swagger/SwaggerSupport.scala | 4 +- 22 files changed, 296 insertions(+), 289 deletions(-) delete mode 100644 examples/src/main/scala/com/http4s/rho/swagger/demo/JsonEncoder.scala diff --git a/build.sbt b/build.sbt index b44f22a55..82d9f2c27 100644 --- a/build.sbt +++ b/build.sbt @@ -89,7 +89,6 @@ lazy val `rho-examples` = project .settings(Revolver.settings) .settings( exampleDeps, - libraryDependencies ++= Seq(logbackClassic, http4sXmlInstances), dontPublish ) .dependsOn(`rho-swagger`, `rho-swagger-ui`) @@ -132,6 +131,7 @@ lazy val buildSettings = publishing ++ shapeless, silencerPlugin, silencerLib, + kindProjector, http4sServer % "provided", logbackClassic % "test" ), diff --git a/core/src/main/scala/org/http4s/rho/Result.scala b/core/src/main/scala/org/http4s/rho/Result.scala index 6b21f1d3f..c85e81cd9 100644 --- a/core/src/main/scala/org/http4s/rho/Result.scala +++ b/core/src/main/scala/org/http4s/rho/Result.scala @@ -4,6 +4,7 @@ package rho import cats._ import org.http4s.headers.`Content-Type` import org.typelevel.vault._ +import org.typelevel.ci.CIString /** A helper for capturing the result types and status codes from routes */ sealed case class Result[ @@ -102,23 +103,19 @@ trait ResultSyntaxInstances[F[_]] { def withHeaders(headers: Headers): Self = Result(resp.withHeaders(headers)) - def withHeaders(headers: Header*): Self = Result(resp.withHeaders(headers: _*)) + def withHeaders(headers: Header.ToRaw*): Self = Result(resp.withHeaders(headers: _*)) def withAttributes(attributes: Vault): Self = Result(resp.withAttributes(attributes)) def transformHeaders(f: Headers => Headers): Self = Result(resp.transformHeaders(f)) - def filterHeaders(f: Header => Boolean): Self = Result(resp.filterHeaders(f)) + def filterHeaders(f: Header.Raw => Boolean): Self = Result(resp.filterHeaders(f)) - def removeHeader(key: HeaderKey): Self = Result(resp.removeHeader(key)) + def removeHeader(key: CIString): Self = Result(resp.removeHeader(key)) - def putHeaders(headers: Header*): Self = Result(resp.putHeaders(headers: _*)) + def removeHeader[A](implicit h: Header[A, _]): Self = Result(resp.removeHeader(h)) - @scala.deprecated("Use withHeaders instead", "0.20.0-M2") - def replaceAllHeaders(headers: Headers): Self = Result(resp.replaceAllHeaders(headers)) - - @scala.deprecated("Use withHeaders instead", "0.20.0-M2") - def replaceAllHeaders(headers: Header*): Self = Result(resp.replaceAllHeaders(headers: _*)) + def putHeaders(headers: Header.ToRaw*): Self = Result(resp.putHeaders(headers: _*)) def withTrailerHeaders(trailerHeaders: F[Headers]): Self = Result( resp.withTrailerHeaders(trailerHeaders) @@ -128,9 +125,6 @@ trait ResultSyntaxInstances[F[_]] { def trailerHeaders(implicit F: Applicative[F]): F[Headers] = resp.trailerHeaders(F) - @scala.deprecated("Use withContentType(`Content-Type`(t)) instead", "0.20.0-M2") - def withType(t: MediaType)(implicit F: Functor[F]): Self = Result(resp.withType(t)(F)) - def withContentType(contentType: `Content-Type`): Self = Result( resp.withContentType(contentType) ) diff --git a/core/src/main/scala/org/http4s/rho/RhoDslHeaderExtractors.scala b/core/src/main/scala/org/http4s/rho/RhoDslHeaderExtractors.scala index 3a1cee8a9..fb8442e54 100644 --- a/core/src/main/scala/org/http4s/rho/RhoDslHeaderExtractors.scala +++ b/core/src/main/scala/org/http4s/rho/RhoDslHeaderExtractors.scala @@ -1,7 +1,8 @@ package org.http4s.rho -import cats.syntax.functor._ -import cats.Monad +import cats._ +import cats.data._ +import cats.implicits._ import org.http4s._ import org.http4s.rho.Result.BaseResult import org.http4s.rho.bits.RequestAST.CaptureRule @@ -15,92 +16,10 @@ trait RhoDslHeaderExtractors[F[_]] extends FailureResponseOps[F] { private[this] val logger: Logger = getLogger - /** Requires that the header exists - * - * @param header `HeaderKey` that identifies the header which is required - */ - def exists(header: HeaderKey.Extractable)(implicit F: Monad[F]): TypedHeader[F, HNil] = - existsAndR(header)(_ => None) - - /** Requires that the header exists and satisfies the condition - * - * @param header `HeaderKey` that identifies the header to capture and parse - * @param f predicate function where a return value of `false` signals an invalid - * header and aborts evaluation with a _BadRequest_ response. - */ - def existsAnd[H <: HeaderKey.Extractable](header: H)(f: H#HeaderT => Boolean)(implicit - F: Monad[F]): TypedHeader[F, HNil] = - existsAndR[H](header) { h => - if (f(h)) None - else Some(invalidHeaderResponse(header)) - } - - /** Check that the header exists and satisfies the condition - * - * @param header `HeaderKey` that identifies the header to capture and parse - * @param f function that evaluates the header and returns a Some(Response) to - * immediately send back to the user or None to continue evaluation. - */ - def existsAndR[H <: HeaderKey.Extractable](header: H)(f: H#HeaderT => Option[F[BaseResult[F]]])( - implicit F: Monad[F]): TypedHeader[F, HNil] = - captureMapR(header, None) { h => - f(h) match { - case Some(r) => Left(r) - case None => Right(()) - } - }.ignore - - /** Capture the header and put it into the function args stack, if it exists - * - * @param key `HeaderKey` used to identify the header to capture - */ - def captureOptionally[H <: HeaderKey.Extractable](key: H)(implicit - F: Monad[F]): TypedHeader[F, Option[H#HeaderT] :: HNil] = - _captureMapR[H, Option[H#HeaderT]](key, isRequired = false)(Right(_)) - - /** requires the header and will pull this header from the pile and put it into the function args stack - * - * @param key `HeaderKey` used to identify the header to capture - */ - def capture[H <: HeaderKey.Extractable](key: H)(implicit - F: Monad[F]): TypedHeader[F, H#HeaderT :: HNil] = - captureMap(key)(identity) - - /** Capture the header and put it into the function args stack, if it exists otherwise put the default in the args stack - * - * @param key `HeaderKey` used to identify the header to capture - * @param default The default to be used if the header was not present - */ - def captureOrElse[H <: HeaderKey.Extractable](key: H)(default: H#HeaderT)(implicit - F: Monad[F]): TypedHeader[F, H#HeaderT :: HNil] = - captureOptionally[H](key).map { header: Option[H#HeaderT] => - header.getOrElse(default) - } - - /** Capture a specific header and map its value - * - * @param key `HeaderKey` used to identify the header to capture - * @param f mapping function - */ - def captureMap[H <: HeaderKey.Extractable, R](key: H)(f: H#HeaderT => R)(implicit - F: Monad[F]): TypedHeader[F, R :: HNil] = - captureMapR(key, None, isRequired = true)(f.andThen(Right(_))) - - /** Capture a specific header and map its value with an optional override of the missing header response - * - * @param key `HeaderKey` used to identify the header to capture - * @param missingHeaderResult optional override result for the case of a missing header - * @param isRequired indicates for metadata purposes that the header is required, always true if `missingHeaderResult` is unset - * @param f mapping function - */ - def captureMapR[H <: HeaderKey.Extractable, R]( - key: H, - missingHeaderResult: Option[F[BaseResult[F]]] = None, - isRequired: Boolean = false)(f: H#HeaderT => Either[F[BaseResult[F]], R])(implicit - F: Monad[F]): TypedHeader[F, R :: HNil] = - _captureMapR(key, missingHeaderResult, missingHeaderResult.isEmpty || isRequired)(f) - /** Create a header capture rule using the `Request`'s `Headers` + * + * In general, this function should be avoided because no metadata will be captured and + * added to the swagger documention * * @param f function generating the result or failure */ @@ -118,45 +37,149 @@ trait RhoDslHeaderExtractors[F[_]] extends FailureResponseOps[F] { f: Request[F] => ResultResponse[F, R]): TypedHeader[F, R :: HNil] = TypedHeader[F, R :: HNil](CaptureRule(f)) - private def _captureMapR[H <: HeaderKey.Extractable, R]( - key: H, - missingHeaderResult: Option[F[BaseResult[F]]], - isRequired: Boolean)(f: H#HeaderT => Either[F[BaseResult[F]], R])(implicit - F: Monad[F]): TypedHeader[F, R :: HNil] = - _captureMapR[H, R](key, isRequired) { + def H[A](implicit H: Header[A, _], S: Header.Select[A], F: Monad[F]): HeaderOps[A, S.F] = + new HeaderOps[A, S.F]()(H, S, F) + + class HeaderOps[A, H[_]](implicit H: Header[A, _], S: Header.Select.Aux[A, H], F: Monad[F]) { + type HR = H[A] + + /** Requires that the header exists + */ + def exists: TypedHeader[F, HNil] = + existsAndR(_ => None) + + /** Requires that the header exists and satisfies the condition + * + * @param f predicate function where a return value of `false` signals an invalid + * header and aborts evaluation with a _BadRequest_ response. + */ + def existsAnd(f: HR => Boolean): TypedHeader[F, HNil] = + existsAndR { (h: HR) => + if (f(h)) None + else Some(invalidHeaderResponse[A]) + } + + /** Check that the header exists and satisfies the condition + * + * @param f function that evaluates the header and returns a Some(Response) to + * immediately send back to the user or None to continue evaluation. + */ + def existsAndR(f: HR => Option[F[BaseResult[F]]]): TypedHeader[F, HNil] = + captureMapR(None) { h => + f(h) match { + case Some(r) => Left(r) + case None => Right(()) + } + }.ignore + + /** Capture the header and put it into the function args stack, if it exists + */ + def captureOptionally: TypedHeader[F, Option[HR] :: HNil] = + _captureMapR(isRequired = false)(SuccessResponse[F, Option[HR]](_)) + + /** requires the header and will pull this header from the pile and put it into the function args stack + */ + def capture: TypedHeader[F, HR :: HNil] = + captureMap(identity) + + /** Capture the header and put it into the function args stack, if it exists otherwise put the default in the args stack + * + * @param default The default to be used if the header was not present + */ + def captureOrElse(default: HR): TypedHeader[F, HR :: HNil] = + captureOptionally.map { header: Option[HR] => + header.getOrElse(default) + } + + /** Capture a specific header and map its value + * + * @param f mapping function + */ + def captureMap[R](f: HR => R): TypedHeader[F, R :: HNil] = + captureMapR[R](None, isRequired = true)(f.andThen(Right(_))) + + /** Capture a specific header and map its value with an optional override of the missing header response + * + * @param missingHeaderResult optional override result for the case of a missing header + * @param isRequired indicates for metadata purposes that the header is required, always true if `missingHeaderResult` is unset + * @param f mapping function + */ + def captureMapR[R]( + missingHeaderResult: Option[F[BaseResult[F]]] = None, + isRequired: Boolean = false)( + f: HR => Either[F[BaseResult[F]], R]): TypedHeader[F, R :: HNil] = + _captureMapR( + missingHeaderResult.map(FailureResponse.result[F](_)), + missingHeaderResult.isEmpty || isRequired + )(f.andThen(ResultResponse.fromEither(_))) + + /** Capture a specific header and map its value + * + * @param f mapping function + */ + def captureMapR[R](f: HR => Either[F[BaseResult[F]], R]): TypedHeader[F, R :: HNil] = + captureMapR()(f) + } + + private def _captureMapR[A, H[_], R]( + missingHeaderResult: Option[FailureResponse[F]], + isRequired: Boolean)(f: H[A] => ResultResponse[F, R])(implicit + F: Monad[F], + H: Header[A, _], + S: Header.Select.Aux[A, H]): TypedHeader[F, R :: HNil] = + _captureMapR[A, H, R](isRequired) { case Some(header) => f(header) - case None => Left(missingHeaderResult.getOrElse(missingHeaderResponse(key))) + case None => missingHeaderResult.getOrElse(FailureResponse.result(missingHeaderResponse[A])) } - private def _captureMapR[H <: HeaderKey.Extractable, R](key: H, isRequired: Boolean)( - f: Option[H#HeaderT] => Either[F[BaseResult[F]], R])(implicit - F: Monad[F]): TypedHeader[F, R :: HNil] = + private def _captureMapR[A, H[_], R](isRequired: Boolean)( + f: Option[H[A]] => ResultResponse[F, R])(implicit + F: Monad[F], + H: Header[A, _], + S: Header.Select.Aux[A, H]): TypedHeader[F, R :: HNil] = genericHeaderCapture[R] { headers => - val h = headers.get(key) - try f(h) match { - case Right(r) => SuccessResponse(r) - case Left(r) => FailureResponse.result(r) - } catch { - case NonFatal(e) => - FailureResponse.result(errorProcessingHeaderResponse(key, h, e)) + def process(h: Option[H[A]]): ResultResponse[F, R] = + try f(h) + catch { + case NonFatal(e) => + FailureResponse.result { + errorProcessingHeaderResponse[A](h.map(S.toRaw), e) + } + } + + def errorParsingHeader( + errors: NonEmptyList[ParseFailure] + ) = FailureResponse.result(errorParsingHeaderResponse(errors)) + + S.fromSafe(headers.headers) match { + case None => process(Option.empty) + case Some(Ior.Right(value)) => process(Option(value)) + case Some(Ior.Both(errors, _)) => errorParsingHeader(errors) + case Some(Ior.Left(errors)) => errorParsingHeader(errors) } - }.withMetadata(HeaderMetaData(key, isRequired = isRequired)) - - protected def invalidHeaderResponse[H <: HeaderKey](h: H)(implicit - F: Monad[F]): F[BaseResult[F]] = - BadRequest(s"Invalid header: ${h.name}").widen - - protected def missingHeaderResponse[H <: HeaderKey](key: H)(implicit - F: Monad[F]): F[BaseResult[F]] = - BadRequest(s"Missing header: ${key.name}").widen - - protected def errorProcessingHeaderResponse[H <: HeaderKey.Extractable]( - key: H, - header: Option[H#HeaderT], - nonfatal: Throwable)(implicit F: Monad[F]): F[BaseResult[F]] = { - logger.error(nonfatal)(s"""Failure during header capture: "${key.name}" ${header.fold( - "Undefined" - )(v => s"""= "${v.value}"""")}""") + + }.withMetadata(HeaderMetaData[A](H.name, isRequired)) + + protected def invalidHeaderResponse[A](implicit F: Monad[F], H: Header[A, _]): F[BaseResult[F]] = + BadRequest(s"Invalid header: ${H.name}").widen + + protected def missingHeaderResponse[A](implicit F: Monad[F], H: Header[A, _]): F[BaseResult[F]] = + BadRequest(s"Missing header: ${H.name}").widen + + protected def errorParsingHeaderResponse[A, H[_]]( + errors: NonEmptyList[ParseFailure] + )(implicit F: Monad[F], H: Header[A, _]): F[BaseResult[F]] = + BadRequest( + s"Failed to parse header: ${H.name} with ${errors.map(_.sanitized).mkString_(",")}" + ).widen + + protected def errorProcessingHeaderResponse[A]( + header: Option[Header.Raw], + nonfatal: Throwable)(implicit F: Monad[F], H: Header[A, _]): F[BaseResult[F]] = { + logger.error(nonfatal) { + val headerValue = header.fold(show""""${H.name}" was Undefined""")(_.show) + s"""Failure during header capture: $headerValue""" + } InternalServerError("Error processing request.").widen } } diff --git a/core/src/main/scala/org/http4s/rho/bits/Metadata.scala b/core/src/main/scala/org/http4s/rho/bits/Metadata.scala index 4ff041b5d..f3ed24eeb 100644 --- a/core/src/main/scala/org/http4s/rho/bits/Metadata.scala +++ b/core/src/main/scala/org/http4s/rho/bits/Metadata.scala @@ -2,6 +2,7 @@ package org.http4s package rho.bits import scala.reflect.runtime.universe.TypeTag +import org.typelevel.ci.CIString /** Base type for data that can be used to decorate the rules trees * @@ -31,4 +32,4 @@ case class QueryMetaData[F[_], T]( extends Metadata /** Metadata about a header rule */ -case class HeaderMetaData[T <: HeaderKey.Extractable](key: T, isRequired: Boolean) extends Metadata +case class HeaderMetaData[T](key: CIString, isRequired: Boolean) extends Metadata diff --git a/core/src/main/scala/org/http4s/rho/bits/ResultResponse.scala b/core/src/main/scala/org/http4s/rho/bits/ResultResponse.scala index b81e99b35..1ba57aa2e 100644 --- a/core/src/main/scala/org/http4s/rho/bits/ResultResponse.scala +++ b/core/src/main/scala/org/http4s/rho/bits/ResultResponse.scala @@ -2,7 +2,7 @@ package org.http4s.rho.bits import cats.data.OptionT import cats.{Applicative, Functor, Monad} -import cats.syntax.functor._ +import cats.implicits._ import org.http4s._ import org.http4s.rho.Result.BaseResult @@ -49,6 +49,11 @@ sealed trait ResultResponse[F[_], +T] extends RouteResult[F, T] { } } +object ResultResponse { + def fromEither[F[_]: Functor, R](e: Either[F[BaseResult[F]], R]): ResultResponse[F, R] = + e.fold(FailureResponse.result[F](_), SuccessResponse.apply _) +} + /** Successful response */ final case class SuccessResponse[F[_], +T](result: T) extends ResultResponse[F, T] diff --git a/core/src/test/scala/ApiExamples.scala b/core/src/test/scala/ApiExamples.scala index 84daead3f..e280910d5 100644 --- a/core/src/test/scala/ApiExamples.scala +++ b/core/src/test/scala/ApiExamples.scala @@ -86,7 +86,7 @@ class ApiExamples extends Specification { /// src_inlined HeaderCapture new RhoRoutes[IO] { - GET / "hello" >>> capture(ETag) |>> { tag: ETag => + GET / "hello" >>> H[ETag].capture |>> { tag: ETag => Ok(s"Thanks for the tag: $tag") } } @@ -95,8 +95,8 @@ class ApiExamples extends Specification { /// src_inlined HeaderRuleCombine new RhoRoutes[IO] { // Header rules are composable - val ensureLength = existsAnd(`Content-Length`)(_.length > 0) - val getTag = capture(ETag) + val ensureLength = H[`Content-Length`].existsAnd(_.length > 0) + val getTag = H[ETag].capture POST / "sequential" >>> getTag >>> ensureLength |>> { tag: ETag => Ok(s"Thanks for the $tag and the non-empty body!") @@ -116,8 +116,8 @@ class ApiExamples extends Specification { val path1 = "one" / pathVar[Int] val path2 = "two" / pathVar[Int] - val getLength = captureMap(`Content-Length`)(_.length) - val getTag = captureMap(ETag)(_ => -1L) + val getLength = H[`Content-Length`].captureMap(_.length) + val getTag = H[ETag].captureMap(_ => -1L) GET / (path1 || path2) +? param[String]("foo") >>> (getLength || getTag) |>> { (i: Int, foo: String, v: Long) => Ok(s"Received $i, $foo, $v") diff --git a/core/src/test/scala/org/http4s/rho/ApiTest.scala b/core/src/test/scala/org/http4s/rho/ApiTest.scala index f4d3f4ef8..2a5a8a2f5 100644 --- a/core/src/test/scala/org/http4s/rho/ApiTest.scala +++ b/core/src/test/scala/org/http4s/rho/ApiTest.scala @@ -14,6 +14,7 @@ import org.specs2.matcher.MatchResult import org.specs2.mutable._ import shapeless.{HList, HNil} import scala.util.control.NoStackTrace +import org.http4s.headers.Accept class ApiTest extends Specification { @@ -33,19 +34,19 @@ class ApiTest extends Specification { ETag(ETag.EntityTag("foo")) val RequireETag = - exists(ETag) + H[ETag].exists val RequireNonZeroLen = - existsAnd(headers.`Content-Length`)(h => h.length != 0) + H[`Content-Length`].existsAnd(h => h.length != 0) val RequireThrowException = - existsAnd(headers.`Content-Length`)(_ => + H[`Content-Length`].existsAnd(_ => throw new RuntimeException("this could happen") with NoStackTrace ) def fetchETag(p: IO[Response[IO]]): ETag = { val resp = p.unsafeRunSync() - resp.headers.get(ETag).getOrElse(sys.error("No ETag: " + resp)) + resp.headers.get[ETag].getOrElse(sys.error("No ETag: " + resp)) } def checkETag(p: OptionT[IO, Response[IO]], s: String): MatchResult[Any] = @@ -94,12 +95,12 @@ class ApiTest extends Specification { Seq( { - val c = captureOptionally(headers.`Content-Length`) + val c = H[`Content-Length`].captureOptionally ruleExecutor.runRequestRules(c.rule, req) should_== SuccessResponse( Some(lenheader) :: HNil ) }, { - val c = captureOptionally(ETag) + val c = H[ETag].captureOptionally ruleExecutor.runRequestRules(c.rule, req) should_== SuccessResponse(None :: HNil) } ).reduce(_ and _) @@ -109,10 +110,10 @@ class ApiTest extends Specification { val req = Request[IO]().putHeaders(etag, lenheader) Seq( { - val c2 = RequireETag && capture(headers.`Content-Length`) + val c2 = RequireETag && H[`Content-Length`].capture ruleExecutor.runRequestRules(c2.rule, req) should_== SuccessResponse(lenheader :: HNil) }, { - val c3 = capture(headers.`Content-Length`) && capture(ETag) + val c3 = H[`Content-Length`].capture && H[ETag].capture ruleExecutor.runRequestRules(c3.rule, req) should_== SuccessResponse( etag :: lenheader :: HNil ) @@ -122,7 +123,7 @@ class ApiTest extends Specification { "Capture params with default" in { val default: `Content-Length` = headers.`Content-Length`.unsafeFromLong(10) - val c = captureOrElse(headers.`Content-Length`)(default) + val c = H[`Content-Length`].captureOrElse(default) Seq( { val req = Request[IO]().putHeaders() @@ -136,14 +137,14 @@ class ApiTest extends Specification { "Map header params" in { val req = Request[IO]().putHeaders(etag, lenheader) - val c = captureMap(headers.`Content-Length`)(_.length) + val c = H[`Content-Length`].captureMap(_.length) ruleExecutor.runRequestRules(c.rule, req) should_== SuccessResponse(4 :: HNil) } "Map header params with exception" in { val req = Request[IO]().putHeaders(etag, lenheader) - val c = captureMap(headers.`Content-Length`)(_.length / 0) + val c = H[`Content-Length`].captureMap(_.length / 0) ruleExecutor.runRequestRules(c.rule, req) must beAnInstanceOf[FailureResponse[IO]] } @@ -151,14 +152,14 @@ class ApiTest extends Specification { "map simple header params into a complex type" in { case class Foo(age: Long, s: HttpDate) val paramFoo = - (captureMap(headers.`Content-Length`)(_.length) && captureMap(headers.Date)(_.date)) + (H[`Content-Length`].captureMap(_.length) && H[headers.Date].captureMap(_.date)) .map(Foo.apply _) val path = GET / "hello" >>> paramFoo val testDate = HttpDate.unsafeFromInstant(java.time.Instant.now) val req = Request[IO]( 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(testDate)) + headers = Headers(headers.`Content-Length`.unsafeFromLong(10), headers.Date(testDate)) ) val expectedFoo = Foo(10, testDate) @@ -172,11 +173,11 @@ class ApiTest extends Specification { "Map with missing header result" in { val req = Request[IO]().putHeaders(etag, lenheader) - val c1 = captureMapR(headers.`Content-Length`)(r => Right(r.length)) + val c1 = H[`Content-Length`].captureMapR(r => Right(r.length)) ruleExecutor.runRequestRules(c1.rule, req) should_== SuccessResponse(4 :: HNil) val r2 = Gone("Foo") - val c2 = captureMapR(headers.`Content-Length`)(_ => Left(r2)) + val c2 = H[`Content-Length`].captureMapR(_ => Left(r2)) val v1 = ruleExecutor.runRequestRules(c2.rule, req) v1 must beAnInstanceOf[FailureResponse[IO]] v1.asInstanceOf[FailureResponse[IO]].toResponse.unsafeRunSync().status must_== r2 @@ -184,7 +185,7 @@ class ApiTest extends Specification { .resp .status - val c3 = captureMapR(headers.Accept, Some(r2))(_ => ???) + val c3 = H[Accept].captureMapR(Some(r2))(_ => ???) val v2 = ruleExecutor.runRequestRules(c3.rule, req) v2 must beAnInstanceOf[FailureResponse[IO]] v2.asInstanceOf[FailureResponse[IO]].toResponse.unsafeRunSync().status must_== r2 @@ -195,10 +196,10 @@ class ApiTest extends Specification { "Append headers to a Route" in { val path = POST / "hello" / pv"world" +? param[Int]("fav") - val validations = existsAnd(headers.`Content-Length`)(h => h.length != 0) + val validations = H[`Content-Length`].existsAnd(h => h.length != 0) val route = - runWith((path >>> validations >>> capture(ETag)).decoding(EntityDecoder.text[IO])) { + runWith((path >>> validations >>> H[ETag].capture).decoding(EntityDecoder.text[IO])) { (world: String, fav: Int, tag: ETag, body: String) => Ok(s"Hello to you too, $world. Your Fav number is $fav. You sent me $body") .map(_.putHeaders(tag)) @@ -212,20 +213,20 @@ class ApiTest extends Specification { .withEntity("cool") val resp = route(req).value.unsafeRunSync().getOrElse(Response.notFound) - resp.headers.get(ETag) must beSome(etag) + resp.headers.get[ETag] must beSome(etag) } "accept compound or sequential header rules" in { val path = POST / "hello" / pv"world" - val lplus1 = captureMap(headers.`Content-Length`)(_.length + 1) + val lplus1 = H[`Content-Length`].captureMap(_.length + 1) - val route1 = runWith((path >>> lplus1 >>> capture(ETag)).decoding(EntityDecoder.text)) { + val route1 = runWith((path >>> lplus1 >>> H[ETag].capture).decoding(EntityDecoder.text)) { (_: String, _: Long, _: ETag, _: String) => Ok("") } - val route2 = runWith((path >>> (lplus1 && capture(ETag))).decoding(EntityDecoder.text)) { + val route2 = runWith((path >>> (lplus1 && H[ETag].capture)).decoding(EntityDecoder.text)) { (_: String, _: Long, _: ETag, _: String) => Ok("") } @@ -259,8 +260,8 @@ class ApiTest extends Specification { "Execute a complicated route" in { val path = POST / "hello" / pv"world" +? param[Int]("fav") - val validations = existsAnd(headers.`Content-Length`)(h => h.length != 0) && - capture(ETag) + val validations = H[`Content-Length`].existsAnd(h => h.length != 0) && + H[ETag].capture val route = runWith((path >>> validations).decoding(EntityDecoder.text)) { @@ -284,7 +285,7 @@ class ApiTest extends Specification { 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 + result.headers.headers.size must_== 0 result.status must_== Status.SwitchingProtocols } } @@ -415,7 +416,7 @@ class ApiTest extends Specification { "Decoders" should { "Decode a body" in { - val reqHeader = existsAnd(headers.`Content-Length`)(h => h.length < 10) + val reqHeader = H[`Content-Length`].existsAnd(h => h.length < 10) val path = POST / "hello" >>> reqHeader @@ -455,7 +456,7 @@ class ApiTest extends Specification { val req = Request[IO](uri = uri("/hello")) .putHeaders(headers.`Content-Length`.unsafeFromLong("foo".length)) - val reqHeader = existsAnd(headers.`Content-Length`)(h => h.length < 2) + val reqHeader = H[`Content-Length`].existsAnd(h => h.length < 2) val route1 = runWith(path.validate(reqHeader)) { () => Ok("shouldn't get here.") } @@ -465,7 +466,7 @@ class ApiTest extends Specification { .getOrElse(Response.notFound) .status should_== Status.BadRequest - val reqHeaderR = existsAndR(headers.`Content-Length`)(_ => Some(Unauthorized("Foo."))) + val reqHeaderR = H[`Content-Length`].existsAndR(_ => Some(Unauthorized("Foo."))) val route2 = runWith(path.validate(reqHeaderR)) { () => Ok("shouldn't get here.") } diff --git a/core/src/test/scala/org/http4s/rho/AuthedContextSpec.scala b/core/src/test/scala/org/http4s/rho/AuthedContextSpec.scala index cf38f0a5e..d15a40173 100644 --- a/core/src/test/scala/org/http4s/rho/AuthedContextSpec.scala +++ b/core/src/test/scala/org/http4s/rho/AuthedContextSpec.scala @@ -50,7 +50,7 @@ class AuthedContextSpec extends Specification { "AuthedContext execution" should { "Be able to have access to authInfo" in { - val request = Request[IO](Method.GET, Uri(path = Uri.Path.fromString("/"))) + val request = Request[IO](Method.GET, uri"/") val resp = routes.run(request).value.unsafeRunSync().getOrElse(Response.notFound) if (resp.status == Status.Ok) { val body = @@ -60,7 +60,7 @@ class AuthedContextSpec extends Specification { } "Does not prevent route from being executed without authentication" in { - val request = Request[IO](Method.GET, Uri(path = Uri.Path.fromString("/public/public"))) + val request = Request[IO](Method.GET, uri"/public/public") val resp = routes.run(request).value.unsafeRunSync().getOrElse(Response.notFound) if (resp.status == Status.Ok) { val body = @@ -70,7 +70,7 @@ class AuthedContextSpec extends Specification { } "Does not prevent route from being executed without authentication, but allows to extract it" in { - val request = Request[IO](Method.GET, Uri(path = Uri.Path.fromString("/private/private"))) + val request = Request[IO](Method.GET, uri"/private/private") val resp = routes.run(request).value.unsafeRunSync().getOrElse(Response.notFound) if (resp.status == Status.Ok) { val body = diff --git a/core/src/test/scala/org/http4s/rho/CodecRouterSpec.scala b/core/src/test/scala/org/http4s/rho/CodecRouterSpec.scala index c833e4094..85c81e468 100644 --- a/core/src/test/scala/org/http4s/rho/CodecRouterSpec.scala +++ b/core/src/test/scala/org/http4s/rho/CodecRouterSpec.scala @@ -6,7 +6,6 @@ import fs2.Stream import org.specs2.mutable.Specification import scala.collection.compat.immutable.ArraySeq -import org.http4s.Uri.Path class CodecRouterSpec extends Specification { @@ -27,8 +26,8 @@ class CodecRouterSpec extends Specification { "Decode a valid body" in { 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 = Path.fromString("/foo")), headers = h, body = b) + val h = Headers(headers.`Content-Type`(MediaType.text.plain)) + val req = Request[IO](Method.POST, uri"/foo", headers = h, body = b) val result = routes(req).value.unsafeRunSync().getOrElse(Response.notFound) val (bb, s) = bodyAndStatus(result) @@ -38,8 +37,9 @@ class CodecRouterSpec extends Specification { "Fail on invalid body" in { 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 = Path.fromString("/form")), headers = h, body = b) + val h = Headers(headers.`Content-Type`(MediaType.application.`x-www-form-urlencoded`)) + val req = + Request[IO](Method.POST, uri"/form", headers = h, body = b) routes(req).value.unsafeRunSync().map(_.status) must be some Status.BadRequest } diff --git a/core/src/test/scala/org/http4s/rho/CompileRoutesSpec.scala b/core/src/test/scala/org/http4s/rho/CompileRoutesSpec.scala index 77ffb1a96..f05b59aae 100644 --- a/core/src/test/scala/org/http4s/rho/CompileRoutesSpec.scala +++ b/core/src/test/scala/org/http4s/rho/CompileRoutesSpec.scala @@ -3,9 +3,8 @@ package org.http4s.rho import cats.effect.IO import org.http4s.rho.bits.MethodAliases._ import org.http4s.rho.io._ -import org.http4s.{Method, Request, Uri} +import org.http4s.{Method, Request} import org.specs2.mutable.Specification -import org.http4s.Uri.Path class CompileRoutesSpec extends Specification { @@ -20,7 +19,7 @@ class CompileRoutesSpec extends Specification { val c = RoutesBuilder[IO]() getFoo(c) - "GetFoo" === RRunner(c.toRoutes()).checkOk(Request(uri=Uri(path=Path.fromString("/hello")))) + "GetFoo" === RRunner(c.toRoutes()).checkOk(Request(uri = uri"/hello")) } "Build multiple routes" in { @@ -28,8 +27,8 @@ class CompileRoutesSpec extends Specification { getFoo(c) putFoo(c) - "GetFoo" === RRunner(c.toRoutes()).checkOk(Request(uri=Uri(path=Path.fromString("/hello")))) - "PutFoo" === RRunner(c.toRoutes()).checkOk(Request(method = Method.PUT, uri=Uri(path=Path.fromString("/hello")))) + "GetFoo" === RRunner(c.toRoutes()).checkOk(Request(uri = uri"/hello")) + "PutFoo" === RRunner(c.toRoutes()).checkOk(Request(method = Method.PUT, uri = uri"/hello")) } "Make routes from a collection of RhoRoutes" in { @@ -39,8 +38,8 @@ class CompileRoutesSpec extends Specification { (PUT / "hello" |>> "PutFoo") :: Nil val srvc = CompileRoutes.foldRoutes[IO](routes) - "GetFoo" === RRunner(srvc).checkOk(Request(uri=Uri(path=Path.fromString("/hello")))) - "PutFoo" === RRunner(srvc).checkOk(Request(method = Method.PUT, uri=Uri(path=Path.fromString("/hello")))) + "GetFoo" === RRunner(srvc).checkOk(Request(uri = uri"/hello")) + "PutFoo" === RRunner(srvc).checkOk(Request(method = Method.PUT, uri = uri"/hello")) } "Concatenate correctly" in { @@ -48,8 +47,8 @@ class CompileRoutesSpec extends Specification { val c2 = RoutesBuilder[IO](); putFoo(c2) val srvc = c1.append(c2.routes()).toRoutes() - "GetFoo" === RRunner(srvc).checkOk(Request(uri=Uri(path=Path.fromString("/hello")))) - "PutFoo" === RRunner(srvc).checkOk(Request(method = Method.PUT, uri=Uri(path=Path.fromString("/hello")))) + "GetFoo" === RRunner(srvc).checkOk(Request(uri = uri"/hello")) + "PutFoo" === RRunner(srvc).checkOk(Request(method = Method.PUT, uri = uri"/hello")) } } diff --git a/core/src/test/scala/org/http4s/rho/ParamDefaultValueSpec.scala b/core/src/test/scala/org/http4s/rho/ParamDefaultValueSpec.scala index 07da32cab..87bd7e4fa 100644 --- a/core/src/test/scala/org/http4s/rho/ParamDefaultValueSpec.scala +++ b/core/src/test/scala/org/http4s/rho/ParamDefaultValueSpec.scala @@ -20,11 +20,11 @@ class ParamDefaultValueSpec extends Specification { .foldLeft(Array[Byte]())(_ :+ _) ) - def requestGet(s: String, h: Header*): Request[IO] = + def requestGet(s: String, h: Header.ToRaw*): Request[IO] = Request( bits.MethodAliases.GET, Uri.fromString(s).getOrElse(sys.error("Failed.")), - headers = Headers.of(h: _*) + headers = Headers(h: _*) ) "GET /test1" should { diff --git a/core/src/test/scala/org/http4s/rho/ResultSpec.scala b/core/src/test/scala/org/http4s/rho/ResultSpec.scala index 1e22862ae..fd70a56bf 100644 --- a/core/src/test/scala/org/http4s/rho/ResultSpec.scala +++ b/core/src/test/scala/org/http4s/rho/ResultSpec.scala @@ -17,14 +17,14 @@ class ResultSpec extends Specification { .unsafeRunSync() .resp - resp.headers.get(Date) must beSome(date) + resp.headers.get[Date] must beSome(date) val respNow = Ok("Foo") .map(_.putHeaders(date)) .unsafeRunSync() .resp - respNow.headers.get(Date) must beSome(date) + respNow.headers.get[Date] must beSome(date) } "Add atributes" in { diff --git a/core/src/test/scala/org/http4s/rho/RhoRoutesSpec.scala b/core/src/test/scala/org/http4s/rho/RhoRoutesSpec.scala index 9e293c12a..0fe91b8e6 100644 --- a/core/src/test/scala/org/http4s/rho/RhoRoutesSpec.scala +++ b/core/src/test/scala/org/http4s/rho/RhoRoutesSpec.scala @@ -16,11 +16,11 @@ import org.typelevel.ci.CIString import scala.util.control.NoStackTrace class RhoRoutesSpec extends Specification with RequestRunner { - def construct(method: Method, s: String, h: Header*): Request[IO] = - Request(method, Uri.fromString(s).getOrElse(sys.error("Failed.")), headers = Headers.of(h: _*)) + def construct(method: Method, s: String, h: Header.ToRaw*): Request[IO] = + Request(method, Uri.fromString(s).getOrElse(sys.error("Failed.")), headers = Headers(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: _*) + def Get(s: String, h: Header.ToRaw*): Request[IO] = construct(Method.GET, s, h: _*) + def Put(s: String, h: Header.ToRaw*): Request[IO] = construct(Method.PUT, s, h: _*) val httpRoutes = new RhoRoutes[IO] { GET +? param("foo", "bar") |>> { foo: String => Ok(s"just root with parameter 'foo=$foo'") } @@ -98,7 +98,7 @@ class RhoRoutesSpec extends Specification with RequestRunner { "RhoRoutes execution" should { "Handle definition without a path, which points to '/'" in { - val request = Request[IO](Method.GET, Uri(path = Path.fromString("/"))) + val request = Request[IO](Method.GET, uri"/") checkOk(request) should_== "just root with parameter 'foo=bar'" } @@ -112,7 +112,9 @@ class RhoRoutesSpec extends Specification with RequestRunner { val resp = httpRoutes(request).value.unsafeRunSync().getOrElse(Response.notFound) resp.status must_== Status.MethodNotAllowed - resp.headers.get(CIString("Allow")) must beSome(Header.Raw(CIString("Allow"), "GET")) + resp.headers.get(CIString("Allow")).map(_.toList).toList.flatten must contain( + Header.Raw(CIString("Allow"), "GET") + ) } "Yield `MethodNotAllowed` when invalid method used" in { @@ -127,10 +129,10 @@ class RhoRoutesSpec extends Specification with RequestRunner { GET / "" / "foo" |>> Ok("bar") }.toRoutes() - val req1 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo"))) + val req1 = Request[IO](Method.GET, Uri(path = Path.unsafeFromString("/foo"))) getBody(service(req1).value.unsafeRunSync().getOrElse(Response.notFound).body) should_== "bar" - val req2 = Request[IO](Method.GET, Uri(path = Path.fromString("//foo"))) + val req2 = Request[IO](Method.GET, Uri(path = Path.unsafeFromString("//foo"))) getBody(service(req2).value.unsafeRunSync().getOrElse(Response.notFound).body) should_== "bar" } @@ -256,21 +258,21 @@ class RhoRoutesSpec extends Specification with RequestRunner { } "Level one path definition to /some" in { - val req1 = Request[IO](Method.GET, Uri(path = Path.fromString("/some"))) + val req1 = Request[IO](Method.GET, uri"/some") checkOk(req1) should_== "root to some" } "Execute a directly provided Task every invocation" in { - val req = Request[IO](Method.GET, Uri(path = Path.fromString("directTask"))) + val req = Request[IO](Method.GET, Uri(path = Path.unsafeFromString("directTask"))) checkOk(req) should_== "0" checkOk(req) should_== "1" } "Interpret uris ending in '/' differently than those without" in { - val req1 = Request[IO](Method.GET, Uri(path = Path.fromString("terminal/"))) + val req1 = Request[IO](Method.GET, Uri(path = Path.unsafeFromString("terminal/"))) checkOk(req1) should_== "terminal/" - val req2 = Request[IO](Method.GET, Uri(path = Path.fromString("terminal"))) + val req2 = Request[IO](Method.GET, Uri(path = Path.unsafeFromString("terminal"))) checkOk(req2) should_== "terminal" } @@ -282,17 +284,17 @@ class RhoRoutesSpec extends Specification with RequestRunner { GET / "foo" |>> Ok("none") }.toRoutes() - val req1 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo")).+?("bar", "0")) + val req1 = Request[IO](Method.GET, Uri(path = Path.unsafeFromString("/foo")).+?("bar", "0")) getBody( service(req1).value.unsafeRunSync().getOrElse(Response.notFound).body ) must_== "Int: 0" - val req2 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo")).+?("bar", "s")) + val req2 = Request[IO](Method.GET, Uri(path = Path.unsafeFromString("/foo")).+?("bar", "s")) getBody( service(req2).value.unsafeRunSync().getOrElse(Response.notFound).body ) must_== "String: s" - val req3 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo"))) + val req3 = Request[IO](Method.GET, Uri(path = Path.unsafeFromString("/foo"))) getBody(service(req3).value.unsafeRunSync().getOrElse(Response.notFound).body) must_== "none" } @@ -302,12 +304,12 @@ class RhoRoutesSpec extends Specification with RequestRunner { GET / "foo" +? param[Int]("bar") |>> { i: Int => Ok(s"Int: $i") } }.toRoutes() - val req1 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo")).+?("bar", "0")) + val req1 = Request[IO](Method.GET, Uri(path = Path.unsafeFromString("/foo")).+?("bar", "0")) getBody( service(req1).value.unsafeRunSync().getOrElse(Response.notFound).body ) must_== "String: 0" - val req2 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo")).+?("bar", "s")) + val req2 = Request[IO](Method.GET, Uri(path = Path.unsafeFromString("/foo")).+?("bar", "s")) getBody( service(req2).value.unsafeRunSync().getOrElse(Response.notFound).body ) must_== "String: s" @@ -322,17 +324,17 @@ class RhoRoutesSpec extends Specification with RequestRunner { GET / "foo" |>> Ok(s"failure") }.toRoutes() - val req1 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo")).+?("bar", "s")) + val req1 = Request[IO](Method.GET, Uri(path = Path.unsafeFromString("/foo")).+?("bar", "s")) getBody( service(req1).value.unsafeRunSync().getOrElse(Response.notFound).body ) must_== "String: s" - val req2 = Request[IO](Method.GET, Uri(path = Path.fromString("/foo"))) + val req2 = Request[IO](Method.GET, Uri(path = Path.unsafeFromString("/foo"))) getBody(service(req2).value.unsafeRunSync().getOrElse(Response.notFound).body) must_== "none" } "work with all syntax elements" in { - val reqHeader = existsAnd(headers.`Content-Length`)(h => h.length <= 3) + val reqHeader = H[`Content-Length`].existsAnd(h => h.length <= 3) val srvc = new RhoRoutes[IO] { POST / "foo" / pathVar[Int] +? param[String]("param") >>> reqHeader ^ EntityDecoder @@ -361,7 +363,7 @@ class RhoRoutesSpec extends Specification with RequestRunner { throw new Error("an error") with NoStackTrace; Ok("Wont get here...") } }.toRoutes() - val req = Request[IO](Method.GET, Uri(path = Path.fromString("/error"))) + val req = Request[IO](Method.GET, Uri(path = Path.unsafeFromString("/error"))) service(req).value.unsafeRunSync().getOrElse(Response.notFound).status must equalTo( Status.InternalServerError @@ -370,7 +372,7 @@ class RhoRoutesSpec extends Specification with RequestRunner { "give a None for missing route" in { val service = new RhoRoutes[IO] {}.toRoutes() - val req = Request[IO](Method.GET, Uri(path = Path.fromString("/missing"))) + val req = Request[IO](Method.GET, Uri(path = Path.unsafeFromString("/missing"))) service(req).value.unsafeRunSync().getOrElse(Response.notFound).status must_== Status.NotFound } } @@ -404,7 +406,7 @@ class RhoRoutesSpec extends Specification with RequestRunner { val routes2: HttpRoutes[IO] = ("foo" /: routes1).toRoutes() - val req1 = Request[IO](uri = Uri(path = Path.fromString("/foo/bar"))) + val req1 = Request[IO](uri = Uri(path = Path.unsafeFromString("/foo/bar"))) getBody(routes2(req1).value.unsafeRunSync().getOrElse(Response.notFound).body) === "bar" } } diff --git a/core/src/test/scala/org/http4s/rho/UriConvertibleSpec.scala b/core/src/test/scala/org/http4s/rho/UriConvertibleSpec.scala index 38274c455..364378a20 100644 --- a/core/src/test/scala/org/http4s/rho/UriConvertibleSpec.scala +++ b/core/src/test/scala/org/http4s/rho/UriConvertibleSpec.scala @@ -14,7 +14,7 @@ object UriConvertibleSpec extends Specification { "UriConvertible.respectPathInfo" should { "respect if URI template is available" in { val request = Request[IO]( - uri = Uri(path = Uri.Path.fromString("/some")), + uri = uri"/some", attributes = Vault.empty.insert(Request.Keys.PathInfoCaret, 5) ) val path = List(PathElm("here")) @@ -35,7 +35,7 @@ object UriConvertibleSpec extends Specification { "UriConvertible.addPathInfo" should { "keep the path if PathInfoCaret is not available" in { - val request = Request[IO](uri = Uri(path = Uri.Path.fromString("/some"))) + val request = Request[IO](uri = uri"/some") val path = List(PathElm("here")) val query = List(ParamVarExp("ref", "path")) val tpl = UriTemplate(path = path, query = query) @@ -44,7 +44,7 @@ object UriConvertibleSpec extends Specification { } "keep the path if PathInfoCaret is 0" in { val request = Request[IO]( - uri = Uri(path = Uri.Path.fromString("/some")), + uri = uri"/some", attributes = Vault.empty.insert(Request.Keys.PathInfoCaret, 0) ) val path = List(PathElm("here")) @@ -55,7 +55,7 @@ object UriConvertibleSpec extends Specification { } "keep the path if PathInfoCaret is 1" in { val request = Request[IO]( - uri = Uri(path = Uri.Path.fromString("/some")), + uri = uri"/some", attributes = Vault.empty.insert(Request.Keys.PathInfoCaret, 1) ) val path = List(PathElm("here")) @@ -66,7 +66,7 @@ object UriConvertibleSpec extends Specification { } "manipulate the path if PathInfoCaret greater than 1" in { val request = Request[IO]( - uri = Uri(path = Uri.Path.fromString("/some")), + uri = uri"/some", attributes = Vault.empty.insert(Request.Keys.PathInfoCaret, 5) ) val path = List(PathElm("here")) 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 e04996cf1..04ffbec6c 100644 --- a/core/src/test/scala/org/http4s/rho/bits/HListToFuncSpec.scala +++ b/core/src/test/scala/org/http4s/rho/bits/HListToFuncSpec.scala @@ -13,11 +13,11 @@ class HListToFuncSpec extends Specification { service(r).value.unsafeRunSync().getOrElse(Response.notFound).body ) - def Get(s: String, h: Header*): Request[IO] = + def Get(s: String, h: Header.ToRaw*): Request[IO] = Request( bits.MethodAliases.GET, Uri.fromString(s).getOrElse(sys.error("Failed.")), - headers = Headers.of(h: _*) + headers = Headers(h: _*) ) val service = new RhoRoutes[IO] { 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 547c97b66..063acf49e 100644 --- a/core/src/test/scala/org/http4s/rho/bits/PathTreeSpec.scala +++ b/core/src/test/scala/org/http4s/rho/bits/PathTreeSpec.scala @@ -6,10 +6,9 @@ import java.nio.charset.StandardCharsets import cats.effect.IO import org.specs2.mutable.Specification -import org.http4s.Uri.uri +import org.http4s.Uri._ import org.http4s.server.middleware.TranslateUri import org.http4s.server.Router -import org.http4s.Uri.Path class PathTreeSpec extends Specification { object pathTree extends PathTreeOps[IO] @@ -46,7 +45,7 @@ class PathTreeSpec extends Specification { GET / "foo" |>> "foo" }.toRoutes())).orNotFound - val req = Request[IO](Method.GET, uri = Uri(path = Path.fromString("/bar/foo"))) + val req = Request[IO](Method.GET, uri = uri"/bar/foo") val resp = svc(req).unsafeRunSync() resp.status must_== Status.Ok 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 b2c9c8599..fc6627709 100644 --- a/core/src/test/scala/org/http4s/rho/bits/ResponseGeneratorSpec.scala +++ b/core/src/test/scala/org/http4s/rho/bits/ResponseGeneratorSpec.scala @@ -16,12 +16,12 @@ class ResponseGeneratorSpec extends Specification { new String(resp.body.compile.toVector.unsafeRunSync().foldLeft(Array[Byte]())(_ :+ _)) str must_== "Foo" - resp.headers.get(`Content-Length`) must beSome( + resp.headers.get[`Content-Length`] must beSome( `Content-Length`.unsafeFromLong("Foo".getBytes.length) ) - resp.headers - .filterNot(_.is(`Content-Length`)) - .toList must_== EntityEncoder.stringEncoder.headers.toList + resp.headers.headers + .filterNot(_.name == `Content-Length`.name) + .toList must_== EntityEncoder.stringEncoder.headers.headers.toList } "Build a response without a body" in { @@ -30,8 +30,8 @@ class ResponseGeneratorSpec extends Specification { resp.body.compile.toVector.unsafeRunSync().length must_== 0 resp.status must_== Status.SwitchingProtocols - resp.headers.get(`Content-Length`) must beNone - resp.headers.get(`Transfer-Encoding`) must beNone + resp.headers.get[`Content-Length`] must beNone + resp.headers.get[`Transfer-Encoding`] must beNone } "Build a redirect response" in { @@ -41,9 +41,9 @@ class ResponseGeneratorSpec extends Specification { resp.body.compile.toVector.unsafeRunSync().length must_== 0 resp.status must_== Status.MovedPermanently - resp.headers.get(`Content-Length`) must beNone - resp.headers.get(`Transfer-Encoding`) must beNone - resp.headers.get(Location) must beSome(Location(location)) + resp.headers.get[`Content-Length`] must beNone + resp.headers.get[`Transfer-Encoding`] must beNone + resp.headers.get[Location] must beSome(Location(location)) } "Build a redirect response with a body" in { @@ -54,11 +54,11 @@ class ResponseGeneratorSpec extends Specification { EntityDecoder[IO, String].decode(resp, false).value.unsafeRunSync() must beRight(testBody) resp.status must_== Status.MovedPermanently - resp.headers.get(`Content-Length`) must beSome( + resp.headers.get[`Content-Length`] must beSome( `Content-Length`.unsafeFromLong(testBody.length) ) - resp.headers.get(`Transfer-Encoding`) must beNone - resp.headers.get(Location) must beSome(Location(location)) + resp.headers.get[`Transfer-Encoding`] must beNone + resp.headers.get[Location] must beSome(Location(location)) } "Explicitly added headers have priority" in { @@ -67,11 +67,11 @@ class ResponseGeneratorSpec extends Specification { EntityEncoder.stringEncoder[IO].toEntity(_) ) - Ok("some content", Headers.of(`Content-Type`(MediaType.application.json))) + Ok("some content", Headers(`Content-Type`(MediaType.application.json))) .unsafeRunSync() .resp .headers - .get(`Content-Type`) + .get[`Content-Type`] .get must_== `Content-Type`(MediaType.application.json) } } 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 deleted file mode 100644 index d40b7eeb0..000000000 --- a/examples/src/main/scala/com/http4s/rho/swagger/demo/JsonEncoder.scala +++ /dev/null @@ -1,26 +0,0 @@ -package com.http4s.rho.swagger.demo - -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._ - import org.json4s.jackson.Serialization - import org.json4s.jackson.Serialization.write - - trait AutoSerializable extends AnyRef with Product - - private implicit val formats: Formats = - Serialization.formats(NoTypeHints) - - 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(ArraySeq.unsafeWrapArray(bytes)), Some(bytes.length)) - } -} 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 3f0e65f89..fd51227f9 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 @@ -3,30 +3,32 @@ package com.http4s.rho.swagger.demo import cats.Monad import cats.effect._ import cats.implicits._ -import com.http4s.rho.swagger.demo.JsonEncoder.{AutoSerializable, _} import com.http4s.rho.swagger.demo.MyRoutes._ import fs2.Stream import org.http4s.rho.RhoRoutes import org.http4s.rho.bits._ import org.http4s.rho.swagger.{SwaggerFileResponse, SwaggerSyntax} -import org.http4s.{EntityDecoder, Headers, HttpDate, Request, ResponseCookie, Uri, headers} +import org.http4s.{EntityDecoder, Headers, HttpDate, Request, ResponseCookie, headers} import shapeless.HNil +import org.http4s.circe.CirceEntityEncoder +import org.http4s.implicits._ -class MyRoutes[F[+_]: Effect](swaggerSyntax: SwaggerSyntax[F]) extends RhoRoutes[F] { - +class MyRoutes[F[+_]: Effect](swaggerSyntax: SwaggerSyntax[F]) + extends RhoRoutes[F] + with CirceEntityEncoder { import swaggerSyntax._ - val requireCookie: TypedHeader[F, HNil] = existsAndR(headers.Cookie) { cookie => + val requireCookie: TypedHeader[F, HNil] = H[headers.Cookie].existsAndR { cookie => cookie.values.toList.find(c => c.name == "Foo" && c.content == "bar") match { case Some(_) => // Cookie found, good to go None case None => // Didn't find cookie - Some(TemporaryRedirect(Uri(path = Uri.Path.fromString("/addcookie"))).widen) + Some(TemporaryRedirect(uri"/addcookie").widen) } } "We don't want to have a real 'root' route anyway... " ** - GET |>> TemporaryRedirect(Uri(path = Uri.Path.fromString("/swagger-ui"))) + GET |>> TemporaryRedirect(uri"/swagger-ui") // We want to define this chunk of the service as abstract for reuse below val hello = "hello" @@ GET / "hello" @@ -85,7 +87,7 @@ class MyRoutes[F[+_]: Effect](swaggerSyntax: SwaggerSyntax[F]) extends RhoRoutes "Clears the cookies" ** "cookies" @@ GET / "clearcookies" |>> { req: Request[F] => - val hs = req.headers.get(headers.Cookie) match { + val hs = req.headers.get[headers.Cookie] match { case None => Headers.empty case Some(cookie) => Headers(cookie.values.toList.map { c => @@ -149,5 +151,6 @@ object MyRoutes { implicit def barParser[F[_]: Monad]: StringParser[F, Bar] = StringParser.intParser[F].map(Bar) - case class JsonResult(name: String, number: Int) extends AutoSerializable + case class JsonResult(name: String, number: Int) + implicit val jsonResultCodec: io.circe.Codec[JsonResult] = io.circe.generic.semiauto.deriveCodec } diff --git a/project/Dependencies.scala b/project/Dependencies.scala index 573fabf7c..ddd93cb3e 100644 --- a/project/Dependencies.scala +++ b/project/Dependencies.scala @@ -3,20 +3,22 @@ import Keys._ // format: off object Dependencies { - lazy val http4sVersion = "0.22.0-RC1" - lazy val specs2Version = "4.10.6" + val http4sVersion = "0.22.0-RC1" + val specs2Version = "4.10.6" + val circeVersion = "0.12.3" val scala_213 = "2.13.4" val scala_212 = "2.12.13" + + lazy val circeCore = "io.circe" %% "circe-core" % circeVersion + lazy val circeGeneric = "io.circe" %% "circe-generic" % circeVersion + lazy val circeParser = "io.circe" %% "circe-parser" % circeVersion lazy val http4sServer = "org.http4s" %% "http4s-server" % http4sVersion lazy val http4sDSL = "org.http4s" %% "http4s-dsl" % http4sVersion lazy val http4sBlaze = "org.http4s" %% "http4s-blaze-server" % http4sVersion - lazy val http4sJetty = "org.http4s" %% "http4s-servlet" % http4sVersion - lazy val http4sJson4sJackson = "org.http4s" %% "http4s-json4s-jackson" % http4sVersion + lazy val http4sCirce = "org.http4s" %% "http4s-circe" % http4sVersion lazy val http4sXmlInstances = "org.http4s" %% "http4s-scala-xml" % http4sVersion - lazy val json4s = "org.json4s" %% "json4s-ext" % "3.6.11" - lazy val json4sJackson = "org.json4s" %% "json4s-jackson" % json4s.revision lazy val swaggerModels = "io.swagger" % "swagger-models" % "1.6.2" lazy val swaggerCore = "io.swagger" % "swagger-core" % swaggerModels.revision lazy val logbackClassic = "ch.qos.logback" % "logback-classic" % "1.2.3" @@ -33,26 +35,29 @@ object Dependencies { val silencerVersion = "1.7.4" lazy val silencerPlugin = compilerPlugin("com.github.ghik" % "silencer-plugin" % silencerVersion cross CrossVersion.full) lazy val silencerLib = "com.github.ghik" % "silencer-lib" % silencerVersion % Provided cross CrossVersion.full + lazy val kindProjector = compilerPlugin("org.typelevel" % "kind-projector" % "0.11.3" cross CrossVersion.full) - lazy val halDeps = libraryDependencies ++= Seq(json4sJackson) + lazy val halDeps = libraryDependencies ++= Seq(http4sCirce) lazy val swaggerDeps = libraryDependencies ++= Seq( scalaXml, swaggerCore, swaggerModels, - json4s % "test", - json4sJackson % "test" + http4sCirce % "test" ) lazy val swaggerUiDeps = libraryDependencies ++= Seq(swaggerUi) lazy val exampleDeps = libraryDependencies ++= Seq( + circeCore, + circeGeneric, + circeParser, http4sBlaze, http4sDSL, - json4s, - json4sJackson, - http4sJson4sJackson, - uadetector + http4sCirce, + http4sXmlInstances, + logbackClassic, + uadetector, ) } diff --git a/swagger/src/main/scala/org/http4s/rho/swagger/SwaggerModelsBuilder.scala b/swagger/src/main/scala/org/http4s/rho/swagger/SwaggerModelsBuilder.scala index b933009f6..8ffb3404c 100644 --- a/swagger/src/main/scala/org/http4s/rho/swagger/SwaggerModelsBuilder.scala +++ b/swagger/src/main/scala/org/http4s/rho/swagger/SwaggerModelsBuilder.scala @@ -13,6 +13,7 @@ import scala.collection.immutable.ListMap import scala.collection.immutable.Seq import scala.reflect.runtime.universe._ import scala.util.control.NonFatal +import org.typelevel.ci.CIString private[swagger] class SwaggerModelsBuilder[F[_]](formats: SwaggerFormats)(implicit st: ShowType, @@ -443,8 +444,8 @@ private[swagger] class SwaggerModelsBuilder[F[_]](formats: SwaggerFormats)(impli } } - def mkHeaderParam(key: HeaderKey.Extractable, isRequired: Boolean): HeaderParameter = - HeaderParameter(`type` = "string", name = key.name.toString.some, required = isRequired) + def mkHeaderParam(key: CIString, isRequired: Boolean): HeaderParameter = + HeaderParameter(`type` = "string", name = key.toString.some, required = isRequired) def linearizeRoute(rr: RhoRoute[F, _]): List[LinearRoute] = { 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 c9ef2f839..c02076a23 100644 --- a/swagger/src/main/scala/org/http4s/rho/swagger/SwaggerSupport.scala +++ b/swagger/src/main/scala/org/http4s/rho/swagger/SwaggerSupport.scala @@ -68,7 +68,7 @@ abstract class SwaggerSupport[F[_]](implicit F: Sync[F], etag: WeakTypeTag[F[_]] .mapper() .writerWithDefaultPrettyPrinter() .writeValueAsString(swagger.toJModel), - Headers.of(`Content-Type`(MediaType.application.json)) + Headers(`Content-Type`(MediaType.application.json)) ) "Swagger documentation (YAML)" ** GET / yamlApiPath |>> (() => yamlResponse) @@ -79,7 +79,7 @@ abstract class SwaggerSupport[F[_]](implicit F: Sync[F], etag: WeakTypeTag[F[_]] .mapper() .writerWithDefaultPrettyPrinter() .writeValueAsString(swagger.toJModel), - Headers.of(`Content-Type`(MediaType.text.yaml)) + Headers(`Content-Type`(MediaType.text.yaml)) ) } } From c034c67da1bde85f48bdccc7ce88b44e22d8f31b Mon Sep 17 00:00:00 2001 From: "Ross A. Baker" Date: Sun, 23 May 2021 17:59:33 -0400 Subject: [PATCH 4/6] errorParsingHeaderResponse takes an Option[NEL] --- .../main/scala/org/http4s/rho/RhoDslHeaderExtractors.scala | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/core/src/main/scala/org/http4s/rho/RhoDslHeaderExtractors.scala b/core/src/main/scala/org/http4s/rho/RhoDslHeaderExtractors.scala index fb8442e54..065e61d8f 100644 --- a/core/src/main/scala/org/http4s/rho/RhoDslHeaderExtractors.scala +++ b/core/src/main/scala/org/http4s/rho/RhoDslHeaderExtractors.scala @@ -151,7 +151,7 @@ trait RhoDslHeaderExtractors[F[_]] extends FailureResponseOps[F] { errors: NonEmptyList[ParseFailure] ) = FailureResponse.result(errorParsingHeaderResponse(errors)) - S.fromSafe(headers.headers) match { + S.from(headers.headers) match { case None => process(Option.empty) case Some(Ior.Right(value)) => process(Option(value)) case Some(Ior.Both(errors, _)) => errorParsingHeader(errors) @@ -174,7 +174,7 @@ trait RhoDslHeaderExtractors[F[_]] extends FailureResponseOps[F] { ).widen protected def errorProcessingHeaderResponse[A]( - header: Option[Header.Raw], + header: Option[NonEmptyList[Header.Raw]], nonfatal: Throwable)(implicit F: Monad[F], H: Header[A, _]): F[BaseResult[F]] = { logger.error(nonfatal) { val headerValue = header.fold(show""""${H.name}" was Undefined""")(_.show) From c2d515f85d1f7c136fe2dc0b41ae89a55d10c1e1 Mon Sep 17 00:00:00 2001 From: Darren Gibson Date: Mon, 7 Jun 2021 20:37:59 -0500 Subject: [PATCH 5/6] remove json4s deps --- project/Dependencies.scala | 6 +- .../swagger/SwaggerModelsBuilderSpec.scala | 36 ++-- .../rho/swagger/SwaggerSupportSpec.scala | 204 +++++++++--------- 3 files changed, 121 insertions(+), 125 deletions(-) diff --git a/project/Dependencies.scala b/project/Dependencies.scala index ddd93cb3e..8c9313767 100644 --- a/project/Dependencies.scala +++ b/project/Dependencies.scala @@ -5,7 +5,7 @@ import Keys._ object Dependencies { val http4sVersion = "0.22.0-RC1" val specs2Version = "4.10.6" - val circeVersion = "0.12.3" + val circeVersion = "0.14.1" val scala_213 = "2.13.4" val scala_212 = "2.12.13" @@ -44,7 +44,9 @@ object Dependencies { swaggerCore, swaggerModels, - http4sCirce % "test" + http4sCirce % "test", + circeParser % "test", + circeGeneric % "test", ) lazy val swaggerUiDeps = libraryDependencies ++= Seq(swaggerUi) 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 d01212807..221efd976 100644 --- a/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerModelsBuilderSpec.scala +++ b/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerModelsBuilderSpec.scala @@ -4,6 +4,8 @@ package swagger import cats.effect.IO import cats.syntax.all._ import cats.{Applicative, Monad} +import _root_.io.circe._, _root_.io.circe.parser._, _root_.io.circe.generic.auto._ + import fs2.{Chunk, Stream} import org.http4s.Method._ import org.http4s._ @@ -16,7 +18,6 @@ import org.specs2.mutable.Specification import scala.collection.compat.immutable.ArraySeq import scala.collection.immutable.Seq -import scala.reflect._ import scala.reflect.runtime.universe._ object SwaggerModelsBuilderSpec { @@ -24,19 +25,13 @@ object SwaggerModelsBuilderSpec { case class Bar(c: Long, d: List[Foo]) case class FooVal(str: String) extends AnyVal - import org.json4s._ - import org.json4s.jackson.JsonMethods - - private implicit val format: DefaultFormats = - DefaultFormats - - implicit def jsonParser[A: TypeTag: ClassTag]: StringParser[IO, A] = new StringParser[IO, A] + implicit def jsonParser[A: Decoder: TypeTag]: StringParser[IO, A] = new StringParser[IO, A] with FailureResponseOps[IO] { override val typeTag: Option[TypeTag[A]] = implicitly[TypeTag[A]].some override def parse(s: String)(implicit F: Monad[IO]): ResultResponse[IO, A] = - Either.catchNonFatal(JsonMethods.parse(s).extract[A]) match { + decode[A](s) match { case Left(t) => badRequest[String](t.getMessage) case Right(t) => SuccessResponse(t) } @@ -239,12 +234,11 @@ class SwaggerModelsBuilderSpec extends Specification { } "handle an action with query parameters of empty data types" in { - val ra = fooPath +? param[Unit]("unit") & param[Void]("void") |>> { (_: Unit, _: Void) => "" } + val ra = fooPath +? param[Unit]("unit") |>> { (_: Unit) => "" } sb.collectQueryParams(singleLinearRoute(ra)) must_== List( - QueryParameter(`type` = "string".some, name = "unit".some, required = true), - QueryParameter(`type` = "string".some, name = "void".some, required = true) + QueryParameter(`type` = "string".some, name = "unit".some, required = true) ) } } @@ -252,38 +246,38 @@ class SwaggerModelsBuilderSpec extends Specification { "SwaggerModelsBuilder.collectHeaderParams" should { "handle an action with single header rule" in { - val ra = fooPath >>> exists(`Content-Length`) |>> { () => "" } + val ra = fooPath >>> H[`Content-Length`].exists |>> { () => "" } sb.collectHeaderParams(singleLinearRoute(ra)) must_== List(HeaderParameter(`type` = "string", name = "Content-Length".some, required = true)) } "handle an action with two header rules" in { - val ra = fooPath >>> (exists(`Content-Length`) && exists(`Content-MD5`)) |>> { () => "" } + val ra = fooPath >>> (H[`Content-Length`].exists && H[`Content-Type`].exists) |>> { () => "" } sb.collectHeaderParams(singleLinearRoute(ra)) must_== List( HeaderParameter(`type` = "string", name = "Content-Length".some, required = true), - HeaderParameter(`type` = "string", name = "Content-MD5".some, required = true) + HeaderParameter(`type` = "string", name = "Content-Type".some, required = true) ) } "handle an action with or-structure header rules" in { def orStr(str: String) = s"Optional if the following headers are satisfied: [$str]".some - val ra = fooPath >>> (exists(`Content-Length`) || exists(`Content-MD5`)) |>> { () => "" } + val ra = fooPath >>> (H[`Content-Length`].exists || H[`Content-Type`].exists) |>> { () => "" } sb.collectHeaderParams(singleLinearRoute(ra)) must_== List( HeaderParameter( `type` = "string", name = "Content-Length".some, - description = orStr("Content-MD5"), + description = orStr("Content-Type"), required = true ), HeaderParameter( `type` = "string", - name = "Content-MD5".some, + name = "Content-Type".some, description = orStr("Content-Length"), required = true ) @@ -292,7 +286,7 @@ class SwaggerModelsBuilderSpec extends Specification { "set required = false if there is a default value for the header" in { val ra = - fooPath >>> captureOrElse(`Content-Length`)(`Content-Length`.unsafeFromLong(20)) |>> { + fooPath >>> H[`Content-Length`].captureOrElse(`Content-Length`.unsafeFromLong(20)) |>> { (_: `Content-Length`) => "" } @@ -301,7 +295,7 @@ class SwaggerModelsBuilderSpec extends Specification { } "set required = false if the header is optional" in { - val ra = fooPath >>> captureOptionally(`Content-Length`) |>> { + val ra = fooPath >>> H[`Content-Length`].captureOptionally |>> { (_: Option[`Content-Length`]) => "" } @@ -310,7 +304,7 @@ class SwaggerModelsBuilderSpec extends Specification { } "set required = false by default if there is a missingHeaderResult for the header" in { - val ra = fooPath >>> captureMapR(`Content-Length`, Option(Ok("5")))(Right(_)) |>> { + val ra = fooPath >>> H[`Content-Length`].captureMapR(Option(Ok("5")))(Right(_)) |>> { (_: `Content-Length`) => "" } 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 c7007f919..c0e051553 100644 --- a/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerSupportSpec.scala +++ b/swagger/src/test/scala/org/http4s/rho/swagger/SwaggerSupportSpec.scala @@ -2,8 +2,11 @@ package org.http4s package rho package swagger -import cats.data.NonEmptyList +import cats.data._ import cats.effect.IO +import _root_.io.circe.parser._ +import _root_.io.circe._ +import _root_.io.circe.syntax._ import org.http4s.rho.bits.MethodAliases.GET import org.http4s.rho.io._ import org.http4s.rho.swagger.models._ @@ -11,8 +14,6 @@ import org.http4s.rho.swagger.syntax.io._ import org.specs2.mutable.Specification class SwaggerSupportSpec extends Specification { - import org.json4s.JsonAST._ - import org.json4s.jackson._ val baseRoutes = new RhoRoutes[IO] { GET / "hello" |>> { () => Ok("hello world") } @@ -41,26 +42,41 @@ class SwaggerSupportSpec extends Specification { } "SwaggerSupport" should { + case class SwaggerRoot( + paths: Map[String, Json] = Map.empty + ) + implicit lazy val swaggerRootDecoder: Decoder[SwaggerRoot] = + _root_.io.circe.generic.semiauto.deriveDecoder + "Expose an API listing" in { val service = baseRoutes.toRoutes(createRhoMiddleware(swaggerRoutesInSwagger = true)) - val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) + val r = Request[IO](GET, Uri(path = Uri.Path.unsafeFromString("/swagger.json"))) + + val json = decode[SwaggerRoot](RRunner(service).checkOk(r)) - val JObject(List((a, JObject(_)), (b, JObject(_)), (c, JObject(_)), (d, JObject(_)))) = - parseJson(RRunner(service).checkOk(r)) \\ "paths" + json should beRight + val swaggerRoot = json.getOrElse(???) - Set(a, b, c, d) should_== Set("/swagger.json", "/swagger.yaml", "/hello", "/hello/{string}") + swaggerRoot.paths.keySet should_== Set( + "/swagger.json", + "/swagger.yaml", + "/hello", + "/hello/{string}" + ) } "Support prefixed routes" in { val service = ("foo" /: baseRoutes).toRoutes(createRhoMiddleware(swaggerRoutesInSwagger = true)) - val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) + val r = Request[IO](GET, Uri(path = Uri.Path.unsafeFromString("/swagger.json"))) - val JObject(List((a, JObject(_)), (b, JObject(_)), (c, JObject(_)), (d, JObject(_)))) = - parseJson(RRunner(service).checkOk(r)) \\ "paths" + val json = decode[SwaggerRoot](RRunner(service).checkOk(r)) - Set(a, b, c, d) should_== Set( + json should beRight + val swaggerRoot = json.getOrElse(???) + + swaggerRoot.paths.keySet should_== Set( "/swagger.json", "/swagger.yaml", "/foo/hello", @@ -79,29 +95,42 @@ class SwaggerSupportSpec extends Specification { val swaggerRoutes = createSwaggerRoute(aggregateSwagger) val httpRoutes = NonEmptyList.of(baseRoutes, moarRoutes, swaggerRoutes).reduceLeft(_ and _) - val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) + val r = Request[IO](GET, Uri(path = Uri.Path.unsafeFromString("/swagger.json"))) + + val json = decode[SwaggerRoot](RRunner(httpRoutes.toRoutes()).checkOk(r)) - val JObject(List((a, JObject(_)), (b, JObject(_)), (c, JObject(_)), (d, JObject(_)))) = - parseJson(RRunner(httpRoutes.toRoutes()).checkOk(r)) \\ "paths" + json should beRight + val swaggerRoot = json.getOrElse(???) - Set(a, b, c, d) should_== Set("/hello", "/hello/{string}", "/goodbye", "/goodbye/{string}") + swaggerRoot.paths.keySet should_== Set( + "/hello", + "/hello/{string}", + "/goodbye", + "/goodbye/{string}" + ) } "Support endpoints which end in a slash" in { val service = trailingSlashRoutes.toRoutes(createRhoMiddleware()) - val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) - val JObject(List((a, JObject(_)))) = parseJson(RRunner(service).checkOk(r)) \\ "paths" + val r = Request[IO](GET, Uri(path = Uri.Path.unsafeFromString("/swagger.json"))) - a should_== "/foo/" + val json = decode[SwaggerRoot](RRunner(service).checkOk(r)) + + json should beRight + val swaggerRoot = json.getOrElse(???) + + swaggerRoot.paths.keys.head should_== "/foo/" } "Support endpoints which end in a slash being mixed with normal endpoints" in { val service = mixedTrailingSlashesRoutes.toRoutes(createRhoMiddleware()) - val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) - val JObject(List((a, JObject(_)), (b, JObject(_)), (c, JObject(_)))) = - parseJson(RRunner(service).checkOk(r)) \\ "paths" + val r = Request[IO](GET, Uri(path = Uri.Path.unsafeFromString("/swagger.json"))) + val json = decode[SwaggerRoot](RRunner(service).checkOk(r)) + + json should beRight + val swaggerRoot = json.getOrElse(???) - Set(a, b, c) should_== Set("/foo/", "/foo", "/bar") + swaggerRoot.paths.keySet should_== Set("/foo/", "/foo", "/bar") } "Provide a way to agregate routes from multiple RhoRoutes, with mixed trailing slashes and non-trailing slashes" in { @@ -111,22 +140,14 @@ class SwaggerSupportSpec extends Specification { val swaggerRoutes = createSwaggerRoute(aggregateSwagger) val httpRoutes = NonEmptyList.of(baseRoutes, moarRoutes, swaggerRoutes).reduceLeft(_ and _) - val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) - - val JObject( - List( - (a, JObject(_)), - (b, JObject(_)), - (c, JObject(_)), - (d, JObject(_)), - (e, JObject(_)), - (f, JObject(_)), - (g, JObject(_)) - ) - ) = - parseJson(RRunner(httpRoutes.toRoutes()).checkOk(r)) \\ "paths" + val r = Request[IO](GET, Uri(path = Uri.Path.unsafeFromString("/swagger.json"))) + + val json = decode[SwaggerRoot](RRunner(httpRoutes.toRoutes()).checkOk(r)) - Set(a, b, c, d, e, f, g) should_== Set( + json should beRight + val swaggerRoot = json.getOrElse(???) + + swaggerRoot.paths.keySet should_== Set( "/hello", "/hello/{string}", "/goodbye", @@ -140,23 +161,22 @@ class SwaggerSupportSpec extends Specification { "Check metadata in API listing" in { val service = metaDataRoutes.toRoutes(createRhoMiddleware(swaggerRoutesInSwagger = true)) - val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) + val r = Request[IO](GET, Uri(path = Uri.Path.unsafeFromString("/swagger.json"))) - val json = parseJson(RRunner(service).checkOk(r)) + val json = parse(RRunner(service).checkOk(r)).getOrElse(???) - val JObject( - List( - (a, JArray(List(JObject(List((e, JArray(List(JString(f))))))))), - (b, JArray(List(JObject(List((g, JArray(List(JString(h))))))))) + val security: List[Json] = (json \\ "security").flatMap(_.asArray).flatten + security should contain( + Json.obj( + "bye" := List("hello") + ), + Json.obj( + "hello" := List("bye") ) - ) = json \\ "security" - val JObject(List(_, (c, JString(i)), _, (d, JString(j)))) = json \\ "summary" - - Set(a, b) should_== Set("security", "security") - Set(c, d) should_== Set("summary", "summary") - Set(e, f) should_== Set("hello", "bye") - Set(g, h) should_== Set("bye", "hello") - Set(i, j) should_== Set("Hello", "Bye") + ) + + val summary = (json \\ "summary").flatMap(_.asString) + summary should contain("Bye", "Hello") } "Swagger support for complex meta data" in { @@ -205,57 +225,37 @@ class SwaggerSupportSpec extends Specification { ) ) - val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger-test.json"))) - val json = parseJson(RRunner(service).checkOk(r)) - - val JString(icn) = json \ "info" \ "contact" \ "name" - val JString(h) = json \ "host" - val JArray(List(JString(s1), JString(s2))) = json \ "schemes" - val JString(bp) = json \ "basePath" - val JArray(List(JString(c))) = json \ "consumes" - val JArray(List(JString(p))) = json \ "produces" - val JArray(List(JArray(sec1))) = json \ "security" \ "apiKey" - val JArray(List(JArray(List(JString(sec2))))) = json \ "security" \ "vendor_jwk" - val JString(t) = json \ "securityDefinitions" \ "api_key" \ "type" - val JString(vi) = json \ "securityDefinitions" \ "vendor_jwk" \ "x-vendor-issuer" - val JString(ve) = json \ "x-vendor-endpoints" \ "target" - - Set(icn, h, s1, s2, bp, c, p, sec2, t, vi, ve) should_== Set( - "Name", - "www.test.com", - "http", - "https", - "/v1", - "application/json", - "application/json", - "admin", - "apiKey", - "https://www.test.com/", - "298.0.0.1" - ) - sec1 should_== Nil - } - - "Check metadata in API listing" in { - val service = metaDataRoutes.toRoutes(createRhoMiddleware(swaggerRoutesInSwagger = true)) - - val r = Request[IO](GET, Uri(path = Uri.Path.fromString("/swagger.json"))) - - val json = parseJson(RRunner(service).checkOk(r)) - - val JObject( - List( - (a, JArray(List(JObject(List((e, JArray(List(JString(f))))))))), - (b, JArray(List(JObject(List((g, JArray(List(JString(h))))))))) - ) - ) = json \\ "security" - val JObject(List(_, (c, JString(i)), _, (d, JString(j)))) = json \\ "summary" - - Set(a, b) should_== Set("security", "security") - Set(c, d) should_== Set("summary", "summary") - Set(e, f) should_== Set("hello", "bye") - Set(g, h) should_== Set("bye", "hello") - Set(i, j) should_== Set("Hello", "Bye") + val r = Request[IO](GET, Uri(path = Uri.Path.unsafeFromString("/swagger-test.json"))) + val json = parse(RRunner(service).checkOk(r)).getOrElse(???) + val cursor = json.hcursor + + val icn = cursor.downField("info").downField("contact").get[String]("name") + val h = cursor.get[String]("host") + val s = cursor.downField("schemes").as[List[String]] + val bp = cursor.get[String]("basePath") + val c = cursor.downField("consumes").downArray.as[String] + val p = cursor.downField("produces").downArray.as[String] + val sec1 = cursor.downField("security").downArray.downField("apiKey").as[List[String]] + val sec2 = + cursor.downField("security").downArray.right.downField("vendor_jwk").downArray.as[String] + val t = cursor.downField("securityDefinitions").downField("api_key").get[String]("type") + val vi = cursor + .downField("securityDefinitions") + .downField("vendor_jwk") + .get[String]("x-vendor-issuer") + val ve = cursor.downField("x-vendor-endpoints").get[String]("target") + + icn should beRight("Name") + h should beRight("www.test.com") + s should beRight(List("http", "https")) + bp should beRight("/v1") + c should beRight("application/json") + p should beRight("application/json") + sec2 should beRight("admin") + t should beRight("apiKey") + vi should beRight("https://www.test.com/") + ve should beRight("298.0.0.1") + sec1 should beRight(List.empty[String]) } } } From 339de5d598a46174f0480880082ead2f3d59d6ef Mon Sep 17 00:00:00 2001 From: Darren Gibson Date: Wed, 9 Jun 2021 08:30:14 -0500 Subject: [PATCH 6/6] sbt-tpolecat and some cleanup --- build.sbt | 41 +++++++++---------- .../scala/org/http4s/rho/PathBuilder.scala | 2 +- .../scala/org/http4s/rho/QueryBuilder.scala | 2 +- .../main/scala/org/http4s/rho/Router.scala | 2 +- .../http4s/rho/bits/HeaderAppendable.scala | 2 +- .../org/http4s/rho/CompileRoutesSpec.scala | 4 +- project/plugins.sbt | 2 + 7 files changed, 27 insertions(+), 28 deletions(-) diff --git a/build.sbt b/build.sbt index 82d9f2c27..8b8b29866 100644 --- a/build.sbt +++ b/build.sbt @@ -65,7 +65,6 @@ lazy val docs = project version.value, apiVersion.value ), - (ScalaUnidoc / unidoc / scalacOptions) ++= versionSpecificEnabledFlags(scalaVersion.value), (ScalaUnidoc / unidoc / unidocProjectFilter) := inProjects( `rho-core`, `rho-swagger` @@ -93,22 +92,16 @@ lazy val `rho-examples` = project ) .dependsOn(`rho-swagger`, `rho-swagger-ui`) -lazy val compilerFlags = Seq( - "-feature", - "-deprecation", - "-unchecked", - "-language:higherKinds", - "-language:existentials", - "-language:implicitConversions", - "-Ywarn-unused", - "-Xfatal-warnings" +lazy val disabledCompilerFlags = Seq( // TODO: Fix code and re-enable these. + "-Xlint:package-object-classes", + "-Ywarn-numeric-widen", + "-Wnumeric-widen", + "-Xlint:adapted-args", + "-Yno-adapted-args", + "-Wdead-code", + "-Ywarn-dead-code" ) -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 @@ -120,7 +113,7 @@ lazy val buildSettings = publishing ++ Seq( scalaVersion := scala_213, crossScalaVersions := Seq(scala_213, scala_212), - scalacOptions := compilerFlags ++ versionSpecificEnabledFlags(scalaVersion.value), + scalacOptions --= disabledCompilerFlags, resolvers += Resolver.sonatypeRepo("snapshots"), (run / fork) := true, (ThisBuild / organization) := "org.http4s", @@ -128,15 +121,19 @@ lazy val buildSettings = publishing ++ description := "A self documenting DSL build upon the http4s framework", license, libraryDependencies ++= Seq( - shapeless, - silencerPlugin, - silencerLib, - kindProjector, http4sServer % "provided", logbackClassic % "test" ), - libraryDependencies ++= specs2, - libraryDependencies += `scala-reflect` % scalaVersion.value + libraryDependencies ++= (if (scalaVersion.value.startsWith("2")) + Seq( + shapeless, + silencerPlugin, + silencerLib, + kindProjector, + `scala-reflect` % scalaVersion.value + ) + else Seq.empty), + libraryDependencies ++= specs2 ) // to keep REPL usable diff --git a/core/src/main/scala/org/http4s/rho/PathBuilder.scala b/core/src/main/scala/org/http4s/rho/PathBuilder.scala index 9a355f0d9..ecc2ae8d7 100644 --- a/core/src/main/scala/org/http4s/rho/PathBuilder.scala +++ b/core/src/main/scala/org/http4s/rho/PathBuilder.scala @@ -21,7 +21,7 @@ final class PathBuilder[F[_], T <: HList](val method: Method, val path: PathRule with HeaderAppendable[F, T] with RoutePrependable[F, PathBuilder[F, T]] with UriConvertible[F] { - type HeaderAppendResult[T <: HList] = Router[F, T] + type HeaderAppendResult[T0 <: HList] = Router[F, T0] override val rules: RequestRule[F] = EmptyRule[F]() diff --git a/core/src/main/scala/org/http4s/rho/QueryBuilder.scala b/core/src/main/scala/org/http4s/rho/QueryBuilder.scala index 52f75818b..6624af722 100644 --- a/core/src/main/scala/org/http4s/rho/QueryBuilder.scala +++ b/core/src/main/scala/org/http4s/rho/QueryBuilder.scala @@ -41,7 +41,7 @@ case class QueryBuilder[F[_], T <: HList](method: Method, path: PathRule, rules: override def /:(prefix: TypedPath[F, HNil]): QueryBuilder[F, T] = new QueryBuilder[F, T](method, PathAnd(prefix.rule, path), rules) - override type HeaderAppendResult[T <: HList] = Router[F, T] + override type HeaderAppendResult[T0 <: HList] = Router[F, T0] override def makeRoute(action: Action[F, T]): RhoRoute[F, T] = RhoRoute(Router(method, path, rules), action) diff --git a/core/src/main/scala/org/http4s/rho/Router.scala b/core/src/main/scala/org/http4s/rho/Router.scala index dafc7583a..41b7aef01 100644 --- a/core/src/main/scala/org/http4s/rho/Router.scala +++ b/core/src/main/scala/org/http4s/rho/Router.scala @@ -36,7 +36,7 @@ case class Router[F[_], T <: HList](method: Method, path: PathRule, rules: Reque with RoutePrependable[F, Router[F, T]] { type Self = Router[F, T] - override type HeaderAppendResult[T <: HList] = Router[F, T] + override type HeaderAppendResult[T0 <: HList] = Router[F, T0] override def /:(prefix: TypedPath[F, HNil]): Self = copy(path = PathAnd(prefix.rule, path)) 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 0abd9b5b7..b87bab336 100644 --- a/core/src/main/scala/org/http4s/rho/bits/HeaderAppendable.scala +++ b/core/src/main/scala/org/http4s/rho/bits/HeaderAppendable.scala @@ -8,7 +8,7 @@ import shapeless.ops.hlist.Prepend * @tparam T The `HList` representation of the values to be extracted from the `Request`. */ trait HeaderAppendable[F[_], T <: HList] { - type HeaderAppendResult[T <: HList] <: HeaderAppendable[F, T] + type HeaderAppendResult[T0 <: HList] <: HeaderAppendable[F, T0] /** Append the header to the builder, generating a new typed representation of the route */ def >>>[T1 <: HList](header: TypedHeader[F, T1])(implicit diff --git a/core/src/test/scala/org/http4s/rho/CompileRoutesSpec.scala b/core/src/test/scala/org/http4s/rho/CompileRoutesSpec.scala index f05b59aae..d3513c0e7 100644 --- a/core/src/test/scala/org/http4s/rho/CompileRoutesSpec.scala +++ b/core/src/test/scala/org/http4s/rho/CompileRoutesSpec.scala @@ -8,10 +8,10 @@ import org.specs2.mutable.Specification class CompileRoutesSpec extends Specification { - def getFoo(implicit c: CompileRoutes[IO, _]): Unit = + def getFoo(implicit c: CompileRoutes[IO, _]) = GET / "hello" |>> "GetFoo" - def putFoo(implicit c: CompileRoutes[IO, _]): Unit = + def putFoo(implicit c: CompileRoutes[IO, _]) = PUT / "hello" |>> "PutFoo" "CompileService" should { diff --git a/project/plugins.sbt b/project/plugins.sbt index 1a1c13b20..a04a16aa2 100644 --- a/project/plugins.sbt +++ b/project/plugins.sbt @@ -13,3 +13,5 @@ addSbtPlugin("com.eed3si9n" % "sbt-buildinfo" % "0.10.0") addSbtPlugin("com.geirsson" % "sbt-ci-release" % "1.5.7") addSbtPlugin("org.scalameta" % "sbt-scalafmt" % "2.4.2") + +addSbtPlugin("io.github.davidgregory084" % "sbt-tpolecat" % "0.1.20")