| id | allows |
|---|---|
| title | Allows |
import Tabs from '@theme/Tabs'; import TabItem from '@theme/TabItem';
Allows[A, S] is a compile-time capability token that proves, at the call site, that type A satisfies the structural grammar S. A capability token is a compile-time phantom proof value — it carries no runtime data and exists solely to pass evidence through the type system that a structural constraint has been satisfied.
Allows does not require or use Schema[A]. It inspects the Scala type structure of A directly at compile time, using nothing but the Scala type system. Any Schema[A] that appears alongside Allows in examples is the library author's own separate constraint — it is not imposed by Allows itself.
sealed abstract class Allows[A, S <: Allows.Structural]The gap Allows fills is structural preconditions at the call site, at compile time, with precise error messages. Structural preconditions are constraints on the shape of a type's fields (e.g., "all fields must be scalars"), unlike runtime checks which happen during execution and produce exceptions or errors.
ZIO Blocks gives library authors a powerful way to build data-oriented DSLs. A library can accept A: Schema and use the schema at runtime to serialize, deserialize, query, or transform values of A. A data-oriented DSL is a generic API built around a data description (Schema) rather than a fixed interface, allowing one function to serialize, validate, or transform any conforming type. Many generic functions have structural preconditions that don't require a schema.
Consider these real-world scenarios:
- A CSV serializer requires flat records of scalars — nested records should fail at the call site, not deep inside the serializer.
- An RDBMS layer cannot handle nested records as column values — the error should name the problematic field.
- An event bus expects a sealed trait of flat record cases — violations should be caught before publishing.
- A JSON document store allows arbitrarily nested records but not
DynamicValueleaves — the schema validation should be precise. DynamicValue is the schema-less escape hatch that can hold arbitrary data — a DynamicValue leaf bypasses compile-time checking entirely, making it impossible for the compiler to enforce any structural grammar.
Without Allows, these constraints can only be checked at runtime, producing confusing errors deep inside library internals. With Allows[A, S], the constraint is verified at the call site, at compile time, with precise, path-aware error messages and concrete fix suggestions.
Allows[A, S] is an upper bound. A type A that uses only a strict subset of what S permits also satisfies it — just as A <: Foo does not require that A uses every method of Foo. Upper bound semantics is the right choice because a lower bound would require using every shape (impractical), exact matching would require naming every shape used (too rigid), whereas upper bound says "your type may use any of these shapes" — a permission, not a mandate.
import zio.blocks.schema.comptime.Allows
import Allows._
// Both satisfy Record[Primitive | Optional[Primitive]] — the upper bound
case class UserRow(name: String, age: Int)
// UserRow satisfies the grammar: all fields are Primitive
case class UserRowOpt(name: String, age: Int, email: Option[String])
// UserRowOpt also satisfies the grammar: all fields are Primitive or Optional[Primitive]
val ev1: Allows[UserRow, Record[Primitive | Optional[Primitive]]] = implicitly
val ev2: Allows[UserRowOpt, Record[Primitive | Optional[Primitive]]] = implicitlyAllows[A, S] is not instantiated directly. Instead, you summon an evidence value at the point where you need the constraint. The macro automatically verifies the constraint at compile time.
import zio.blocks.schema.comptime.Allows
import Allows._
def toJson[A](doc: A)(implicit ev: Allows[A, Record[Primitive]]): String = ???
// Or summon at the call site:
val evidence = implicitly[Allows[Int, Primitive]]import zio.blocks.schema.comptime.Allows
import Allows._
def toJson[A](doc: A)(using Allows[A, Record[Primitive]]): String = ???
// Calling the function:
case class Person(name: String, age: Int)
val json = toJson(Person("Alice", 30)) // Compiles if Person satisfies Record[Primitive]The constraint is checked once, at the call site. If the type A does not satisfy S, you get a compile-time error with a precise message showing exactly which field violates the grammar.
All grammar nodes extend Allows.Structural.
| Node | Matches |
|---|---|
Primitive |
Any scalar — catch-all for all 30 Schema 2 primitive types |
Primitive.Boolean |
scala.Boolean only |
Primitive.Int |
scala.Int only |
Primitive.Long |
scala.Long only |
Primitive.Double |
scala.Double only |
Primitive.Float |
scala.Float only |
Primitive.String |
java.lang.String only |
Primitive.BigDecimal |
scala.BigDecimal only |
Primitive.BigInt |
scala.BigInt only |
Primitive.Unit |
scala.Unit only |
Primitive.Byte |
scala.Byte only |
Primitive.Short |
scala.Short only |
Primitive.Char |
scala.Char only |
Primitive.UUID |
java.util.UUID only |
Primitive.Currency |
java.util.Currency only |
Primitive.Instant / LocalDate / LocalDateTime / … |
Each specific java.time.* type |
Record[A] |
A case class / product type whose every field satisfies A. Vacuously true for zero-field records. Sealed traits and enums are automatically unwrapped: each case is checked individually, so no Variant node is needed. |
Sequence[A] |
Any collection (List, Vector, Set, Array, Chunk, …) whose element type satisfies A |
Map[K, V] |
Map, HashMap, … whose key satisfies K and value satisfies V |
Optional[A] |
Option[X] where the inner type X satisfies A |
Wrapped[A] |
A ZIO Prelude Newtype/Subtype wrapper whose underlying type satisfies A |
Self |
Recursive self-reference back to the entire enclosing Allows[A, S] grammar |
Dynamic |
DynamicValue — the schema-less escape hatch |
IsType[A] |
Exact nominal type match: satisfied only when the checked type is exactly A (=:=) |
`|` |
Union of two grammar nodes: A | B. In Scala 2 write A `|` B in infix position. |
Every specific Primitive.Xxx node also satisfies the top-level Primitive node (which matches any of the 30 primitive types). This means a type annotated with Primitive.Int is valid wherever Primitive or Primitive | Primitive.Long is required.
Allows[A, S] is a proof token, not an ordinary value. It carries zero public methods that you call directly. Instead, you use it in three ways:
- As a constraint in function signatures — Declare
Allows[A, S]as an implicit/using parameter to require that callers pass only types satisfying the grammar. - To summon evidence — Use
implicitly[Allows[A, S]](Scala 2) orsummon[Allows[A, S]](Scala 3) at a call site to check the constraint and get an error message if it fails. - In type aliases — Define type aliases like
type FlatRecord = Allows[_, Record[Primitive | Optional[Primitive]]]to name constraints and reuse them across functions.
The macro that powers Allows checks the constraint at compile time and emits nothing but a reference to a single private singleton at runtime, so there is zero per-call-site overhead.
The Primitive parent class is the catch-all: it accepts any of the 30 Schema 2 primitive types. For stricter control — such as when the target serialisation format only supports a subset — use the specific subtype nodes in Allows.Primitive:
import zio.blocks.schema.comptime.Allows
import Allows._
// Only JSON-representable scalars (no UUID, Char, java.time.*)
type JsonPrimitive =
Primitive.Boolean | Primitive.Int | Primitive.Long |
Primitive.Double | Primitive.String | Primitive.BigDecimal |
Primitive.BigInt | Primitive.Unit
def toJson[A](doc: A)(using Allows[A, Record[JsonPrimitive | Self]]): String = ???
// Only numeric types
type Numeric = Primitive.Int | Primitive.Long | Primitive.Double | Primitive.Float |
Primitive.BigInt | Primitive.BigDecimal
def aggregate[A](data: A)(using Allows[A, Record[Numeric]]): Double = ???A type annotated with Primitive.Int satisfies Primitive (the catch-all) because Primitive.Int extends Primitive:
import zio.blocks.schema.comptime.Allows
import Allows._
val ev: Allows[Int, Primitive] = implicitly // Primitive (catch-all) — ✓
val sp: Allows[Int, Primitive.Int] = implicitly // Primitive.Int (specific) — ✓JSON's primitive value set is null | boolean | number | string. Types such as UUID, Char, and all java.time.* types have no native JSON representation and must be encoded as strings at the application layer. Using JsonPrimitive instead of the catch-all Primitive enforces this at compile time.
A JSON document grammar is straightforward: a JSON value is either a record (JSON object) or a sequence (JSON array), and Self handles all nesting:
import zio.blocks.schema.comptime.Allows
import Allows._
type JsonPrimitive =
Primitive.Boolean | Primitive.Int | Primitive.Long | Primitive.Double |
Primitive.String | Primitive.BigDecimal | Primitive.BigInt | Primitive.Unit
type Json = Record[JsonPrimitive | Self] | Sequence[JsonPrimitive | Self]
def toJson[A](doc: A)(using Allows[A, Json]): String = ???Self recurses back to Json at every nested position, so List[String] satisfies Sequence[JsonPrimitive | Self] (String is JsonPrimitive), List[Author] satisfies it too (Author satisfies Record[JsonPrimitive | Self] via Self), and top-level arrays work directly.
A type with a UUID or Instant field fails at compile time with this error:
[error] Schema shape violation at WithUUID.id: found Primitive(java.util.UUID),
required Primitive.Boolean | Primitive.Int | ... | Primitive.String | ...
UUID is not a JSON-native type — encode it as Primitive.String.
Union types express "or" in the grammar.
Uses the infix operator Primitive `|` Optional[Primitive] from Allows:
import zio.blocks.schema.comptime.Allows
import Allows._
def writeCsv[A](rows: Seq[A])(implicit
ev: Allows[A, Record[Primitive | Optional[Primitive]]]
): Unit = ???Uses native union type syntax:
import zio.blocks.schema.comptime.Allows
import Allows._
def writeCsv[A](rows: Seq[A])(using
Allows[A, Record[Primitive | Optional[Primitive]]]
): Unit = ???Both spellings compile and produce the same semantic behavior. The grammar is identical — the only difference is how the union type is expressed.
import zio.blocks.schema.Schema
import zio.blocks.schema.comptime.Allows
import Allows._
// Flat record: only primitives and optional primitives allowed
def writeCsv[A: Schema](rows: Seq[A])(using
Allows[A, Record[Primitive | Optional[Primitive]]]
): Unit = ???
// RDBMS INSERT: primitives, optional primitives, or string-keyed maps (JSONB)
def insert[A: Schema](value: A)(using
Allows[A, Record[Primitive | Optional[Primitive] | Allows.Map[Primitive, Primitive]]]
): String = ???If a user passes a type with nested records, they get a precise compile-time error like this:
[error] Schema shape violation at UserWithAddress.address: found Record(Address),
required Primitive | Optional[Primitive] | Map[Primitive, Primitive]
Published events are typically sealed traits of flat record cases. No Variant node is needed — sealed traits are automatically unwrapped:
import zio.blocks.schema.Schema
import zio.blocks.schema.comptime.Allows
import Allows._
// DomainEvent is a sealed trait; its cases must each satisfy Record[Primitive | Sequence[Primitive]]
def publish[A: Schema](event: A)(using
Allows[A, Record[Primitive | Optional[Primitive] | Sequence[Primitive]]]
): Unit = ???If a case of the sealed trait has a nested record field, the error names that case and field like this:
[error] Schema shape violation at DomainEvent.OrderPlaced.items.<element>:
found Record(OrderItem), required Primitive | Optional[Primitive] | Sequence[Primitive]
A document store accepts arbitrarily nested records but not DynamicValue leaves. The Self node expresses the recursive grammar:
import zio.blocks.schema.Schema
import zio.blocks.schema.comptime.Allows
import Allows._
type JsonDocument =
Record[Primitive | Self | Optional[Primitive | Self] | Sequence[Primitive | Self] | Allows.Map[Primitive, Primitive | Self]]
def toJson[A: Schema](doc: A)(using Allows[A, JsonDocument]): String = ???This grammar allows:
case class Author(name: String, email: String)— Record[Primitive] ✓case class Book(title: String, author: Author, tags: List[String])— Record with Self-nested record and Sequence[Primitive] ✓case class Category(name: String, subcategories: List[Category])— recursive ✓
But rejects:
case class Bad(name: String, payload: DynamicValue)— DynamicValue is not in the grammar ✗
import zio.blocks.schema.Schema
import zio.blocks.schema.comptime.Allows
import Allows._
def graphqlType[A: Schema]()(using
Allows[A, Record[Primitive | Optional[Self] | Sequence[Self]]]
): String = ???
// Works:
case class TreeNode(value: Int, children: List[TreeNode])
object TreeNode { implicit val schema: Schema[TreeNode] = Schema.derived }
// graphqlType[TreeNode]() — compiles fineThe Sequence[A] node accepts any collection type. When a DSL needs to restrict to a specific kind of collection — for example, a DynamoDB Set operation that is only valid on sets, not lists — use the Sequence subtypes:
import zio.blocks.schema.comptime.Allows
import Allows._
// Only an immutable List is accepted
val listOnly: Allows[List[Int], Sequence.List[Primitive]] = implicitly
// Only an immutable Set is accepted
val setOnly: Allows[Set[Int], Sequence.Set[Primitive]] = implicitly
// Only a Vector
val vecOnly: Allows[Vector[String], Sequence.Vector[Primitive]] = implicitly
// Only an Array
val arrOnly: Allows[Array[Int], Sequence.Array[Primitive]] = implicitly
// Only a Chunk
import zio.blocks.chunk.Chunk
val chkOnly: Allows[Chunk[String], Sequence.Chunk[Primitive]] = implicitlyEach subtype extends Sequence[A], so a grammar written with the parent Sequence still accepts all collection kinds. A grammar written with a subtype rejects other kinds at compile time:
// Set[Int] does NOT satisfy Sequence.List[Primitive] — compile error:
[error] Shape violation at Set: found Sequence[scala.Int], required Sequence.List[...]
val bad: Allows[Set[Int], Sequence.List[Primitive]] = implicitly
A DynamoDB grammar can encode the distinction between set types and list types exactly, without any additional runtime proof. We use Sequence.Set to narrow the grammar to sets-only operations:
import zio.blocks.schema.comptime.Allows
import Allows._
type N = Primitive.Int | Primitive.Long | Primitive.Float | Primitive.Double | Primitive.Short
type S = Primitive.String
type NS = Sequence.Set[N | Wrapped[N]]
type SS = Sequence.Set[S | Wrapped[S]]
// addSet is only callable with a Set — List[Int] or Vector[Int] would fail at compile time
def addSet[A](set: scala.collection.immutable.Set[A])(implicit
ev: Allows[scala.collection.immutable.Set[A], NS | SS]
): String = "ok"IsType[A] is a nominal type predicate. It is satisfied only when the checked Scala type is exactly A (i.e. checked =:= A). It is most useful as an element constraint inside Sequence subtypes, where it aligns the element type of a collection with a polymorphic method type parameter.
The primary motivation (GitHub issue #1172) is DSL methods that must constrain both the container kind and the element type in a single Allows expression. Without IsType, a separate type class (like Containable) is needed to connect A in contains[A] to the element type of the collection. With IsType, the connection is expressed directly in the grammar.
To use IsType[A] with a polymorphic A, require IsNominalType[A] from zio-blocks-typeid at the call site. This ensures the macro always sees a concrete type when it evaluates IsType[A]:
import zio.blocks.schema.comptime.Allows
import Allows._
import zio.blocks.typeid.IsNominalType
// `To` must be a Set whose element type is exactly `A`.
// IsNominalType[A] ensures A is concrete at the call site — an unresolved
// type parameter would fail to produce IsNominalType and the call site
// would not compile.
def contains[To, From, A: IsNominalType](a: A)(implicit
ev: Allows[To, Sequence.Set[IsType[A]]]
): Boolean = ev.ne(null)
// Compiles: Set[Int] satisfies Sequence.Set[IsType[Int]]
val r1: Boolean = contains[Set[Int], Nothing, Int](42)
// Compiles: Set[String] satisfies Sequence.Set[IsType[String]]
val r2: Boolean = contains[Set[String], Nothing, String]("hello")A mismatch between the element type and A is a compile-time error:
[error] Shape violation at ...<element>: found Primitive(java.lang.String),
required IsType[Int]
IsType[A] can also appear as a standalone constraint, or anywhere a grammar node is accepted:
import zio.blocks.schema.comptime.Allows
// Int satisfies IsType[Int] exactly
val ev: Allows[Int, Allows.IsType[Int]] = implicitly
// List[String] satisfies Sequence[IsType[String]]
val ev2: Allows[List[String], Allows.Sequence[Allows.IsType[String]]] = implicitlySelf refers back to the entire enclosing Allows[A, S] grammar. It allows the grammar to describe recursive data structures.
Non-recursive types satisfy Self-containing grammars without issue: if no field ever recurses back to the root type, the Self position is never reached, and the constraint is vacuously satisfied.
Mutual recursion between two or more distinct types is a compile-time error reported as:
[error] Mutually recursive types are not supported by Allows.
Cycle: Forest -> Tree -> Forest
The Wrapped[A] node matches ZIO Prelude Newtype and Subtype wrappers. The underlying type must satisfy A. Here's an example:
import zio.prelude.Newtype
import zio.blocks.schema.Schema
import zio.blocks.schema.comptime.Allows
import Allows._
// ZIO Prelude Newtype pattern:
object ProductCode extends Newtype[String]
type ProductCode = ProductCode.Type
given Schema[ProductCode] =
Schema[String].transform(_.asInstanceOf[ProductCode], _.asInstanceOf[String])
// ProductCode satisfies Wrapped[Primitive] — its underlying String is Primitive
val ev: Allows[ProductCode, Wrapped[Primitive]] = implicitlyScala 3 opaque types are resolved to their underlying type by the macro (they are transparent), so an opaque alias like this satisfies Primitive directly:
opaque type UserId = java.util.UUID
// UserId satisfies Allows[UserId, Primitive] — resolved to UUID (a primitive)Sealed traits and enums are automatically unwrapped by the macro. Whenever a sealed type is encountered at any grammar check position, the macro recursively checks every case against the same grammar. This makes a Variant grammar node unnecessary.
import zio.blocks.schema.comptime.Allows
import Allows._
sealed trait Shape
case class Circle(radius: Double) extends Shape
case class Rectangle(width: Double, height: Double) extends Shape
case object Point extends Shape
// No Variant node — Shape is auto-unwrapped, all cases checked against Record[Primitive]
val ev: Allows[Shape, Record[Primitive]] = implicitlyAuto-unwrap is recursive: if a case is itself a sealed trait, its cases are unwrapped too, to any depth.
Union branches (A | B) work naturally with auto-unwrap: unused branches are fine under Allows upper-bound semantics.
When a type does not satisfy the grammar, the macro reports:
- The path to the violating field:
Order.items.<element> - What was found:
Record(OrderItem) - What was required:
Primitive | Sequence[Primitive] - A hint where applicable
Multiple violations are reported in a single compilation pass — the user sees all problems at once, for example:
[error] Schema shape violation at UserWithAddress.address: found Record(Address),
required Primitive | Optional[Primitive] | Map[Primitive, Primitive]
[error] Hint: Type 'Address' does not match any allowed shape
Record[A] is vacuously true for case objects and zero-field records, since there are no fields to violate the constraint:
import zio.blocks.schema.Schema
import zio.blocks.schema.comptime.Allows
import Allows._
case object EmptyEvent
implicit val schema: Schema[EmptyEvent.type] = Schema.derived
val ev: Allows[EmptyEvent.type, Record[Primitive]] = implicitly // vacuously trueAllows[A, S] carries zero runtime overhead. The macro emits a reference to a single private singleton Allows.instance cast to the required type. There is no per-call-site allocation.
| Feature | Scala 2 | Scala 3 |
|---|---|---|
| Union syntax | A `|` B infix |
A | B native union type |
| Summon syntax | implicitly[Allows[A, S]] |
summon[Allows[A, S]] or implicitly |
| Evidence parameter | (implicit ev: Allows[A, S]) |
(using Allows[A, S]) |
| Opaque type detection | ZIO Prelude only | Scala 3 opaque types + ZIO Prelude + neotype |
| Derivation keyword | Schema.derived implicit |
Schema.derived or derives Schema |
Both Scala versions produce the same macro behavior and the same error messages.
Allows and Schema are complementary but independent:
Schema[A]describes what anAlooks like at runtime — how to serialize, deserialize, introspect, or transform it. It requires explicit derivation and handles the full type signature.Allows[A, S]describes what anAmay look like at compile time — a structural grammar thatAmust satisfy. It requires no schema and uses only the Scala type system.
You can use Allows without Schema:
import zio.blocks.schema.comptime.Allows
import Allows._
// Pure shape constraint, no Schema required
def writeCsv[A](rows: Seq[A])(using Allows[A, Record[Primitive | Optional[Primitive]]]): Unit = ???Or combine them when runtime encoding and shape validation are both needed:
import zio.blocks.schema.Schema
import zio.blocks.schema.comptime.Allows
import Allows._
// Shape constraint + runtime encoding
def writeCsv[A: Schema](rows: Seq[A])(using
Allows[A, Record[Primitive | Optional[Primitive]]]
): Unit = ???When combined, Allows enforces the structural guarantee that Schema can use — for example, a CSV serializer can assume that every field is a primitive or optional primitive and skip defensive type checks.
See Schema for more on runtime encoding and decoding with schemas.
All code from this guide is available as runnable examples in the schema-examples module.
1. Clone the repository and navigate to the project:
git clone https://github.com/zio/zio-blocks.git
cd zio-blocks2. Run individual examples with sbt:
CSV serializer with flat record compile-time constraints (source)
sbt "schema-examples/runMain comptime.AllowsCsvExample"import docs.SourceFile
SourceFile.print("schema-examples/src/main/scala/comptime/AllowsCsvExample.scala")Event bus with sealed trait auto-unwrap and nested hierarchies (source)
sbt "schema-examples/runMain comptime.AllowsEventBusExample"import docs.SourceFile
SourceFile.print("schema-examples/src/main/scala/comptime/AllowsEventBusExample.scala")GraphQL / tree structures using Self for recursive grammars (source)
sbt "schema-examples/runMain comptime.AllowsGraphQLTreeExample"import docs.SourceFile
SourceFile.print("schema-examples/src/main/scala/comptime/AllowsGraphQLTreeExample.scala")Sealed trait auto-unwrap with nested hierarchies and case objects (source)
sbt "schema-examples/runMain comptime.AllowsSealedTraitExample"import docs.SourceFile
SourceFile.print("schema-examples/src/main/scala/comptime/AllowsSealedTraitExample.scala")RDBMS library with CREATE TABLE and INSERT using flat record constraints (compile-only) (source)
Demonstrates how Allows constraints are verified at compile time — the code below shows valid examples that compile successfully, and includes comments showing which patterns would be rejected:
import docs.SourceFile
SourceFile.print("schema-examples/src/main/scala/comptime/RdbmsExample.scala")JSON document store with specific primitives and recursive Self grammar (compile-only) (source)
Demonstrates how Allows enforces recursive schema constraints at compile time:
import docs.SourceFile
SourceFile.print("schema-examples/src/main/scala/comptime/DocumentStoreExample.scala")