Skip to content

quanterall/HaskellMaterials

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

HaskellMaterials

Work In Progress

This repository is (and likely will be for quite some time) a work in progress. Suggestions for articles on concepts and themes are welcome, as well as corrections/clarifications on already available material.

Chapters

The below documents contain information and examples about different topics. Generally speaking they are readable from top to bottom in terms of the assumptions they make about knowledge level.

Real World Haskell

The purpose of these documents is to at some point be able to teach Haskell, but as a supplement to them it can be useful to look at the book Real World Haskell as it is freely available online. The need for this may lessen in time but since learning is such a difficult enterprise it's useful to simply look at many resources and absorb as much as possible.

Real World Haskell is not super up-to-date but it teaches the basics of Haskell reasonably and also has in mind that Haskell is a language for writing actual solutions in, not code for its own sake. Additionally, it does not teach relatively modern Haskell application architecture, but the resources for this have been included in these documents and will be taught separately.

Learning Haskell

Learning Haskell is hard, much like learning any programming language. I've found that many people seem to make it an unnecessarily hard one because they imagine there is more theoretical background that you have to learn in order to use Haskell. This is for the most part a fiction that people convince themselves of. Haskell is first and foremost a practical language meant to solve problems and there is very little that you can get out of theoretical (math or programming language theory) knowledge when it comes to getting things done in it.

There are some things I would like to stress to everyone who wants to learn Haskell:

Resist the urge to try to learn exactly how everything works immediately

I've seen a lot of people try to learn too much of how Haskell works immediately and I can pretty much guarantee that they never treated learning any other language like this. With the exception of the surface syntax of a language like C, it's near impossible. Getting comfortable with a language takes time and effort and Haskell is no exception.

Sometimes you have to use features that you aren't intimately familiar with in order to do things. Once you practice using them enough your intuition will catch up.

Don't read Monad tutorials, use things that implement the Monad type class and understand how it's used in practice, that'll be more useful than wasting your time reading theoretical (and usually misleading/wrong) explanations of what monads are.

Keep in mind that learning Haskell is about getting things done

Learning Haskell is not a badge of honor. Learning more advanced things is only useful if it helps you get things done or helps you understand more. There is lots of machinery that's created in the Haskell world that is basically not applicable to almost any application. Learning is fun, but you know what Haskell has too little of? People showing cool projects they created that actually do things.

It would be a failure for this material to have you walking way saying things like:

Haskell made me think differently. I don't use it often but it shaped the way I think.

In short, this repository is meant to serve not only as a jumping off point for exploring Haskell, but also for using it. With that in mind it's important to stop at key points in your Haskell journey and apply the things you think you know in order to iron out the specifics of them, as well as get a sense of what you are able to create with them.

Category theory is a branch of mathematics

While it can be stimulating to learn it's not useful knowledge for getting things done in Haskell. It is my experience that people who don't use Haskell will often misrepresent the usefulness of theory. This applies to more things than category theory.

If you feel compelled to learn category theory, keep in mind that it is unlikely to further your knowledge of Haskell in any meaningful way and that it is more likely to serve as a potential motivator for learning about programming language theory.

If you actually want to learn Haskell, couple pointed pieces of theory with practical exercise and then write programs that actually do things. You'll be infinitely more productive and better at Haskell and programming in general than someone who spent their time on things like category theory.

Asking questions is good

We learn when we bump up against things. We might not understand the behavior of some code or we might not be able to express ourselves clearly yet. We might just wonder if maybe there's a better way to do the thing we're doing because it feels awkward. Talking to other people about our code and solutions is a very effective way to stimulate thinking deeply about these things.

If you have any questions/suggestions on the material or things not covered by the material (yet?) you can pick any way to contact me (RocketChat, Discord, [email protected]) and I'll do my very best to answer, correct or otherwise handle it. Another avenue would be to create an issue or submit a pull request with the changes and/or questions.

Leave most of what you've heard at the door

It's not uncommon for people to have opinions on Haskell. It's only slightly less common for people to share those opinions on the Internet. What you'll come to realize after using Haskell is that most of those opinions (good and bad) don't necessarily match real life. Haskell is more talked about than it is used and with that comes a certain "lore".

Since it's very common to talk about Haskell without actually knowing it people feel free to, for lack of a better word, make up attributes and characteristics. Unless someone has provably used Haskell for creating applications to solve problems, there is no reason to trust anything they say about the language (This actually applies to all languages and programming in general, but it's worth noting here specifically).

Installing the build- and project management-tool stack

How to install Stack is a reasonable place to start.

Why stack? It's likely the easiest way for new people to start and (probably) the most common choice in industry, though other tools are catching up.

Running the examples in this repository

Most of the examples are created to work well with the quanterall/basic template referenced below in this document. You can choose to run this in a development container or not, but the bigger point is that this template sets up a default set of extensions. This means that it effectively assumes/establishes a standard language subset to learn and use.

If you're having issues running the examples, it might be wise to create a project using stack new my-project-name quanterall/basic and paste the example code in the generated Library.hs file. This has been tested with many of the examples and that is in general how it's supposed to work. If you find examples that do not, please make an issue or pull request about it.

