Using Scala path-dependent types for awesome
Scala’s path-dependent types allow objects to carry types along with them, and these types can be used in type-checking. This is useful in all sorts of cases; in particular, it lets you encode relationships where you get objects from another object (e.g. database cursors from a connection), and they objects can only be used with the particular object they came from. You can’t use a cursor with a different connection, even if the connection is an instance of the same class. Path-dependent types allow this requirement to be statically type-checked.
I use these in some of my current research code. I have a spider that
makes web requests and saves the results. This spider needs to be able
to run against multiple backends, and each backend has a different set
of requests and data types. They all have Node
s, but the
requests for nodes differ from backend to backend. Abstracting the
spider into its own component allows me to quickly write new backends
just by specifying the requests with their post-processing/storage
capabilities.
The type of a request with its storage (called an
InfoNeed
) looks something like this:
Different backends can have wildly different request structures, so long as they can specify needs that have a request to fetch the data, a way to extract new neighbors, and a way to save the data appropriately in the data storage backend.
Now, a piece of the spider code (vastly simplified) looks about like this:
The variable res
is of type
Result[need.DataType]
; its type is dependent on a type
carried in the value of the parameter need
. As a
result, Scala statically type-checks that the need’s save
method receives the same type of data as it retrieves in with its
request, without the surrounding spider code having any knowledge of
what type that data might have (because the DataType
type
is fully abstract).
I can type-check safe flow of arbitrary data through the
process
function, knowing nothing about the data but only
that the type is the same and carried through properly. That is very,
very cool.
OCaml allows a similar thing with its first-class modules, but they tend to carry around some more syntactic baggage in my experience.