diff --git a/src/main/g8/$if(graal_native_image.truthy)$native-image-readme.md$endif$ b/src/main/g8/$if(graal_native_image.truthy)$native-image-readme.md$endif$ deleted file mode 100644 index e022957d..00000000 --- a/src/main/g8/$if(graal_native_image.truthy)$native-image-readme.md$endif$ +++ /dev/null @@ -1,6 +0,0 @@ -You can build a native-image binary as mentioned in the http4s deployment [section] (https://github.com/drocsid/http4s/blob/docs/deployment/docs/src/main/tut/deployment.md) . You will need to follow the directions there to provide GraalVM / native-image plugin and provide a muslC bundle. Then populate the UseMuslC path with it's location. - -``` -native-image $if(is_linux_build.truthy)$--static -H:UseMuslC="/path.to/muslC"$endif$ -H:+ReportExceptionStackTraces -H:+AddAllCharsets --allow-incomplete-classpath --no-fallback --initialize-at-build-time --enable-http --enable-https --enable-all-security-services --verbose -jar "./target/$scala_assembly_target$/$name$-assembly-0.0.1-SNAPSHOT.jar" $name$BinaryImage -``` - diff --git a/src/main/g8/build.sbt b/src/main/g8/build.sbt index 573c0807..b3a9bc17 100644 --- a/src/main/g8/build.sbt +++ b/src/main/g8/build.sbt @@ -9,22 +9,16 @@ lazy val root = (project in file(".")) organization := "$organization$", name := "$name;format="norm"$", version := "0.0.1-SNAPSHOT", - scalaVersion := "$scala_version$", + scalaVersion := "3.3.3", libraryDependencies ++= Seq( "org.http4s" %% "http4s-ember-server" % Http4sVersion, "org.http4s" %% "http4s-ember-client" % Http4sVersion, "org.http4s" %% "http4s-circe" % Http4sVersion, "org.http4s" %% "http4s-dsl" % Http4sVersion, - "io.circe" %% "circe-generic" % CirceVersion, "org.scalameta" %% "munit" % MunitVersion % Test, "org.typelevel" %% "munit-cats-effect" % MunitCatsEffectVersion % Test, "ch.qos.logback" % "logback-classic" % LogbackVersion % Runtime, - $if(graal_native_image.truthy)$ - "org.scalameta" % "svm-subs" % "101.0.0" - $endif$ ), - addCompilerPlugin("org.typelevel" %% "kind-projector" % "0.13.3" cross CrossVersion.full), - addCompilerPlugin("com.olegpy" %% "better-monadic-for" % "0.3.1"), assembly / assemblyMergeStrategy := { case "module-info.class" => MergeStrategy.discard case x => (assembly / assemblyMergeStrategy).value.apply(x) diff --git a/src/main/g8/default.properties b/src/main/g8/default.properties index 00691a53..e4b678d8 100644 --- a/src/main/g8/default.properties +++ b/src/main/g8/default.properties @@ -1,9 +1,3 @@ name = quickstart organization = com.example package = $organization$.$name;format="norm,word"$ - -scala_version = 2.13.14 -# graal_vm_specific -graal_native_image = true -is_linux_build = false -scala_assembly_target = scala-2.13 diff --git a/src/main/g8/src/main/resources/$if(graal_native_image.truthy)$META-INF$endif$/native-image/$package__packaged$/reflect-config.json b/src/main/g8/src/main/resources/$if(graal_native_image.truthy)$META-INF$endif$/native-image/$package__packaged$/reflect-config.json deleted file mode 100644 index d297bd6a..00000000 --- a/src/main/g8/src/main/resources/$if(graal_native_image.truthy)$META-INF$endif$/native-image/$package__packaged$/reflect-config.json +++ /dev/null @@ -1,180 +0,0 @@ -[ - { - "name": "org.slf4j.impl.StaticLoggerBinder", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.DateConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.MessageConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.ThrowableProxyConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.NopThrowableInformationConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.ContextNameConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.color.BoldYellowCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.LoggerConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.ReplacingCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.color.BoldBlueCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.color.CyanCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.color.RedCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.color.WhiteCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.PropertyConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.ExtendedThrowableProxyConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.RootCauseFirstThrowableProxyConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.MethodOfCallerConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.LevelConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.IdentityCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.color.BoldWhiteCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.MarkerConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.color.BoldCyanCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.color.BoldMagentaCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.RelativeTimeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.color.MagentaCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.ClassOfCallerConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.LineOfCallerConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.FileOfCallerConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.color.BoldGreenCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.LocalSequenceNumberConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.color.YellowCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.ExtendedThrowableProxyConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.color.HighlightingCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.color.GrayCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.MDCConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.ClassOfCallerConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.color.BoldRedCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.color.GreenCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.pattern.color.BlackCompositeConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.ThreadConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.pattern.LineSeparatorConverter", - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.classic.encoder.PatternLayoutEncoder", - "allPublicMethods":true, - "allDeclaredConstructors": true - }, - { - "name": "ch.qos.logback.core.ConsoleAppender", - "allPublicMethods":true, - "allDeclaredConstructors": true - }, - { - "name": "com.sun.org.apache.xerces.internal.jaxp.SAXParserFactoryImpl", - "allDeclaredConstructors": true - } -] diff --git a/src/main/g8/src/main/scala/$package__packaged$/$name__Camel$Routes.scala b/src/main/g8/src/main/scala/$package__packaged$/$name__Camel$Routes.scala index 7560df86..f25c1102 100644 --- a/src/main/g8/src/main/scala/$package__packaged$/$name__Camel$Routes.scala +++ b/src/main/g8/src/main/scala/$package__packaged$/$name__Camel$Routes.scala @@ -1,15 +1,15 @@ package $package$ import cats.effect.Sync -import cats.implicits._ +import cats.syntax.all.* import org.http4s.HttpRoutes import org.http4s.dsl.Http4sDsl -object $name;format="Camel"$Routes { +object $name;format="Camel"$Routes: - def jokeRoutes[F[_]: Sync](J: Jokes[F]): HttpRoutes[F] = { + def jokeRoutes[F[_]: Sync](J: Jokes[F]): HttpRoutes[F] = val dsl = new Http4sDsl[F]{} - import dsl._ + import dsl.* HttpRoutes.of[F] { case GET -> Root / "joke" => for { @@ -17,11 +17,10 @@ object $name;format="Camel"$Routes { resp <- Ok(joke) } yield resp } - } - def helloWorldRoutes[F[_]: Sync](H: HelloWorld[F]): HttpRoutes[F] = { + def helloWorldRoutes[F[_]: Sync](H: HelloWorld[F]): HttpRoutes[F] = val dsl = new Http4sDsl[F]{} - import dsl._ + import dsl.* HttpRoutes.of[F] { case GET -> Root / "hello" / name => for { @@ -29,5 +28,3 @@ object $name;format="Camel"$Routes { resp <- Ok(greeting) } yield resp } - } -} \ No newline at end of file diff --git a/src/main/g8/src/main/scala/$package__packaged$/$name__Camel$Server.scala b/src/main/g8/src/main/scala/$package__packaged$/$name__Camel$Server.scala index 89b84666..40a8884d 100644 --- a/src/main/g8/src/main/scala/$package__packaged$/$name__Camel$Server.scala +++ b/src/main/g8/src/main/scala/$package__packaged$/$name__Camel$Server.scala @@ -1,15 +1,15 @@ package $package$ import cats.effect.Async -import cats.syntax.all._ -import com.comcast.ip4s._ +import cats.syntax.all.* +import com.comcast.ip4s.* import fs2.io.net.Network import org.http4s.ember.client.EmberClientBuilder import org.http4s.ember.server.EmberServerBuilder -import org.http4s.implicits._ +import org.http4s.implicits.* import org.http4s.server.middleware.Logger -object $name;format="Camel"$Server { +object $name;format="Camel"$Server: def run[F[_]: Async: Network]: F[Nothing] = { for { @@ -37,4 +37,3 @@ object $name;format="Camel"$Server { .build } yield () }.useForever -} diff --git a/src/main/g8/src/main/scala/$package__packaged$/HelloWorld.scala b/src/main/g8/src/main/scala/$package__packaged$/HelloWorld.scala index d0fa790b..65bee056 100644 --- a/src/main/g8/src/main/scala/$package__packaged$/HelloWorld.scala +++ b/src/main/g8/src/main/scala/$package__packaged$/HelloWorld.scala @@ -1,16 +1,15 @@ package $package$ import cats.Applicative -import cats.implicits._ +import cats.syntax.all.* import io.circe.{Encoder, Json} import org.http4s.EntityEncoder -import org.http4s.circe._ +import org.http4s.circe.* -trait HelloWorld[F[_]]{ +trait HelloWorld[F[_]]: def hello(n: HelloWorld.Name): F[HelloWorld.Greeting] -} -object HelloWorld { +object HelloWorld: final case class Name(name: String) extends AnyVal /** * More generally you will want to decouple your edge representations from @@ -18,18 +17,15 @@ object HelloWorld { * create encoders for your data. **/ final case class Greeting(greeting: String) extends AnyVal - object Greeting { - implicit val greetingEncoder: Encoder[Greeting] = new Encoder[Greeting] { + object Greeting: + given Encoder[Greeting] = new Encoder[Greeting]: final def apply(a: Greeting): Json = Json.obj( ("message", Json.fromString(a.greeting)), ) - } - implicit def greetingEntityEncoder[F[_]]: EntityEncoder[F, Greeting] = + + given [F[_]]: EntityEncoder[F, Greeting] = jsonEncoderOf[F, Greeting] - } - def impl[F[_]: Applicative]: HelloWorld[F] = new HelloWorld[F]{ + def impl[F[_]: Applicative]: HelloWorld[F] = new HelloWorld[F]: def hello(n: HelloWorld.Name): F[HelloWorld.Greeting] = Greeting("Hello, " + n.name).pure[F] - } -} diff --git a/src/main/g8/src/main/scala/$package__packaged$/Jokes.scala b/src/main/g8/src/main/scala/$package__packaged$/Jokes.scala index edd18b01..88af5110 100644 --- a/src/main/g8/src/main/scala/$package__packaged$/Jokes.scala +++ b/src/main/g8/src/main/scala/$package__packaged$/Jokes.scala @@ -1,41 +1,33 @@ package $package$ import cats.effect.Concurrent -import cats.implicits._ +import cats.syntax.all.* import io.circe.{Encoder, Decoder} -import io.circe.generic.semiauto._ -import org.http4s._ -import org.http4s.implicits._ +import org.http4s.* +import org.http4s.implicits.* import org.http4s.client.Client import org.http4s.client.dsl.Http4sClientDsl -import org.http4s.circe._ -import org.http4s.Method._ +import org.http4s.circe.* +import org.http4s.Method.* -trait Jokes[F[_]]{ +trait Jokes[F[_]]: def get: F[Jokes.Joke] -} -object Jokes { - def apply[F[_]](implicit ev: Jokes[F]): Jokes[F] = ev +object Jokes: + def apply[F[_]](using ev: Jokes[F]): Jokes[F] = ev - final case class Joke(joke: String) extends AnyVal - object Joke { - implicit val jokeDecoder: Decoder[Joke] = deriveDecoder[Joke] - implicit def jokeEntityDecoder[F[_]: Concurrent]: EntityDecoder[F, Joke] = - jsonOf - implicit val jokeEncoder: Encoder[Joke] = deriveEncoder[Joke] - implicit def jokeEntityEncoder[F[_]]: EntityEncoder[F, Joke] = - jsonEncoderOf - } + final case class Joke(joke: String) + object Joke: + given Decoder[Joke] = Decoder.derived[Joke] + given [F[_]: Concurrent]: EntityDecoder[F, Joke] = jsonOf + given Encoder[Joke] = Encoder.AsObject.derived[Joke] + given [F[_]]: EntityEncoder[F, Joke] = jsonEncoderOf final case class JokeError(e: Throwable) extends RuntimeException - def impl[F[_]: Concurrent](C: Client[F]): Jokes[F] = new Jokes[F]{ + def impl[F[_]: Concurrent](C: Client[F]): Jokes[F] = new Jokes[F]: val dsl = new Http4sClientDsl[F]{} - import dsl._ - def get: F[Jokes.Joke] = { + import dsl.* + def get: F[Jokes.Joke] = C.expect[Joke](GET(uri"https://icanhazdadjoke.com/")) .adaptError{ case t => JokeError(t)} // Prevent Client Json Decoding Failure Leaking - } - } -} diff --git a/src/main/g8/src/main/scala/$package__packaged$/Main.scala b/src/main/g8/src/main/scala/$package__packaged$/Main.scala index a6ded2d2..abdddbb1 100644 --- a/src/main/g8/src/main/scala/$package__packaged$/Main.scala +++ b/src/main/g8/src/main/scala/$package__packaged$/Main.scala @@ -2,6 +2,5 @@ package $package$ import cats.effect.{IO, IOApp} -object Main extends IOApp.Simple { +object Main extends IOApp.Simple: val run = $name;format="Camel"$Server.run[IO] -} diff --git a/src/main/g8/src/test/scala/$package__packaged$/HelloWorldSpec.scala b/src/main/g8/src/test/scala/$package__packaged$/HelloWorldSpec.scala index 56d4ba92..a50a89ff 100644 --- a/src/main/g8/src/test/scala/$package__packaged$/HelloWorldSpec.scala +++ b/src/main/g8/src/test/scala/$package__packaged$/HelloWorldSpec.scala @@ -1,11 +1,11 @@ package $package$ import cats.effect.IO -import org.http4s._ -import org.http4s.implicits._ +import org.http4s.* +import org.http4s.implicits.* import munit.CatsEffectSuite -class HelloWorldSpec extends CatsEffectSuite { +class HelloWorldSpec extends CatsEffectSuite: test("HelloWorld returns status code 200") { assertIO(retHelloWorld.map(_.status) ,Status.Ok) @@ -15,9 +15,7 @@ class HelloWorldSpec extends CatsEffectSuite { assertIO(retHelloWorld.flatMap(_.as[String]), "{\"message\":\"Hello, world\"}") } - private[this] val retHelloWorld: IO[Response[IO]] = { + private[this] val retHelloWorld: IO[Response[IO]] = val getHW = Request[IO](Method.GET, uri"/hello/world") val helloWorld = HelloWorld.impl[IO] $name;format="Camel"$Routes.helloWorldRoutes(helloWorld).orNotFound(getHW) - } -} \ No newline at end of file