From 5e8c380b16077413a56a22331ed1e7edfe39fc69 Mon Sep 17 00:00:00 2001 From: Tim Nieradzik Date: Tue, 9 Apr 2019 21:35:27 +0200 Subject: [PATCH] Route: Check path in parameter routes Resolves a bug where routes with parameters always matched even if the paths were different. --- shared/src/main/scala/trail/Route.scala | 32 ++++++++++--------- .../test/scala/trail/RouteParamTests.scala | 8 +++++ 2 files changed, 25 insertions(+), 15 deletions(-) diff --git a/shared/src/main/scala/trail/Route.scala b/shared/src/main/scala/trail/Route.scala index 85b6f3a..0c7d4bd 100644 --- a/shared/src/main/scala/trail/Route.scala +++ b/shared/src/main/scala/trail/Route.scala @@ -100,8 +100,9 @@ object Route { override def parseInternal(path: Path): Option[(P, Path)] = for { (_, lp) <- route.parseInternal(path) - rv <- fragment.codec.decode(path.fragment) - } yield (rv, lp.copy(fragment = None)) + v <- fragment.codec.decode(lp.fragment) + p = lp.copy(fragment = None) + } yield (v, p) } case class FragmentRoute[A, P](route: Route[A], fragment: Fragment[P]) extends Route[(A, P)] { @@ -113,8 +114,9 @@ object Route { override def parseInternal(path: Path): Option[((A, P), Path)] = for { (lv, lp) <- route.parseInternal(path) - rv <- fragment.codec.decode(path.fragment) - } yield ((lv, rv), lp.copy(fragment = None)) + rv <- fragment.codec.decode(lp.fragment) + p = lp.copy(fragment = None) + } yield ((lv, rv), p) } case class ParamRoute0[P](route: Route[Unit], param: Param[P]) extends Route[P] { @@ -123,11 +125,13 @@ object Route { .fold("")(v => param.name + "=" + URI.encode(v)) route.url(()) + (if (arg.isEmpty) "" else "?" + arg) } - override def parseInternal(path: Path): Option[(P, Path)] = { - val arg = path.args.find(_._1 == param.name) - param.codec.decode(arg.map(_._2)).map(decode => - (decode, path.copy(args = path.args.diff(arg.toList)))) - } + override def parseInternal(path: Path): Option[(P, Path)] = + for { + (_, lp) <- route.parseInternal(path) + arg = lp.args.find(_._1 == param.name) + v <- param.codec.decode(arg.map(_._2)) + p = lp.copy(args = lp.args.diff(arg.toList)) + } yield (v, p) def &[T](param: Param[T]): ParamRoute[P, T] = ParamRoute(this, param) } @@ -143,11 +147,9 @@ object Route { override def parseInternal(path: Path): Option[((A, P), Path)] = for { (lv, lp) <- route.parseInternal(path) - (rv, rp) <- { - val arg = lp.args.find(_._1 == param.name) - param.codec.decode(arg.map(_._2)).map(decode => - (decode, lp.copy(args = lp.args.diff(arg.toList)))) - } - } yield ((lv, rv), rp) + arg = lp.args.find(_._1 == param.name) + rv <- param.codec.decode(arg.map(_._2)) + p = lp.copy(args = lp.args.diff(arg.toList)) + } yield ((lv, rv), p) } } diff --git a/shared/src/test/scala/trail/RouteParamTests.scala b/shared/src/test/scala/trail/RouteParamTests.scala index 6c6371a..e214b11 100644 --- a/shared/src/test/scala/trail/RouteParamTests.scala +++ b/shared/src/test/scala/trail/RouteParamTests.scala @@ -148,4 +148,12 @@ class RouteParamTests extends FunSpec with Matchers { assert(route.parse("/?test=42&test=value") .contains((42, "value"))) } + + it("Only match parameter routes with same path") { + val route = Root / "api" / "catalogue" / "content" & Param[String]("category") + + assert(route.parse("/catalogue/content?category=Audio").isEmpty) + assert(route.parse("/api/catalogue/content?category=Audio") + .contains("Audio")) + } }