Making your first example project

I've set up templates for project generation via the stack tool. These come pre-configured with appropriate settings (including which language features to use by default) and are very useful for setting up projects, both experimental one-offs, serious libraries and applications.

The most basic of these is called basic:

stack new my-project-name quanterall/basic
cd my-project-name
stack run

The above command will result in a running example that prints to the screen and in the Library.hs file you can edit the behavior of the program (in the main function).

This repository has more templates and a description of their use cases.

Note: See below for a better, more streamlined workflow enabled by docker, docker-compose and VSCode.

Using development containers in VSCode

I have created a template in the Quanterall organization called "basic" that will automatically create an empty project that also includes a development container specification that one can use to create simple Haskell applications.

To use it, issue the following command after having installed both docker and stack:

$ stack new project-name quanterall/basic
<... project creation text ...>

When you enter the directory created by this template, you'll notice that VSCode suggests re-opening the directory in a container. When you do so, it will download the basic Haskell quanterall development image (if you don't already have it) and create a container using it. This image has the tools necessary for you to be able to have proper assistance in VSCode (auto-complete, linting, type hints for the thing your cursor is on, error reporting and more) and the development container specifies a set of extensions that work well with those tools.

Once this is done and you open the terminal execute the following to run the app (as per usual):

$ stack run
<... build output ...>
Hello, World!

I have created a video demostrating how to use the templates:

Watch

Using the REPL/GHCi

It is quite common to use the REPL/interactive environment to write snippets of code and also to query your code base for information. I've made a video on some of the basic things you can do in the REPL, some of the things you might want to keep in mind as well as some of the things I added to make the experience a bit better in the development container REPLs:

Watch

"Help! I want to print a value but I can't print without IO!"

Debug.Trace is your friend. It will even put compiler warnings in there automatically so you don't leave your debug logging statements in there to run in production.

import qualified Debug.Trace as Trace

pureFunction :: Int -> Int
pureFunction parameter =
  -- This will show the first argument, i.e. turn it into a string and print it, then return the
  -- second argument; `parameter`
  Trace.traceShow (parameter + 42) parameter

main :: IO ()
main = do
  -- When this is run we will print `84`, but still get the passed in value bound to `x`.
  let x = pureFunction 42

  -- Here we will be printing 'Done: 42'
  putStrLn $ "Done: " <> show x

"I'm comfortable enough with Haskell basics but I don't know how to structure apps."

For readers who might be using these documents as references and feel comfortable enough with their basics, the following list of articles and suggestions might be useful:

(Note: Some of this will likely just be put in separate steps/lessons where we actually go through how they work, and we most definitely should/will do presentations on them.)

  • Learn about the Reader monad and that it allows one to read values in any code path extending from one with the type Reader dataType.
  • Learn how ReaderT dataType monad allows one to layer Reader on top of another monad, giving both capabilities; ReaderT dataType IO will therefore allow us to read the dataType value everywhere and use IO.
  • Read "The ReaderT design pattern" by Michael Snoyman. There is an overview of application "architecture" in there that is very useful, as well as motivations for choices made in the service of better Haskell applications. It's a very insightful article and the ideas in it should serve basically 90% of all production Haskell applications, easily.

In the article, these points are being stressed in different parts and I want to reiterate them and point them out as things to do and experiment with:

  • Experiment with programs keeping state in Env and executing in ReaderT Env IO, use both IO.IORef.IORef and Control.Concurrent.STM.TVar, recognize how STM is for transactional memory modifications and should be used when memory could be modified at the same time in different places.
  • Experiment with (MonadReader env m, HasSpecificThing env) => ... ReaderT env m to limit what a function can access from your environment, making it impossible that it touches things it shouldn't.
  • Experiment with adding monad constraints like CanModifyUsers m => User -> m () to your functions in order to limit them from being able to do anything else effectful; giving you ultimate control of exactly what effectful thing a function can do.

The steps above can be found in our materials and be explored in the following order:

Workflow screencasts

I've created a playlist with screen casts where I demonstrate a range of tasks in Haskell with a fairly standard workflow, as well as emphasize different parts of them.

Acknowledgements

As is usually the case with these things, people have helped make things possible and with improving things:

Special thanks to Daniel Valchev, Alex Radikov & Alex Yordanov for bravely jumping into early Haskell learning sessions and providing feedback on how they could be improved.

For testing out the materials, learning Haskell and going through exercises and pointing out how they could be improved:

  • Ivan Volkov
  • Ivana Andersson
  • Bogdan Bogdanov
  • Pavlin Nedelchev
  • Georgi Spasov
  • Maxime Kiriakov
  • Zahari Hristov
  • Mihail Dobrev

Radiana Koleva - For giving general advice on structure.

Anton Andonov & Quanterall - For making this repository possible.

Michael Snoyman & FP Complete - For their tireless work in producing material and libraries for Haskell in order to make it more useful and accessible. Several of their blog posts, workshop snippets and so on are included in these repositories.

About

Materials for learning Haskell

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages