Want to contribute? Here are some things we need. If you want to work in any of these, contact us for instructions!
We're self-funded. More funds = more devs = more cool features.
If you'd like to help with donations, grants or funding, obviously let us know (:
The best way to start contributing (and to get familiar with the codebase) is to
just add files to base
. Kind's
base is in a constant state
of evolution. It has several functions that aren't well documented. Some
functions may have inconsistent names here and there. Some obvious functions may
be missing. Many data structures are missing. Find anything you can improve,
work on it and submit a PR. We'll be very happy to review it!
As an example, all the proofs on Nat/Add were added by Eloi (thanks!). That kind of contribution is always welcome!
There are many missing numeric types on Kind/base
, such as I128
. The
existing types, such as U32
, may also have missing functions here and there.
Additions are welcome!
While Kind optimizes operations such as I32.add
to native operations in its
back-ends, these operations still need to be implemented in pure Kind, for
theorem proving purposes. Since implementing these operations for every numeric
type would be repetitive, most of these are implemented on the Word
type,
which represent N-bit values (for example, I32
is a thin wrapper around
Word<32>
, so I32.add
just calls Word.add<32>
). While many operations are
implemented, many are still missing. For example, all these operations are
TODOs:
Word.int.add
Word.int.mul
Word.int.sub
Word.int.mod
Word.int.div
Word.int.pow
Word.int.eql
Word.int.ltn
Word.int.lte
Word.int.eql
Word.int.gte
Word.int.gtn
Word.float.add
Word.float.mul
Word.float.sub
Word.float.mod
Word.float.div
Word.float.pow
Word.float.eql
Word.float.ltn
Word.float.lte
Word.float.eql
Word.float.gte
Word.float.gtn
Adding these would be great.
Are you a full-time JavaScript developer that doesn't like JavScript? You can
just use Kind as your main language, compile it to JS with kind Your.Term --js
and import it with require("Your.Term")
. Imagine being able to do that for
every language? Currently, Kind targets Scheme and JavaScript. We'd like more
backends, as many as possible. Adding a new back-end is somewhat simple: just
add its syntax on
base/Kind/Comp/Target!
We have some very primtive 3D vector operations, but not much else. For game development purposes, it would be amazing to have a rich library of geometric primitives, including matrices, quaternions, collisions, space partitioning structures and so on. Adding these is always welcome!
Right now, the DOM type allows rendering text, HTML nodes and pixelated canvas. It would be amazing to have a render mode that integrated with WebGL. If you'd like to work on that, contact us for more instructions!
Sounds silly, but just creating apps using the App
type would be amazing. Any
app added to base/App
will show up on http://old.kindelia.org/.
Sadly, we don't have a tutorial on how apps work, but it should be learnable
from looking the examples.
Right now, the JavaScript compiler on
JavaScript.kind
is lackluster, compared to the one in
FmcToJs.js. That
is why, when compiling to JS, instead of using the compiler written in Kind, we
compile to FormCore
, and then use FmcToJs.js
. Because of that, the Scheme
back-end will produce much worse JS code than the Node.js back-end, among other
issues. It would be nice to improve JavaScript.kind
to make it as efficient as
FmcToJs.js
, allowing us to get rid of the JavaScript dependency.
There are many missing syntaxes. For example, we don't have a syntax for quadruples, triples, only pairs. We also can't destruct triples, quadruples. We also can't destruct pairs in function arguments, in loops. For example:
let {x,y,z} = my_vector
List.map(({x,y}) x + y, list)
let sum = for {x,y} in positions: x + y + sum
None of the syntaxes above is available yet. Deep and nested pairs aren't available either. There are many syntaxes that aren't available or could be improved. Working on that with us is always welcome (but please, ask before!)
The get/set syntaxes can't be chained. For example,
let list = list[2] <- 100
sets the element of index 2 on list
to 100
. But
let list = list[2][2] <- 100
doesn't work as expected. It must be written as:
let list = list[2] <- (list[2][2] <- 100)
The same is the case for maps (map{"x"} <- 2
) and records (record@x <- 2
).
Adding these syntaxes would be nice.
Moreover, the following syntax would be nice to have:
let list[2] <- 100
This would be equivalent to:
let list = list[2] <- 100
Right now, when using maps, you need to explicitly convert your keys to strings. For example:
map{U256.show(1234)} <- "entry"
It would be nice if we either improved maps to have polymorphic keys, or
improved the parser to automatically add these conversions, in the same way that
operators (like +
) are polymoprhic.
The "forall" syntax requires a parenthesis sometimes. For example:
foo: Type
((A: Type) -> A)
This shouldn't be the case and needs investigation.
Right now, we can derive serializer, deserializer, stringifier, parser
for types.
For example:
type MyType {
foo
bar
} deriving (stringifier, parser)
Derives MyType.stringifier
, MyType.parser
. It would be nice to also allow
deriving other functions such as show, read, equal, larget_than, greater_than, serialize, deserialize
. Most of these are trivial. For example, show
is just
a wrapper that could use stringifier
, and serialize
is just a wrapped that
could use serializer
. Regardless, these are TODOs. Adding these would be
great.
One of the main sources of verbosity in Kind is the lack of implicit arguments.
That is partly improved by holes and !
. For example, Pair.new
can be written
as Pair.new<Nat,Nat>(1,2)
, or Pair.new<_,_>(1,2)
, or Pair.new!!(1,2)
. It
would be better to write just Pair.new(1,2)
. It is not clear how to add implicit
arguments to Kind without making some undesirable compromises, but it would be
a great improvement.
Past versions of Kind/Formality had an option to compile programs to optimal λ-calculus evaluators, which allowed us to explore these, using the language syntax. Sadly, this isn't available anymore. Re-adding would be amazing. In order to do that, the shortest path would be to port the code in this repository to Kind.
Implement a compiler from the low-order, linear λ-calculus to the EVM. Doing so is completely viable and will result in efficient smart-contracts. Once we have this, plus linear types, Kind will be able to be used as a smart-contract language. Contact us for more instructions.
Adding linear types would allow us to separate the linear from the non-linear subset of the language. That would bring several benefits.
-
Mutable structures. Right now,
base/Buffer8
is considered unsafe, because it is optimized to use mutable buffers under the hoods, on the JS back-end. That means that, if you use it non-linearly, your program may behave incorrectly. With linear types, we could apply the optimization only whenBuffer8
is linear. Similarly, we could optimize arrays and maps to use mutable datatypes when suitable. -
EVM compilation. While we have managed to reduce the cost of beta-reduction to miraculous 200 gas, even that is still too much for the very expensive environment of the Ethereum blockchain. Because of that, our best bet to compile to EVM, right now, is to compile the linear, low-order subset of Kind. That sounds lackluster, but it is actually pretty sufficient. That means we'd be able to write smart-contracts using Kind's syntax. As long as you don't do certain things (like using
List.map
or duplicating arguments), it will work fine, it will be inexpensive, and it will compile to efficient Ethereum contracts. But, for that, we need linearity. -
Compile to optimal λ-calculus evaluators. We have done a lot of experimentation with optimal λ-evaluators in the past, but isn't currently available. Adding linear types would allow us to compile it soundly to optimal evaluators in a sound manner.
-
Consistency/Termination checkers. Adding linear types will make the job of making a consistency checker easier. Check the section below.
Adding linearity checker to the compiler isn't a PhD-level task, but it requires some experience with functional programming, a lot of patience and knowledge about our type checker. Contact us if you are interested!
Compared to other proof languages, kind takes an inverted approach. Instead of
consistency being default and expressivity being opt-in (like Agda's
type-in-type
pragma), here, expressivity is default and consistency is a
planned opt-in. That means you're allowed to write programs with no
restrictions, just like most traditional languages like Haskell or JavaScript,
as long as they're total and well-typed. But that also means programs that do
not halt, and logical paradoxes, aren't prohibited.
Regardless, there are several terminating, consistent subsets of Kind, each
admiting different kinds of programs. For example, with structural recursion,
we're allowed to have Agda-like proofs, but no Type:Type
. Under elementary
affine logic, we're allowed to have Type:Type
, but not certain forms of nested
loops. Adding checkers for different consistent subsets would be a nice feature.
For the end user, this could be presented as an icon when type-checking. For example:
$ kind Nat.add
Nat.add: (n:Nat) (m:Nat) Nat ✓ ⊤
All terms check.
With ✓
standing for "well-typed" and ⊤
standing for "terminating".
While we have some interesting insights on the matter (check this blog
post),
Kind isn't capable of expressing the most important HoTT features. We could add
these inspired on Cubical Type Theory, but this would increase the size of
Kind's core by a few multipliers, which we don't want to. In special, the
transp
function seems to account for most of that complexity. Investigating
how to add HoTT features without blowing up the core size is an interesting line
of research.
I'm currently adding items as I remember, so this list isn't complete right now. If you have any improvement in mind, feel free to add here too!