Skip to content

Commit

Permalink
Access nested defaults through this to avoid stack overflow
Browse files Browse the repository at this point in the history
  • Loading branch information
joroKr21 committed Feb 11, 2022
1 parent df33efa commit e36b1a7
Show file tree
Hide file tree
Showing 2 changed files with 22 additions and 9 deletions.
21 changes: 12 additions & 9 deletions core/src/main/scala/shapeless/default.scala
Original file line number Diff line number Diff line change
Expand Up @@ -243,15 +243,18 @@ class DefaultMacros(val c: whitebox.Context) extends CaseClassMacros {
alt => alt.isMethod && hasDefaultParams(alt.asMethod)
}

def defaultsFor(fields: List[(TermName, Type)]) = for {
((_, argTpe), i) <- fields.zipWithIndex
default = tpe.companion.member(TermName(s"apply$$default$$${i + 1}")) orElse
altCompanion.member(TermName(s"$$lessinit$$greater$$default$$${i + 1}"))
} yield if (default.isTerm) {
val defaultTpe = appliedType(someTpe, devarargify(argTpe))
val defaultVal = some(q"$companion.$default")
(defaultTpe, defaultVal)
} else (noneTpe, none)
def defaultsFor(fields: List[(TermName, Type)]) = {
lazy val enclosing = ownerChain(c.internal.enclosingOwner)
for (((_, argTpe), i) <- fields.zipWithIndex) yield {
val default = tpe.companion.member(TermName(s"apply$$default$$${i + 1}")) orElse
altCompanion.member(TermName(s"$$lessinit$$greater$$default$$${i + 1}"))
if (default.isTerm) {
val owner = default.owner
val qualifier = if (!owner.isStatic && enclosing.contains(owner)) This(owner) else companion
(appliedType(someTpe, devarargify(argTpe)), some(q"$qualifier.$default"))
} else (noneTpe, none)
}
}

def mkDefault(defaults: List[(Type, Tree)]) = {
val (types, values) = defaults.unzip
Expand Down
10 changes: 10 additions & 0 deletions core/src/test/scala/shapeless/default.scala
Original file line number Diff line number Diff line change
Expand Up @@ -224,4 +224,14 @@ class DefaultTests {
assertTypedEquals[None.type :: HNil](None :: HNil, default2())
assertTypedEquals[HNil](HNil, default3())
}

@Test
def testInCompanion: Unit =
assertEquals(InCompanion.default, Some(9) :: HNil)

// Note: this has to be inside a class, not an object
case class InCompanion(a: Int = 9)
object InCompanion {
val default = Default[InCompanion].apply()
}
}

0 comments on commit e36b1a7

Please # to comment.