diff --git a/build.sbt b/build.sbt index 21482163..8d989b20 100644 --- a/build.sbt +++ b/build.sbt @@ -7,7 +7,6 @@ val catsEffectVersion = "3.5.4" val circeVersion = "0.14.8" val disciplineMunitVersion = "2.0.0-M3" val doobieVersion = "1.0.0-RC5" -val flywayVersion = "10.15.0" val fs2Version = "3.10.2" val http4sVersion = "0.23.27" val jnrUnixsocketVersion = "0.38.22" @@ -252,24 +251,23 @@ lazy val generic = crossProject(JVMPlatform, JSPlatform, NativePlatform) lazy val demo = project .in(file("demo")) .enablePlugins(NoPublishPlugin, AutomateHeaderPlugin) - .dependsOn(core.jvm, generic.jvm, doobie) + .dependsOn(buildInfo.jvm, core.jvm, generic.jvm, doobie) .settings(commonSettings) .settings( name := "grackle-demo", coverageEnabled := false, libraryDependencies ++= Seq( - "org.typelevel" %% "log4cats-slf4j" % log4catsVersion, - "ch.qos.logback" % "logback-classic" % logbackVersion, - "org.tpolecat" %% "doobie-core" % doobieVersion, - "org.tpolecat" %% "doobie-postgres" % doobieVersion, - "org.tpolecat" %% "doobie-hikari" % doobieVersion, - "org.http4s" %% "http4s-ember-server" % http4sVersion, - "org.http4s" %% "http4s-ember-client" % http4sVersion, - "org.http4s" %% "http4s-circe" % http4sVersion, - "org.http4s" %% "http4s-dsl" % http4sVersion, - "org.flywaydb" % "flyway-database-postgresql" % flywayVersion, - "io.chrisdavenport" %% "whale-tail-manager" % whaleTailVersion, - "com.github.jnr" % "jnr-unixsocket" % jnrUnixsocketVersion + "org.typelevel" %% "log4cats-slf4j" % log4catsVersion, + "ch.qos.logback" % "logback-classic" % logbackVersion, + "org.tpolecat" %% "doobie-core" % doobieVersion, + "org.tpolecat" %% "doobie-postgres" % doobieVersion, + "org.tpolecat" %% "doobie-hikari" % doobieVersion, + "org.http4s" %% "http4s-ember-server" % http4sVersion, + "org.http4s" %% "http4s-ember-client" % http4sVersion, + "org.http4s" %% "http4s-circe" % http4sVersion, + "org.http4s" %% "http4s-dsl" % http4sVersion, + "io.chrisdavenport" %% "whale-tail-manager" % whaleTailVersion, + "com.github.jnr" % "jnr-unixsocket" % jnrUnixsocketVersion ) ) diff --git a/demo/src/main/resources/db/migration/V1__WorldSetup.sql b/demo/src/main/resources/db/world.sql similarity index 100% rename from demo/src/main/resources/db/migration/V1__WorldSetup.sql rename to demo/src/main/resources/db/world.sql diff --git a/demo/src/main/scala/demo/DemoServer.scala b/demo/src/main/scala/demo/DemoServer.scala index 2952d97a..649acb38 100644 --- a/demo/src/main/scala/demo/DemoServer.scala +++ b/demo/src/main/scala/demo/DemoServer.scala @@ -26,7 +26,7 @@ import org.http4s.server.staticcontent.resourceServiceBuilder // #server object DemoServer { - def resource(graphQLRoutes: HttpRoutes[IO]): Resource[IO, Unit] = { + def mkServer(graphQLRoutes: HttpRoutes[IO]): Resource[IO, Unit] = { val httpApp0 = ( // Routes for static resources, i.e. GraphQL Playground resourceServiceBuilder[IO]("/assets").toRoutes <+> diff --git a/demo/src/main/scala/demo/GraphQLService.scala b/demo/src/main/scala/demo/GraphQLService.scala index 64f9bd57..5f8a711f 100644 --- a/demo/src/main/scala/demo/GraphQLService.scala +++ b/demo/src/main/scala/demo/GraphQLService.scala @@ -24,17 +24,8 @@ import org.http4s.dsl.Http4sDsl import org.http4s.{HttpRoutes, InvalidMessageBodyFailure, ParseFailure, QueryParamDecoder} // #service -trait GraphQLService[F[_]] { - def runQuery(op: Option[String], vars: Option[Json], query: String): F[Json] -} - object GraphQLService { - - def fromMapping[F[_]: Concurrent](mapping: Mapping[F]): GraphQLService[F] = - (op: Option[String], vars: Option[Json], query: String) => - mapping.compileAndRun(query, op, vars) - - def routes[F[_]: Concurrent](prefix: String, svc: GraphQLService[F]): HttpRoutes[F] = { + def mkRoutes[F[_]: Concurrent](prefix: String)(mapping: Mapping[F]): HttpRoutes[F] = { val dsl = new Http4sDsl[F]{} import dsl._ @@ -60,7 +51,7 @@ object GraphQLService { errors => BadRequest(errors.map(_.sanitized).mkString_("", ",", "")), vars => for { - result <- svc.runQuery(op, vars, query) + result <- mapping.compileAndRun(query, op, vars) resp <- Ok(result) } yield resp ) @@ -77,7 +68,7 @@ object GraphQLService { ) op = obj("operationName").flatMap(_.asString) vars = obj("variables") - result <- svc.runQuery(op, vars, query) + result <- mapping.compileAndRun(query, op, vars) resp <- Ok(result) } yield resp } diff --git a/demo/src/main/scala/demo/Main.scala b/demo/src/main/scala/demo/Main.scala index 5a3cb529..90198392 100644 --- a/demo/src/main/scala/demo/Main.scala +++ b/demo/src/main/scala/demo/Main.scala @@ -15,99 +15,22 @@ package demo -import java.util.concurrent.Executors - -import scala.concurrent.ExecutionContext -import scala.concurrent.duration._ - -import cats.effect.{ExitCode, IO, IOApp, Resource} +import cats.effect.{ExitCode, IO, IOApp} import cats.syntax.all._ -import demo.starwars.{StarWarsData, StarWarsMapping} +import demo.starwars.StarWarsMapping import demo.world.WorldMapping -import doobie.hikari.HikariTransactor -import io.chrisdavenport.whaletail.Docker -import io.chrisdavenport.whaletail.manager._ -import org.flywaydb.core.Flyway + +import GraphQLService.mkRoutes +import DemoServer.mkServer // #main object Main extends IOApp { def run(args: List[String]): IO[ExitCode] = { - DBSetup.run { xa => - val worldGraphQLRoutes = GraphQLService.routes( - "world", - GraphQLService.fromMapping(WorldMapping.mkMappingFromTransactor(xa)) - ) - val starWarsGraphQLRoutes = GraphQLService.routes[IO]( - "starwars", - GraphQLService.fromMapping(new StarWarsMapping[IO] with StarWarsData[IO]) - ) - DemoServer.resource(worldGraphQLRoutes <+> starWarsGraphQLRoutes) - } + (for { + starWarsRoutes <- StarWarsMapping[IO].map(mkRoutes("starwars")) + worldRoutes <- WorldMapping[IO].map(mkRoutes("world")) + _ <- mkServer(starWarsRoutes <+> worldRoutes) + } yield ()).useForever } } // #main - -object DBSetup { - def run(body: HikariTransactor[IO] => Resource[IO, Unit]): IO[Nothing] = - container.evalTap(dbMigration(_)).flatMap(transactor(_)).flatMap(body).useForever - - case class PostgresConnectionInfo(host: String, port: Int) { - val driverClassName = "org.postgresql.Driver" - val databaseName = "test" - val jdbcUrl = s"jdbc:postgresql://$host:$port/$databaseName" - val username = "test" - val password = "test" - } - object PostgresConnectionInfo { - val DefaultPort = 5432 - } - - val container: Resource[IO, PostgresConnectionInfo] = Docker.default[IO].flatMap(client => - WhaleTailContainer.build( - client, - image = "postgres", - tag = "11.8".some, - ports = Map(PostgresConnectionInfo.DefaultPort -> None), - env = Map( - "POSTGRES_USER" -> "test", - "POSTGRES_PASSWORD" -> "test", - "POSTGRES_DB" -> "test" - ), - labels = Map.empty - ).evalTap( - ReadinessStrategy.checkReadiness( - client, - _, - ReadinessStrategy.LogRegex(".*database system is ready to accept connections.*".r, 2), - 30.seconds - ) - ) - ).flatMap(container => - Resource.eval( - container.ports.get(PostgresConnectionInfo.DefaultPort).liftTo[IO](new Throwable("Missing Port")) - ) - ).map { - case (host, port) => PostgresConnectionInfo(host, port) - } - - def transactor(connInfo: PostgresConnectionInfo): Resource[IO, HikariTransactor[IO]] = { - import connInfo._ - HikariTransactor.newHikariTransactor[IO]( - driverClassName, - jdbcUrl, - username, - password, - ExecutionContext.fromExecutorService(Executors.newFixedThreadPool(4)) - ) - } - - def dbMigration(connInfo: PostgresConnectionInfo): IO[Unit] = { - import connInfo._ - IO.blocking { - val flyway = Flyway - .configure() - .dataSource(jdbcUrl, username, password) - flyway.load().migrate() - }.void - } -} diff --git a/demo/src/main/scala/demo/starwars/StarWarsMapping.scala b/demo/src/main/scala/demo/starwars/StarWarsMapping.scala index e0d2f1d1..4d2dcc96 100644 --- a/demo/src/main/scala/demo/starwars/StarWarsMapping.scala +++ b/demo/src/main/scala/demo/starwars/StarWarsMapping.scala @@ -15,7 +15,9 @@ package demo.starwars +import cats.MonadThrow import cats.syntax.all._ +import cats.effect.Resource import grackle.Predicate._ import grackle.Query._ import grackle.QueryCompiler._ @@ -106,6 +108,11 @@ trait StarWarsMapping[F[_]] extends GenericMapping[F] { self: StarWarsData[F] => // #elaborator } +object StarWarsMapping { + def apply[F[_]: MonadThrow]: Resource[F, StarWarsMapping[F]] = + Resource.pure(new StarWarsMapping[F] with StarWarsData[F]) +} + // The types and values for the in-memory Star Wars example. trait StarWarsData[F[_]] extends GenericMapping[F] { self: StarWarsMapping[F] => import semiauto._ diff --git a/demo/src/main/scala/demo/world/WorldData.scala b/demo/src/main/scala/demo/world/WorldData.scala new file mode 100644 index 00000000..432ca455 --- /dev/null +++ b/demo/src/main/scala/demo/world/WorldData.scala @@ -0,0 +1,85 @@ +// Copyright (c) 2016-2023 Association of Universities for Research in Astronomy, Inc. (AURA) +// Copyright (c) 2016-2023 Grackle Contributors +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package demo.world + +import java.util.concurrent.Executors + +import scala.concurrent.ExecutionContext +import scala.concurrent.duration._ + +import cats.effect.{Async, Resource} +import cats.syntax.all._ +import doobie.hikari.HikariTransactor +import io.chrisdavenport.whaletail.{Containers, Docker} +import io.chrisdavenport.whaletail.manager._ + +object WorldData { + def mkContainer[F[_]: Async]: Resource[F, PostgresConnectionInfo] = + Docker.default[F].flatMap(client => + WhaleTailContainer.build( + client, + image = "postgres", + tag = "11.8".some, + ports = Map(PostgresConnectionInfo.DefaultPort -> None), + binds = List(Containers.Bind(bindPath("demo/src/main/resources/db/"), "/docker-entrypoint-initdb.d/", "ro")), + env = Map( + "POSTGRES_USER" -> "test", + "POSTGRES_PASSWORD" -> "test", + "POSTGRES_DB" -> "test" + ), + labels = Map.empty + ).evalTap( + ReadinessStrategy.checkReadiness( + client, + _, + ReadinessStrategy.LogRegex(".*database system is ready to accept connections.*".r, 2), + 30.seconds + ) + ) + ).flatMap(container => + Resource.eval( + container.ports.get(PostgresConnectionInfo.DefaultPort).liftTo[F](new Throwable("Missing Port")) + ) + ).map { + case (host, port) => PostgresConnectionInfo(host, port) + } + + def mkTransactor[F[_]: Async](connInfo: PostgresConnectionInfo): Resource[F, HikariTransactor[F]] = { + import connInfo._ + HikariTransactor.newHikariTransactor[F]( + driverClassName, + jdbcUrl, + username, + password, + ExecutionContext.fromExecutorService(Executors.newFixedThreadPool(4)) + ) + } + + case class PostgresConnectionInfo(host: String, port: Int) { + val driverClassName = "org.postgresql.Driver" + val databaseName = "test" + val jdbcUrl = s"jdbc:postgresql://$host:$port/$databaseName" + val username = "test" + val password = "test" + } + + object PostgresConnectionInfo { + val DefaultPort = 5432 + } + + def bindPath(path: String): String = + buildinfo.BuildInfo.baseDirectory + "/" + path +} diff --git a/demo/src/main/scala/demo/world/WorldMapping.scala b/demo/src/main/scala/demo/world/WorldMapping.scala index b907f69d..d5858718 100644 --- a/demo/src/main/scala/demo/world/WorldMapping.scala +++ b/demo/src/main/scala/demo/world/WorldMapping.scala @@ -15,8 +15,8 @@ package demo.world -import _root_.doobie.{Meta, Transactor} -import cats.effect.Sync +import cats.effect.{Async, Resource, Sync} +import doobie.{Meta, Transactor} import grackle.Predicate._ import grackle.Query._ import grackle.QueryCompiler._ @@ -28,6 +28,8 @@ import grackle.syntax._ import org.typelevel.log4cats.Logger import org.typelevel.log4cats.slf4j.Slf4jLogger +import WorldData._ + trait WorldMapping[F[_]] extends DoobieMapping[F] { // #db_tables object country extends TableDef("country") { @@ -278,8 +280,15 @@ object WorldMapping extends LoggedDoobieMappingCompanion { def mkMapping[F[_]: Sync](transactor: Transactor[F], monitor: DoobieMonitor[F]): WorldMapping[F] = new DoobieMapping(transactor, monitor) with WorldMapping[F] - def mkMappingFromTransactor[F[_]: Sync](transactor: Transactor[F]): Mapping[F] = { - implicit val logger: Logger[F] = Slf4jLogger.getLoggerFromName[F]("SqlQueryLogger") - mkMapping(transactor) + def mkMappingFromTransactor[F[_]: Sync](transactor: Transactor[F]): WorldMapping[F] = { + val logger: Logger[F] = Slf4jLogger.getLoggerFromName[F]("SqlQueryLogger") + val monitor: DoobieMonitor[F] = DoobieMonitor.loggerMonitor[F](logger) + mkMapping(transactor, monitor) } + + def apply[F[_]: Async]: Resource[F, WorldMapping[F]] = + for { + connInfo <- mkContainer[F] + transactor <- mkTransactor[F](connInfo) + } yield mkMappingFromTransactor[F](transactor) } diff --git a/docs/tutorial/db-backed-model.md b/docs/tutorial/db-backed-model.md index 56b5abfb..1741d0d1 100644 --- a/docs/tutorial/db-backed-model.md +++ b/docs/tutorial/db-backed-model.md @@ -117,8 +117,8 @@ println(grackle.docs.Output.snip("demo/src/main/scala/demo/world/WorldMapping.sc To expose the GraphQL API via http4s we will use the `GraphQLService` and `DemoServer` from the [in-memory example](in-memory-model.md#the-service). -The `run` method starts the dockerized PostgreSQL database, creates the database schema, writes initial data and -exposes the GraphQL API for both the in-memory and the db-backend models, +The `run` method starts the dockerized PostgreSQL database, and exposes the GraphQL API for both the in-memory and the +db-backend models, ```scala mdoc:passthrough println(grackle.docs.Output.snip("demo/src/main/scala/demo/Main.scala", "#main")) diff --git a/docs/tutorial/in-memory-model.md b/docs/tutorial/in-memory-model.md index 7708e730..22c0b1f6 100644 --- a/docs/tutorial/in-memory-model.md +++ b/docs/tutorial/in-memory-model.md @@ -337,11 +337,10 @@ Finally we need to run all of this on top of http4s. Here we have a simple `IOAp ```scala object Main extends IOApp { def run(args: List[String]): IO[ExitCode] = { - val starWarsGraphQLRoutes = GraphQLService.routes[IO]( - "starwars", - GraphQLService.fromMapping(new StarWarsMapping[IO] with StarWarsData[IO]) - ) - DemoServer.resource(starWarsGraphQLRoutes).useForever + (for { + starWarsRoutes <- StarWarsMapping[IO].map(mkRoutes("starwars")) + _ <- mkServer(starWarsRoutes) + } yield ()).useForever } } ```