-
Notifications
You must be signed in to change notification settings - Fork 1.1k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Implement match type amendment: extractors follow aliases and singlet…
…ons (#20161) This implements the change proposed in scala/improvement-proposals#84. The added pos test case presents motivating examples, the added neg test cases demonstrate that errors are correctly reported when cycles are present. The potential for cycle is no worse than with the existing extraction logic as demonstrated by the existing test in `tests/neg/mt-deskolemize.scala`.
- Loading branch information
Showing
5 changed files
with
190 additions
and
6 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,60 @@ | ||
//> using options -language:experimental.betterMatchTypeExtractors | ||
|
||
trait Expr: | ||
type Value | ||
object Expr: | ||
type Of[V] = Expr { type Value = V } | ||
type ExtractValue[F <: Expr] = F match | ||
case Expr.Of[v] => v | ||
import Expr.ExtractValue | ||
|
||
class SimpleLoop1 extends Expr: | ||
type Value = ExtractValue[SimpleLoop2] | ||
|
||
class SimpleLoop2 extends Expr: | ||
type Value = ExtractValue[SimpleLoop1] | ||
|
||
object Test1: | ||
val x: ExtractValue[SimpleLoop1] = 1 // error | ||
|
||
trait Description: | ||
type Elem <: Tuple | ||
|
||
class PrimBroken extends Expr: | ||
type Value = Alias | ||
type Alias = Value // error | ||
|
||
class Prim extends Expr: | ||
type Value = BigInt | ||
|
||
class VecExpr[E <: Expr] extends Expr: | ||
type Value = Vector[ExtractValue[E]] | ||
|
||
trait ProdExpr extends Expr: | ||
val description: Description | ||
type Value = Tuple.Map[description.Elem, [X] =>> ExtractValue[X & Expr]] | ||
|
||
|
||
class MyExpr1 extends ProdExpr: | ||
final val description = new Description: | ||
type Elem = (VecExpr[Prim], MyExpr2) | ||
|
||
class MyExpr2 extends ProdExpr: | ||
final val description = new Description: | ||
type Elem = (VecExpr[VecExpr[MyExpr1]], Prim) | ||
|
||
trait Constable[E <: Expr]: | ||
def lit(v: ExtractValue[E]): E | ||
object Constable: | ||
given [E <: Expr]: Constable[E] = ??? | ||
|
||
object Test2: | ||
def fromLiteral[E <: Expr : Constable](v: ExtractValue[E]): E = | ||
summon[Constable[E]].lit(v) | ||
val x0: ExtractValue[Prim] = "" // error | ||
val x1: ExtractValue[PrimBroken] = 1 // error | ||
|
||
val foo: MyExpr2 = new MyExpr2 | ||
val v: foo.Value = (Vector(Vector()), 1) // error: Recursion limit exceeded | ||
val c: MyExpr2 = fromLiteral: | ||
(Vector(Vector()), 1) // error: Recursion limit exceeded |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,57 @@ | ||
//> using options -language:experimental.betterMatchTypeExtractors | ||
|
||
trait Expr: | ||
type Value | ||
|
||
object Expr: | ||
type Of[V] = Expr { type Value = V } | ||
type ExtractValue[F <: Expr] = F match | ||
case Expr.Of[v] => v | ||
import Expr.ExtractValue | ||
|
||
class Prim extends Expr: | ||
type Value = Alias | ||
type Alias = BigInt | ||
|
||
class VecExpr[E <: Expr] extends Expr: | ||
type Value = Vector[ExtractValue[E]] | ||
|
||
trait Description: | ||
type Elem <: Tuple | ||
|
||
trait ProdExpr extends Expr: | ||
val description: Description | ||
type Value = Tuple.Map[description.Elem, [X] =>> ExtractValue[X & Expr]] | ||
|
||
class MyExpr1 extends ProdExpr: | ||
final val description = new Description: | ||
type Elem = (VecExpr[Prim], Prim) | ||
|
||
class MyExpr2 extends ProdExpr: | ||
final val description = new Description: | ||
type Elem = (VecExpr[VecExpr[MyExpr1]], Prim) | ||
|
||
trait ProdExprAlt[T <: Tuple] extends Expr: | ||
type Value = Tuple.Map[T, [X] =>> ExtractValue[X & Expr]] | ||
|
||
class MyExpr3 extends ProdExprAlt[(Prim, VecExpr[Prim], Prim)] | ||
|
||
trait Constable[E <: Expr]: | ||
def lit(v: ExtractValue[E]): E | ||
object Constable: | ||
given [E <: Expr]: Constable[E] = ??? | ||
|
||
object Test: | ||
def fromLiteral[E <: Expr : Constable](v: ExtractValue[E]): E = | ||
summon[Constable[E]].lit(v) | ||
val a: Prim = fromLiteral(1) | ||
val b: VecExpr[Prim] = fromLiteral(Vector(1)) | ||
val c: MyExpr1 = fromLiteral((Vector(1), 1)) | ||
val d: MyExpr2 = fromLiteral(Vector(Vector((Vector(1), 1))), 2) | ||
val e: MyExpr3 = fromLiteral((1, Vector(1), 1)) | ||
val f: ProdExprAlt[(MyExpr1, VecExpr[MyExpr3])] = fromLiteral: | ||
( | ||
(Vector(1), 1), | ||
Vector((1, Vector(1), 1), (2, Vector(1), 2)) | ||
) | ||
val g: Expr { type Alias = Int; type Value = Alias } = fromLiteral(1) |