• Stars
    star
    213
  • Rank 185,410 (Top 4 %)
  • Language
    Rust
  • License
    Mozilla Public Li...
  • Created over 5 years ago
  • Updated over 1 year ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

Your favourite Haskell type classes for Rust

higher

The functor hierarchy and other terrible ideas for Rust.

Yes, this gives you generalisable monads in Rust. No, they're not very nice compared to Haskell, because Rust's functions aren't quite as first class from the type system's perspective as you might like them to be, type constraints in trait implementations can be a serious headache when you want to implement, say, Functor for HashSet, and the type system can be particularly obtuse at times and need a lot of additional and extremely verbose guidance to get the type inference right, but they exist now.

What you get from this:

  • A set of fine grained traits (Functor, Pure, Apply, Bind, Applicative and Monad) for functors, applicatives and monads, inspired by PureScript and Scala's Cats.
  • Bifunctors, contravariant functors and profunctors, for completeness.
  • The run! macro for Haskell style do notation. I'd have preferred to call it do! or for! but unfortunately those are reserved keywords, even for macros.
  • Derive macros for Functor and Bifunctor.
  • Semigroups and monoids, because Rust's Add isn't quite a semigroup so Add + Default isn't quite a monoid.
  • Effect monads that wrap standard Futures and IO monads that wrap futures that can fail.
  • Most of Foldable, with the ambition of some of Traversable to follow. (It's always traverse.)
  • Rings and algebras, just in case.
  • Not necessarily a lot of good documentation, but like any good Haskell programmer you should be able to immediately infer every function's purpose from its type signature.

What are your intentions with this?

I wrote this for two reasons: first, to see if it was even possible, and second, as a shitpost with some extremely elaborate type signatures. If you think this is actually useful (and I'm mildly horrified to find that I'm starting to think it might be), you may wish to step up to help maintain it, because I doubt I'll keep giving it much attention once the novelty wears off.

Is Rust actually capable of this?

To everyone's surprise, with generic associated types we can now express a subset of what higher kinded types give us in Rust. This subset turns out to be sufficient to implement the abstractions found in the functor hierarchy, but the language still has its limitations which have a significant impact on the usefulness of these abstractions.

No constraint kinds

The first is the absence of constraint kinds. In Rust, when implementing a trait, we're unable to require stricter constraint bounds in our implementations than the trait itself specifies. While this has implications outside of GATs too, we run into it very quickly when trying to implement the most basic part of the functor hierarchy: Functor itself.

We can express Functor as simply as possible like this:

trait Functor<A> {
    type Target<T>;
    fn fmap<B, F: Fn(A) -> B>(self, f: F) -> Self::Target<B>;
}

We can also implement it for the most basic types, such as Vec:

impl<A> Functor<A> for Vec<A> {
    type Target<T> = Vec<T>;
    fn fmap<B, F: Fn(A) -> B>(self, f: F) -> Self::Target<B> {
        self.into_iter().map(f).collect()
    }
}

But we immediately run into a problem when trying to implement it for a type with constraints on its type arguments, like HashSet:

impl<A: Hash + Eq> Functor<A> for HashSet<A> {
    type Target<T> = HashSet<T>;
    fn fmap<B: Hash + Eq, F: Fn(A) -> B>(self, f: F) -> Self::Target<B> {
        self.into_iter().map(f).collect()
    }
}

Here, rustc will complain that the bounds on B are stricter than they are in the trait and refuse to proceed, but we do need those bounds to be able to implement fmap for HashSet. There's no trick available to us, even if we venture into nightly rustc's feature flags, to allow us to express this properly, so we're left entirely unable to implement Functor for HashSet, or for anything requiring bounds on A, as we have no way to express that they should carry over to the target type.

There's an open issue to address this situation which has been going since 2017 without as much as a concrete proposal, so I don't think one should allow oneself to feel too hopeful about a quick resolution to this problem.

GATs aren't higher kinded types and using them as such gets messy

We all knew GATs wouldn't be quite as flexible as higher kinded types, but in some specific ways they can be more flexible, and not necessarily in a good way for our purposes. Let's illustrate with an implementation of Bind:

trait Bind<A> {
    type Target<T>;
    fn then<B, F: Fn(A) -> Self::Target<B>>(self, f: F) -> Self::Target<B>;
}

If we'd like to write a function that composes two Binds into a third, we'd expect it to look like this:

fn compose_binds<A, B, C, M, F1, F2>(ma: M, f1: F1, f2: F2) -> M::Target<C>
where
    M: Bind<A>,
    F1: Fn(A) -> M::Target<B>,
    F2: Fn(B) -> M::Target<C>,
{
    ma.then(|a| f1(a).then(|b| f2(b)))
}

However, we have no guarantee that M::Target<B> actually resolves to anything that even implements Bind in turn, even though we mean the type Target<T> in the trait to always refer back to the original type constructor. This is just a convention on our part, a GAT in an of itself isn't required to follow any such convention. We need to add trait bounds to clarify this in both cases:

fn compose_binds<A, B, C, M, F1, F2>(ma: M, f1: F1, f2: F2) -> M::Target<C>
where
    M: Bind<A>,
    M::Target<B>: Bind<B>,
    M::Target<C>: Bind<C>,
    F1: Fn(A) -> M::Target<B>,
    F2: Fn(B) -> M::Target<C>,
{
    ma.then(|a| f1(a).then(|b| f2(b)))
}

This is fair, and expected. However, we're not done yet, and this is where it gets hairy. We also can't be sure that <M::Target<B> as Thenable<B>>::Target<C> will resolve to the same type as M::Target<C>, so we have to modify our constraints to clarify that this should also be the case:

fn compose_binds<A, B, C, M, F1, F2>(ma: M, f1: F1, f2: F2) -> M::Target<C>
where
    M: Bind<A>,
    M::Target<B>: Bind<B, Target<C> = M::Target<C>>,
    M::Target<C>: Bind<C>,
    F1: Fn(A) -> M::Target<B>,
    F2: Fn(B) -> M::Target<C>,
{
    ma.then(|a| f1(a).then(|b| f2(b)))
}

In the above, one might think we could clarify it by specifying <M::Target<B> as Thenable<B>>::Target<C> instead of M::Target<C> throughout, but this leads nowhere better, as it turns out.

Finally, the type checker at this point fails to infer that the result type of the second bind call should be C, so we need to explicitly provide it.

fn compose_binds<A, B, C, M, F1, F2>(ma: M, f1: F1, f2: F2) -> M::Target<C>
where
    M: Bind<A>,
    M::Target<B>: Bind<B, Target<C> = M::Target<C>>,
    M::Target<C>: Bind<C>,
    F1: Fn(A) -> M::Target<B>,
    F2: Fn(B) -> M::Target<C>,
{
    ma.then(|a| f1(a).then::<C, _>(|b| f2(b)))
}

This happens quite frequently, and is the reason the run! macro includes syntax for specifying the type of a binding. Correct me if I'm wrong, but this part I can't excuse the type checker for, it should have been able to figure this out for itself based on the function's return type.

In conclusion, GATs require considerably more effort to express everyday HTKs than you'd expect from a language with HKTs, even though you can express these things. This isn't a shortcoming of GATs (in fact, it's a feature that HKTs don't provide), but it's a real problem when you're trying to express abstractions like these which on the surface should have been much simpler. There's also the added problem that the more you compensate for this behaviour, the more your implementation details leak out into the type signature in ways they really shouldn't. This is especially unfortunate when you're dealing with traits, where you'll need to predict ahead of time all the ways in which your implementers will need this kind of extra support in the type signatures.

Conclusion

So, is Rust ready for the functor hierarchy? We're frustratingly close, but I think in the end the answer is no.

Basic abstractions like Functor would work well if we had constraint kinds, but as soon as you step outside the basic use casees, you begin to see where GATs aren't an adequate substitute for higher kinded types. That is to say, they solve a different problem from HKTs, and it's not straightforward to attempt to use them as a substitute. There are improvements which could be made to Rust's type checker which could alleviate some, but not all or even most of these issues.

This entire crate should therefore be considered as an exercise, and perhaps as a guide towards new language features, more than as a practical implementation. I do really see a genuine need for constraint kinds in Rust beyond this crate. I also suspect Rust would have been better off with actual higher kinded types, but that's a lost cause at this point.

Interestingly, though, while I was expecting Rust's borrow checker and the fact that it differentiates so clearly between owned values and mutable and immutable references to be a primary barrier to the implementation of this crate, this turned out to be more of an issue of API design than anything else. The primary problem here was the inability to refine Clone requirements where needed because of the absence of constraint kinds.

It's also unfortunate that I couldn't implement Apply for anything but boxed functions (as bare functions with the same type signature on the surface aren't consistently typed in Rust, nor can they be Sized), making it decidedly less than a zero cost abstraction, but I consider this a minor issue even if it should make one cautious when describing Rust as a functional programming language (which I don't think one should).

Licence

Copyright 2019 Bodil Stokke

This software is subject to the terms of the Mozilla Public License, v. 2.0. If a copy of the MPL was not distributed with this file, You can obtain one at http://mozilla.org/MPL/2.0/.

Code of Conduct

Please note that this project is released with a Contributor Code of Conduct. By participating in this project you agree to abide by its terms.

More Repositories

1

typed-html

Type checked JSX for Rust
Rust
1,845
star
2

im-rs

Assorted immutable collection datatypes for Rust
Rust
1,440
star
3

eslint-config-cleanjs

An eslint config which reduces JS to a pure functional language
JavaScript
1,106
star
4

vgtk

A declarative desktop UI framework for Rust built on GTK and Gtk-rs
Rust
1,037
star
5

smartstring

Compact inlined strings for Rust.
Rust
469
star
6

ohai-emacs

The finest hand crafted artisanal emacs.d for your editing pleasure
Emacs Lisp
378
star
7

catnip

A web based Clojure IDE
JavaScript
349
star
8

BODOL

The common BODil Oriented Language
Clojure
312
star
9

purescript-signal

Elm style FRP library for PureScript
PureScript
258
star
10

cljs-noderepl

A ClojureScript REPL running on Node.JS
Clojure
166
star
11

emacs.d

Here is my .emacs.d for public perusal.
Emacs Lisp
141
star
12

purescript-is-magic

An educational game with ponies
PureScript
114
star
13

purescript-smolder

A combinator library for generating markup
PureScript
88
star
14

meowhash-rs

Meow hasher for Rust
Rust
87
star
15

purescript-test-unit

An asynchronous unit test runner for PureScript
PureScript
86
star
16

vscode-file-browser

TypeScript
78
star
17

eulalie

ES6 flavoured parser combinators
JavaScript
69
star
18

microkanrens

Implementations of µKanren in assorted languages
PureScript
60
star
19

refpool

An efficient memory pool mechanism for Rust.
Rust
59
star
20

purescript-eulalie

String parser combinators for PureScript.
PureScript
55
star
21

pylon

A Javascript class system in 100% Clojurescript
Clojure
52
star
22

purescript-vdom

A native PureScript virtual DOM implementation.
PureScript
48
star
23

cljs-nashorn

ClojureScript REPL on Nashorn
Clojure
47
star
24

bitmaps

Bitmap types for Rust
Rust
44
star
25

boogaloo

JavaScript
43
star
26

purescript-typelevel

Type level natural numbers and booleans
PureScript
33
star
27

pink

A presentation tool.
JavaScript
31
star
28

clojurescript-all-the-way-down

Code from my Clojure/conj talk
Clojure
30
star
29

lolkanren

This project has moved:
JavaScript
30
star
30

purescript-observable

ES7 Observables for PureScript
PureScript
29
star
31

sized-chunks

Efficient sized chunk datatypes for immutable.rs
Rust
25
star
32

bodil.github.com

Presentation slides etc
CSS
24
star
33

vscode-use-package

Programmatic configuration for VS Code
TypeScript
24
star
34

sized-vec

Rust vectors with type level size
Rust
24
star
35

atom-use-package

Programmatic package configuration for Atom
JavaScript
24
star
36

palmtree

Don't look, I'm just playing with a B+-tree implementation which may or may not turn into a PALM tree.
Rust
24
star
37

spectre-of-free-software

HTML
23
star
38

vscode-init-script

Programmatic configuration for VS Code
TypeScript
23
star
39

vscode-blueprint

TypeScript
22
star
40

are-we-there-yet

Slides for CUFP 2017 keynote
HTML
20
star
41

simdify

SIMD optimised algorithms and data types
Rust
18
star
42

purescript-sized-vectors

Idris style sized vectors in PureScript
PureScript
18
star
43

nixcfg

My NixOS configuration files
Nix
17
star
44

typeify

A Browserify transform module for automated TypeScript compilation
JavaScript
17
star
45

purescript-kanren

Relational programming for PureScript
PureScript
17
star
46

purescript-store

A simple application state store for PureScript.
PureScript
16
star
47

only-better

This project has moved:
JavaScript
16
star
48

opt

Option types for TypeScript with real gradual typing
TypeScript
15
star
49

building-lisp

This project has moved:
Clojure
15
star
50

baconts

This project has moved:
TypeScript
14
star
51

graham

Parser combinators for JS
JavaScript
14
star
52

purescript-pux-smolder-dom

A drop-in replacement for Pux's React based renderer which needs no foreign dependencies.
PureScript
13
star
53

testlol

A Maven test runner for Javascript using Rhino and Env.js
JavaScript
13
star
54

error

A testing toolkit for ClojureScript, designed for testing asynchronous code
Clojure
11
star
55

ohai-atom

My Atom config.
CoffeeScript
11
star
56

pstalk

A slide deck.
JavaScript
11
star
57

gx

JavaScript
10
star
58

joy-of-milner

A talk about the Hindley-Milner type system.
JavaScript
9
star
59

lms

Lightweight Modular Staging in Rust
Rust
9
star
60

hipster

This project has moved:
JavaScript
9
star
61

purescript-chrome-api

PureScript bindings for the Chrome Platform APIs
PureScript
9
star
62

purescript-webapp

work in progress, don't
PureScript
8
star
63

persistence

A talk.
HTML
8
star
64

post-frp

JavaScript
8
star
65

leiningen-for-dummies

A Windows installer for Leiningen
Clojure
7
star
66

bodega-khajiit

May your road lead you to warm sands.
Rust
7
star
67

ONIForcedExit

Oxygen Not Included mod that restricts your game time.
C#
7
star
68

cargo-template-vgtk

Cargo template for vgtk apps.
Rust
6
star
69

newtype-proxy

Automatic deriving of `From` and `Deref` for Rust newtype structs.
Rust
6
star
70

purescript-smolder-dom

PureScript
6
star
71

hellobackbone

JavaScript
6
star
72

array-ops

Ready made default method implementations for array data types.
Rust
6
star
73

purescript-smolder-vdom

A Smolder renderer for purescript-vdom
PureScript
6
star
74

the-mess

This project has moved:
JavaScript
6
star
75

tweetlol

Twitter sidebar extension for Firefox
JavaScript
6
star
76

future-of-clojure

My talk for the Clojure Exchange 2014.
JavaScript
5
star
77

todo15m

A complete Node webapp you can write live on stage in 15 minutes if you come properly prepared.
JavaScript
5
star
78

mylittlecthulhu

A poorly disguised todo list, in Clojure and Clojurescript.
JavaScript
5
star
79

purescript-observable-channel

purescript-signal like channels for purescript-observable.
PureScript
5
star
80

purescript-signal-socket

Node sockets with a Signal interface
PureScript
5
star
81

vscode-prettier-toml

A VS Code formatter for TOML using Prettier
TypeScript
5
star
82

purescript-geom

PureScript 2D matrix transformations
PureScript
4
star
83

lolisp

It's a lisp, in Python.
Python
4
star
84

generators

generators are dashed jolly spiffy
JavaScript
4
star
85

node-gnomenotify

C++ bindings for GNOME libnotify on-screen notifications
C++
4
star
86

bagwell

A talk about Bagwell tries
HTML
4
star
87

ten-minute-clojure

Slides for teaching Clojure basics in a hurry with Catnip
JavaScript
4
star
88

sudokulol

An instructional Sudoku problem generator and solver, using Noir.
JavaScript
3
star
89

lolcode-mode

LOLCODE mode for Emacs.
Emacs Lisp
3
star
90

hands-on-with-clojure

This project has moved:
Clojure
3
star
91

startuplol

My attempt at solving the Extreme Startup exercise
Clojure
3
star
92

frob

This project has moved:
TypeScript
3
star
93

purescript-asap

The `asap` scheduler as a portable PureScript module
PureScript
2
star
94

unbreakwep

Chrome extension that unbreaks keyboard navigation in the weird time tracking app I have to use at work
JavaScript
2
star
95

mandelbrotlol

A very simple Mandelbrot set renderer using HTML5 canvas and web workers
JavaScript
2
star
96

unpredictable

More randomness for Clojure
Clojure
2
star
97

erlang-the-reckoning

HTML
2
star
98

feedlol

FriendFeed on your desktop!
Python
2
star
99

xbobar

This project has moved:
JavaScript
2
star
100

tickerlol

Stock ticker indicator applet for Unity
Clojure
2
star