diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 2a35aa1..138f6e6 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -28,15 +28,21 @@ jobs: strategy: matrix: os: [ubuntu-latest] - scala: [2.12, 2.13] + scala: [2.12, 2.13, 3] java: [temurin@11, temurin@17] project: [rootJVM] exclude: - scala: 2.12 java: temurin@17 + - scala: 3 + java: temurin@17 runs-on: ${{ matrix.os }} timeout-minutes: 60 steps: + - name: Install sbt + if: contains(runner.os, 'macos') + run: brew install sbt + - name: Checkout current branch (full) uses: actions/checkout@v4 with: @@ -115,6 +121,10 @@ jobs: java: [temurin@11] runs-on: ${{ matrix.os }} steps: + - name: Install sbt + if: contains(runner.os, 'macos') + run: brew install sbt + - name: Checkout current branch (full) uses: actions/checkout@v4 with: @@ -166,6 +176,16 @@ jobs: tar xf targets.tar rm targets.tar + - name: Download target directories (3, rootJVM) + uses: actions/download-artifact@v4 + with: + name: target-${{ matrix.os }}-${{ matrix.java }}-3-rootJVM + + - name: Inflate target directories (3, rootJVM) + run: | + tar xf targets.tar + rm targets.tar + - name: Import signing key if: env.PGP_SECRET != '' && env.PGP_PASSPHRASE == '' env: @@ -199,6 +219,10 @@ jobs: java: [temurin@11] runs-on: ${{ matrix.os }} steps: + - name: Install sbt + if: contains(runner.os, 'macos') + run: brew install sbt + - name: Checkout current branch (full) uses: actions/checkout@v4 with: @@ -233,5 +257,5 @@ jobs: - name: Submit Dependencies uses: scalacenter/sbt-dependency-submission@v2 with: - modules-ignore: rootjs_2.12 rootjs_2.13 rootjvm_2.12 rootjvm_2.13 rootnative_2.12 rootnative_2.13 example1_2.12 example1_2.13 + modules-ignore: rootjs_2.12 rootjs_2.13 rootjs_3 rootjvm_2.12 rootjvm_2.13 rootjvm_3 rootnative_2.12 rootnative_2.13 rootnative_3 example1_2.12 example1_2.13 example1_3 configs-ignore: test scala-tool scala-doc-tool test-internal diff --git a/.gitignore b/.gitignore index a90a026..5efecf8 100644 --- a/.gitignore +++ b/.gitignore @@ -10,3 +10,7 @@ target/ .classpath tmp/ .bsp/ +.metals/ +.vscode/ +.jvm/ +*.semanticdb diff --git a/.scalafmt.conf b/.scalafmt.conf index 103c862..d95963f 100644 --- a/.scalafmt.conf +++ b/.scalafmt.conf @@ -15,3 +15,9 @@ rewrite.rules = [ AsciiSortImports, PreferCurlyFors ] + +fileOverride { + "glob:**/scala-3/**/*.scala" { + runner.dialect = scala3 + } +} \ No newline at end of file diff --git a/.travis.yml b/.travis.yml deleted file mode 100644 index 08173db..0000000 --- a/.travis.yml +++ /dev/null @@ -1,24 +0,0 @@ -language: scala -dist: xenial - -scala: - - 2.13.4 - - 2.12.12 - -jdk: - - openjdk8 - -cache: - directories: - - $HOME/.ivy2/cache - - $HOME/.sbt/boot - -install: - - pip install --user codecov - -script: - - sbt ++$TRAVIS_SCALA_VERSION clean coverage test coverageReport scalastyle scalafmtCheckAll scalafmtSbtCheck && codecov - - # See http://www.scala-sbt.org/0.13/docs/Travis-CI-with-sbt.html - # Tricks to avoid unnecessary cache updates - - find $HOME/.sbt -name "*.lock" | xargs rm diff --git a/build.sbt b/build.sbt index 29cac6d..5533f41 100644 --- a/build.sbt +++ b/build.sbt @@ -1,17 +1,18 @@ -ThisBuild / tlBaseVersion := "0.4" -ThisBuild / description := "Yet another Typesafe Config decoder" +ThisBuild / tlBaseVersion := "0.5" +ThisBuild / description := "Circe Golden Testing" ThisBuild / circeRootOfCodeCoverage := None ThisBuild / startYear := Some(2016) val scala212 = "2.12.19" -val scala213 = "2.13.11" +val scala213 = "2.13.14" +val scala3 = "3.3.3" ThisBuild / scalaVersion := scala213 -ThisBuild / crossScalaVersions := Seq(scala212, scala213) +ThisBuild / crossScalaVersions := Seq(scala212, scala213, scala3) -val circeVersion = "0.14.6" -val scalacheckVersion = "1.17.0" -val disciplineScalatestVersion = "2.2.0" -val scalacheckScalaTestVersion = "3.2.18.0" +val circeVersion = "0.14.9" +val scalacheckVersion = "1.18.0" +val disciplineScalatestVersion = "2.3.0" +val scalacheckScalaTestVersion = "3.2.19.0" val root = tlCrossRootProject.aggregate(golden, example1) @@ -21,15 +22,23 @@ lazy val golden = crossProject(JVMPlatform) .settings( moduleName := "circe-golden", libraryDependencies ++= Seq( - "org.scalatestplus" %%% "scalacheck-1-17" % scalacheckScalaTestVersion, + "org.scalatestplus" %%% "scalacheck-1-18" % scalacheckScalaTestVersion, "io.circe" %%% "circe-core" % circeVersion, "io.circe" %%% "circe-parser" % circeVersion, "io.circe" %%% "circe-testing" % circeVersion, "io.circe" %%% "circe-generic" % circeVersion % Test, "org.scalacheck" %% "scalacheck" % scalacheckVersion, - "org.typelevel" %%% "discipline-scalatest" % disciplineScalatestVersion % Test, - scalaOrganization.value % "scala-reflect" % scalaVersion.value % Provided - ) + "org.typelevel" %%% "discipline-scalatest" % disciplineScalatestVersion % Test + ), + libraryDependencies ++= { + if (tlIsScala3.value) Nil + else + Seq( + scalaOrganization.value % "scala-reflect" % scalaVersion.value % Provided + ) + }, + Test / classLoaderLayeringStrategy := ClassLoaderLayeringStrategy.Flat, + tlVersionIntroduced := List("2.13", "2.12", "3").map(_ -> "0.5.0").toMap ) lazy val example1 = crossProject(JVMPlatform) @@ -37,7 +46,7 @@ lazy val example1 = crossProject(JVMPlatform) .in(file("examples/example-1")) .settings( libraryDependencies ++= Seq( - "io.circe" %% "circe-core" % circeVersion, + "io.circe" %%% "circe-core" % circeVersion, "org.scalacheck" %% "scalacheck" % scalacheckVersion, "org.typelevel" %%% "discipline-scalatest" % disciplineScalatestVersion % Test ) @@ -46,16 +55,16 @@ lazy val example1 = crossProject(JVMPlatform) .dependsOn(golden % Test) ThisBuild / developers := List( - Developer( + tlGitHubDev( "travisbrown", - "Travis Brown", - "travisrobertbrown@gmail.com", - url("https://twitter.com/travisbrown") + "Travis Brown" ), - Developer( + tlGitHubDev( "zarthross", - "Darren Gibson", - "zarthross@gmail.com", - url("https://twitter.com/zarthross") + "Darren Gibson" + ), + tlGitHubDev( + "hamnis", + "Erlend Hamnaberg" ) ) diff --git a/flake.lock b/flake.lock deleted file mode 100644 index 80c450b..0000000 --- a/flake.lock +++ /dev/null @@ -1,156 +0,0 @@ -{ - "nodes": { - "devshell": { - "inputs": { - "flake-utils": "flake-utils", - "nixpkgs": "nixpkgs" - }, - "locked": { - "lastModified": 1708939976, - "narHash": "sha256-O5+nFozxz2Vubpdl1YZtPrilcIXPcRAjqNdNE8oCRoA=", - "owner": "numtide", - "repo": "devshell", - "rev": "5ddecd67edbd568ebe0a55905273e56cc82aabe3", - "type": "github" - }, - "original": { - "owner": "numtide", - "repo": "devshell", - "type": "github" - } - }, - "flake-utils": { - "inputs": { - "systems": "systems" - }, - "locked": { - "lastModified": 1701680307, - "narHash": "sha256-kAuep2h5ajznlPMD9rnQyffWG8EM/C73lejGofXvdM8=", - "owner": "numtide", - "repo": "flake-utils", - "rev": "4022d587cbbfd70fe950c1e2083a02621806a725", - "type": "github" - }, - "original": { - "owner": "numtide", - "repo": "flake-utils", - "type": "github" - } - }, - "flake-utils_2": { - "inputs": { - "systems": "systems_2" - }, - "locked": { - "lastModified": 1709126324, - "narHash": "sha256-q6EQdSeUZOG26WelxqkmR7kArjgWCdw5sfJVHPH/7j8=", - "owner": "numtide", - "repo": "flake-utils", - "rev": "d465f4819400de7c8d874d50b982301f28a84605", - "type": "github" - }, - "original": { - "owner": "numtide", - "repo": "flake-utils", - "type": "github" - } - }, - "nixpkgs": { - "locked": { - "lastModified": 1704161960, - "narHash": "sha256-QGua89Pmq+FBAro8NriTuoO/wNaUtugt29/qqA8zeeM=", - "owner": "NixOS", - "repo": "nixpkgs", - "rev": "63143ac2c9186be6d9da6035fa22620018c85932", - "type": "github" - }, - "original": { - "owner": "NixOS", - "ref": "nixpkgs-unstable", - "repo": "nixpkgs", - "type": "github" - } - }, - "nixpkgs_2": { - "locked": { - "lastModified": 1710097495, - "narHash": "sha256-B7Ea7q7hU7SE8wOPJ9oXEBjvB89yl2csaLjf5v/7jr8=", - "owner": "nixos", - "repo": "nixpkgs", - "rev": "d40e866b1f98698d454dad8f592fe7616ff705a4", - "type": "github" - }, - "original": { - "owner": "nixos", - "ref": "nixpkgs-unstable", - "repo": "nixpkgs", - "type": "github" - } - }, - "root": { - "inputs": { - "flake-utils": [ - "typelevel-nix", - "flake-utils" - ], - "nixpkgs": [ - "typelevel-nix", - "nixpkgs" - ], - "typelevel-nix": "typelevel-nix" - } - }, - "systems": { - "locked": { - "lastModified": 1681028828, - "narHash": "sha256-Vy1rq5AaRuLzOxct8nz4T6wlgyUR7zLU309k9mBC768=", - "owner": "nix-systems", - "repo": "default", - "rev": "da67096a3b9bf56a91d16901293e51ba5b49a27e", - "type": "github" - }, - "original": { - "owner": "nix-systems", - "repo": "default", - "type": "github" - } - }, - "systems_2": { - "locked": { - "lastModified": 1681028828, - "narHash": "sha256-Vy1rq5AaRuLzOxct8nz4T6wlgyUR7zLU309k9mBC768=", - "owner": "nix-systems", - "repo": "default", - "rev": "da67096a3b9bf56a91d16901293e51ba5b49a27e", - "type": "github" - }, - "original": { - "owner": "nix-systems", - "repo": "default", - "type": "github" - } - }, - "typelevel-nix": { - "inputs": { - "devshell": "devshell", - "flake-utils": "flake-utils_2", - "nixpkgs": "nixpkgs_2" - }, - "locked": { - "lastModified": 1710188850, - "narHash": "sha256-KbNmyxEvcnq5h/wfeL1ZxO9RwoNRjJ0IgYlUZpdSlLo=", - "owner": "typelevel", - "repo": "typelevel-nix", - "rev": "60c3868688cb8f5f7ebc781f6e122c061ae35d4d", - "type": "github" - }, - "original": { - "owner": "typelevel", - "repo": "typelevel-nix", - "type": "github" - } - } - }, - "root": "root", - "version": 7 -} diff --git a/flake.nix b/flake.nix deleted file mode 100644 index 667792a..0000000 --- a/flake.nix +++ /dev/null @@ -1,26 +0,0 @@ -{ - inputs = { - typelevel-nix.url = "github:typelevel/typelevel-nix"; - nixpkgs.follows = "typelevel-nix/nixpkgs"; - flake-utils.follows = "typelevel-nix/flake-utils"; - }; - - outputs = { self, nixpkgs, flake-utils, typelevel-nix }: - flake-utils.lib.eachDefaultSystem (system: - let - pkgs = import nixpkgs { - inherit system; - overlays = [ typelevel-nix.overlay ]; - }; - in { - devShell = pkgs.devshell.mkShell { - imports = [ typelevel-nix.typelevelShell ]; - name = "circe-golden"; - typelevelShell = { - jdk.package = pkgs.jdk11; - native.enable = false; - nodejs.enable = false; - }; - }; - }); -} diff --git a/golden/src/main/scala-2/io/circe/testing/golden/GoldenCodecTestsCompanion.scala b/golden/src/main/scala-2/io/circe/testing/golden/GoldenCodecTestsCompanion.scala new file mode 100644 index 0000000..283569d --- /dev/null +++ b/golden/src/main/scala-2/io/circe/testing/golden/GoldenCodecTestsCompanion.scala @@ -0,0 +1,39 @@ +/* + * Copyright 2016 circe + * + * 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 io.circe.testing.golden + +import io.circe.Decoder +import io.circe.Encoder +import io.circe.Printer +import org.scalacheck.Arbitrary + +import scala.reflect.runtime.universe.TypeTag + +trait GoldenCodecTestsCompanion { self: GoldenCodecTests.type => + def apply[A: Decoder: Encoder: Arbitrary: TypeTag]: GoldenCodecTests[A] = + fromLaws[A](ResourceFileGoldenCodecLaws[A]()) + + def apply[A: Decoder: Encoder: Arbitrary: TypeTag](printer: Printer): GoldenCodecTests[A] = + fromLaws[A](ResourceFileGoldenCodecLaws[A](printer = printer)) + + def apply[A: Decoder: Encoder: Arbitrary: TypeTag](count: Int): GoldenCodecTests[A] = + fromLaws[A](ResourceFileGoldenCodecLaws[A](count = count)) + + def apply[A: Decoder: Encoder: Arbitrary: TypeTag](count: Int, printer: Printer): GoldenCodecTests[A] = + fromLaws[A](ResourceFileGoldenCodecLaws[A](count = count, printer = printer)) + +} diff --git a/golden/src/main/scala-2/io/circe/testing/golden/ResourceFileGoldenCodecLawsCompanion.scala b/golden/src/main/scala-2/io/circe/testing/golden/ResourceFileGoldenCodecLawsCompanion.scala new file mode 100644 index 0000000..8410840 --- /dev/null +++ b/golden/src/main/scala-2/io/circe/testing/golden/ResourceFileGoldenCodecLawsCompanion.scala @@ -0,0 +1,46 @@ +/* + * Copyright 2016 circe + * + * 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 io.circe.testing.golden + +import io.circe.Decoder +import io.circe.Encoder +import io.circe.Printer +import org.scalacheck.Arbitrary + +import scala.reflect.runtime.universe.TypeTag + +trait ResourceFileGoldenCodecLawsCompanion { self: ResourceFileGoldenCodecLaws.type => + + def apply[A]( + size: Int = 100, + count: Int = 1, + printer: Printer = Printer.spaces2 + )(implicit + decodeA: Decoder[A], + encodeA: Encoder[A], + arbitraryA: Arbitrary[A], + typeTagA: TypeTag[A] + ): GoldenCodecLaws[A] = + self.apply[A]( + Scala2Naming.inferName[A], + Resources.inferRootDir(typeTagA.mirror.runtimeClass(typeTagA.tpe)), + Scala2Naming.inferPackage[A], + size, + count, + printer + ) +} diff --git a/golden/src/main/scala-2/io/circe/testing/golden/Scala2Naming.scala b/golden/src/main/scala-2/io/circe/testing/golden/Scala2Naming.scala new file mode 100644 index 0000000..0c2818a --- /dev/null +++ b/golden/src/main/scala-2/io/circe/testing/golden/Scala2Naming.scala @@ -0,0 +1,47 @@ +/* + * Copyright 2016 circe + * + * 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 io.circe.testing.golden + +import scala.reflect.runtime.universe.Symbol +import scala.reflect.runtime.universe.Type +import scala.reflect.runtime.universe.TypeTag + +object Scala2Naming { + + /** + * Attempt to guess the packaging of the type indicated by the provided type tag. + */ + def inferPackage[A](implicit A: TypeTag[A]): List[String] = + owners(A.tpe).collectFirst { case s if s.isPackage => s.fullName.split('.').toList }.getOrElse(List.empty) + + private def owners(tpe: Type): Iterator[Symbol] = + Iterator.iterate(tpe.typeSymbol)(_.owner) + + /** + * Attempt to guess the name of the type indicated by the provided type tag. + */ + def inferName[A](implicit A: TypeTag[A]): String = inferNameForType(A.tpe) + + private def baseSymbols(tpe: Type): List[Symbol] = + owners(tpe).takeWhile(!_.isPackage).toList.reverse + + private def inferNameForType(tpe: Type): String = { + val baseNames = baseSymbols(tpe).map(_.name.decodedName.toString) + + (baseNames ::: tpe.typeArgs.map(inferNameForType)).mkString("_") + } +} diff --git a/golden/src/main/scala-3/io/circe/testing/golden/GoldenCodecTestsCompanion.scala b/golden/src/main/scala-3/io/circe/testing/golden/GoldenCodecTestsCompanion.scala new file mode 100644 index 0000000..c8d043a --- /dev/null +++ b/golden/src/main/scala-3/io/circe/testing/golden/GoldenCodecTestsCompanion.scala @@ -0,0 +1,38 @@ +/* + * Copyright 2016 circe + * + * 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 io.circe.testing.golden + +import io.circe.Decoder +import io.circe.Encoder +import io.circe.Printer +import org.scalacheck.Arbitrary + +import scala.reflect.ClassTag + +trait GoldenCodecTestsCompanion { self: GoldenCodecTests.type => + inline def apply[A: Decoder: Encoder: Arbitrary: ClassTag]: GoldenCodecTests[A] = + fromLaws[A](ResourceFileGoldenCodecLaws[A]()) + + inline def apply[A: Decoder: Encoder: Arbitrary: ClassTag](printer: Printer): GoldenCodecTests[A] = + fromLaws[A](ResourceFileGoldenCodecLaws[A](printer = printer)) + + inline def apply[A: Decoder: Encoder: Arbitrary: ClassTag](count: Int): GoldenCodecTests[A] = + fromLaws[A](ResourceFileGoldenCodecLaws[A](count = count)) + + inline def apply[A: Decoder: Encoder: Arbitrary: ClassTag](count: Int, printer: Printer): GoldenCodecTests[A] = + fromLaws[A](ResourceFileGoldenCodecLaws[A](count = count, printer = printer)) +} diff --git a/golden/src/main/scala-3/io/circe/testing/golden/ResourceFileGoldenCodecLawsCompanion.scala b/golden/src/main/scala-3/io/circe/testing/golden/ResourceFileGoldenCodecLawsCompanion.scala new file mode 100644 index 0000000..535209e --- /dev/null +++ b/golden/src/main/scala-3/io/circe/testing/golden/ResourceFileGoldenCodecLawsCompanion.scala @@ -0,0 +1,46 @@ +/* + * Copyright 2016 circe + * + * 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 io.circe.testing.golden + +import io.circe.Decoder +import io.circe.Encoder +import io.circe.Printer +import org.scalacheck.Arbitrary + +import scala.reflect.ClassTag + +trait ResourceFileGoldenCodecLawsCompanion { self: ResourceFileGoldenCodecLaws.type => + + inline def apply[A]( + size: Int = 100, + count: Int = 1, + printer: Printer = Printer.spaces2 + )(using + decodeA: Decoder[A], + encodeA: Encoder[A], + arbitraryA: Arbitrary[A], + classT: ClassTag[A] + ): GoldenCodecLaws[A] = + self.apply[A]( + Scala3Naming.name[A], + Resources.inferRootDir(classT.runtimeClass), + Scala3Naming.pkg[A], + size, + count, + printer + ) +} diff --git a/golden/src/main/scala-3/io/circe/testing/golden/Scala3Naming.scala b/golden/src/main/scala-3/io/circe/testing/golden/Scala3Naming.scala new file mode 100644 index 0000000..2b65bce --- /dev/null +++ b/golden/src/main/scala-3/io/circe/testing/golden/Scala3Naming.scala @@ -0,0 +1,48 @@ +/* + * Copyright 2016 circe + * + * 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 io.circe.testing.golden + +import scala.quoted.* +object Scala3Naming { + + inline def name[T]: String = ${ Scala3Naming.nameImpl[T] } + inline def pkg[T]: List[String] = ${ Scala3Naming.packageImpl[T] } + + def nameImpl[T: Type](using Quotes): Expr[String] = { + import quotes.reflect.* + + val typeRepr = TypeRepr.of[T] + + def expandName(repr: TypeRepr, names: List[String]): List[String] = { + val args = repr.typeArgs + repr.typeSymbol.name :: (if args.isEmpty then names else args.flatMap(x => expandName(x, names))) + } + + Expr(expandName(typeRepr, Nil).reverse.mkString("_")) + } + + def packageImpl[T: Type](using q: Quotes): Expr[List[String]] = { + import q.reflect.* + val repr = TypeRepr.of[T] + val symbol = repr.typeSymbol + + def expandPkg(s: Symbol, names: List[String]): List[String] = + if s.isNoSymbol || s.name == "" then names + else expandPkg(s.maybeOwner, if s.isPackageDef then s.name :: names else names) + Expr(expandPkg(symbol, Nil)) + } +} diff --git a/golden/src/main/scala/io/circe/testing/golden/GoldenCodecTests.scala b/golden/src/main/scala/io/circe/testing/golden/GoldenCodecTests.scala index 56452b6..472290c 100644 --- a/golden/src/main/scala/io/circe/testing/golden/GoldenCodecTests.scala +++ b/golden/src/main/scala/io/circe/testing/golden/GoldenCodecTests.scala @@ -20,16 +20,12 @@ import cats.instances.string._ import cats.kernel.Eq import cats.laws.IsEq import cats.laws.discipline.catsLawsIsEqToProp -import io.circe.Decoder -import io.circe.Encoder import io.circe.Json -import io.circe.Printer import io.circe.testing.CodecTests import org.scalacheck.Arbitrary import org.scalacheck.Prop import org.scalacheck.Shrink -import scala.reflect.runtime.universe.TypeTag import scala.util.Failure import scala.util.Success import scala.util.Try @@ -69,20 +65,11 @@ trait GoldenCodecTests[A] extends CodecTests[A] { ) } -object GoldenCodecTests { - def apply[A: Decoder: Encoder: Arbitrary: TypeTag]: GoldenCodecTests[A] = - apply[A](ResourceFileGoldenCodecLaws[A]()) +object GoldenCodecTests extends GoldenCodecTestsCompanion { - def apply[A: Decoder: Encoder: Arbitrary: TypeTag](printer: Printer): GoldenCodecTests[A] = - apply[A](ResourceFileGoldenCodecLaws[A](printer = printer)) - - def apply[A: Decoder: Encoder: Arbitrary: TypeTag](count: Int): GoldenCodecTests[A] = - apply[A](ResourceFileGoldenCodecLaws[A](count = count)) - - def apply[A: Decoder: Encoder: Arbitrary: TypeTag](count: Int, printer: Printer): GoldenCodecTests[A] = - apply[A](ResourceFileGoldenCodecLaws[A](count = count, printer = printer)) - - def apply[A](laws0: GoldenCodecLaws[A]): GoldenCodecTests[A] = + // for binary compat + protected def apply[A](laws0: GoldenCodecLaws[A]): GoldenCodecTests[A] = fromLaws(laws0) + def fromLaws[A](laws0: GoldenCodecLaws[A]): GoldenCodecTests[A] = new GoldenCodecTests[A] { val laws: GoldenCodecLaws[A] = laws0 } diff --git a/golden/src/main/scala/io/circe/testing/golden/ResourceFileGoldenCodecLaws.scala b/golden/src/main/scala/io/circe/testing/golden/ResourceFileGoldenCodecLaws.scala index f1a977e..33355e1 100644 --- a/golden/src/main/scala/io/circe/testing/golden/ResourceFileGoldenCodecLaws.scala +++ b/golden/src/main/scala/io/circe/testing/golden/ResourceFileGoldenCodecLaws.scala @@ -28,7 +28,6 @@ import org.scalacheck.Gen import java.io.File import java.io.PrintWriter -import scala.reflect.runtime.universe.TypeTag import scala.util.Failure import scala.util.Try import scala.util.matching.Regex @@ -98,7 +97,7 @@ abstract class ResourceFileGoldenCodecLaws[A]( loadGoldenFiles.flatMap(fs => if (fs.isEmpty) generateGoldenFiles else loadGoldenFiles) } -object ResourceFileGoldenCodecLaws { +object ResourceFileGoldenCodecLaws extends ResourceFileGoldenCodecLawsCompanion { def apply[A]( name: String, resourceRootDir: File, @@ -112,16 +111,4 @@ object ResourceFileGoldenCodecLaws { val encode: Encoder[A] = encodeA val gen: Gen[A] = arbitraryA.arbitrary } - - def apply[A]( - size: Int = 100, - count: Int = 1, - printer: Printer = Printer.spaces2 - )(implicit - decodeA: Decoder[A], - encodeA: Encoder[A], - arbitraryA: Arbitrary[A], - typeTagA: TypeTag[A] - ): GoldenCodecLaws[A] = - apply[A](Resources.inferName[A], Resources.inferRootDir, Resources.inferPackage[A], size, count, printer) } diff --git a/golden/src/main/scala/io/circe/testing/golden/Resources.scala b/golden/src/main/scala/io/circe/testing/golden/Resources.scala index d41040c..1444522 100644 --- a/golden/src/main/scala/io/circe/testing/golden/Resources.scala +++ b/golden/src/main/scala/io/circe/testing/golden/Resources.scala @@ -18,21 +18,18 @@ package io.circe.testing.golden import java.io.File import scala.io.Source -import scala.reflect.runtime.universe.Symbol -import scala.reflect.runtime.universe.Type -import scala.reflect.runtime.universe.TypeTag import scala.util.Try /** - * Miscellaneous utilities for guessing resource locations, names, etc. + * Miscellaneous utilities for guessing resource locations */ object Resources { /** * Attempt to guess the test resource root directory for the current project, creating it if it does not exist. */ - lazy val inferRootDir: File = { - var current = new File(getClass.getResource("/").toURI) + def inferRootDir(cls: Class[_]): File = { + var current = new File(cls.getProtectionDomain.getCodeSource.getLocation.getFile) while (current.ne(null) && current.getName != "target") current = current.getParentFile @@ -43,29 +40,6 @@ object Resources { resourceDir } - /** - * Attempt to guess the packaging of the type indicated by the provided type tag. - */ - def inferPackage[A](implicit A: TypeTag[A]): List[String] = - owners(A.tpe).collectFirst { case s if s.isPackage => s.fullName.split('.').toList }.getOrElse(List.empty) - - private def owners(tpe: Type): Iterator[Symbol] = - Iterator.iterate(tpe.typeSymbol)(_.owner) - - /** - * Attempt to guess the name of the type indicated by the provided type tag. - */ - def inferName[A](implicit A: TypeTag[A]): String = inferNameForType(A.tpe) - - private def baseSymbols(tpe: Type): List[Symbol] = - owners(tpe).takeWhile(!_.isPackage).toList.reverse - - private def inferNameForType(tpe: Type): String = { - val baseNames = baseSymbols(tpe).map(_.name.decodedName.toString) - - (baseNames ::: tpe.typeArgs.map(inferNameForType)).mkString("_") - } - def open(path: String): Try[Source] = Try( Source.fromInputStream(getClass.getResourceAsStream(path)) ) diff --git a/project/build.properties b/project/build.properties index 04267b1..ee4c672 100644 --- a/project/build.properties +++ b/project/build.properties @@ -1 +1 @@ -sbt.version=1.9.9 +sbt.version=1.10.1 diff --git a/project/plugins.sbt b/project/plugins.sbt index 50c0653..26771e3 100644 --- a/project/plugins.sbt +++ b/project/plugins.sbt @@ -1 +1 @@ -addSbtPlugin("io.circe" % "sbt-circe-org" % "0.3.1") +addSbtPlugin("io.circe" % "sbt-circe-org" % "0.4.2") diff --git a/scalastyle-config.xml b/scalastyle-config.xml deleted file mode 100644 index 533506a..0000000 --- a/scalastyle-config.xml +++ /dev/null @@ -1,85 +0,0 @@ - - Circe Configuration - - - FOR - IF - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - true - - - - - all - .+ - - -