Custom prelude used in Serokell.
See README.md file for more details.
Universum
universum is a custom prelude used in @Serokell that has:
- Excellent documentation: tutorial, migration guide from
Prelude, Haddock with examples for (almost) every function, all examples are tested withdoctest, documentation regarding internal module structure. universum-specific HLint rules:.hlint.yaml- Focus on safety, convenience and efficiency.
What is this file about?
This README contains introduction to Universum and a tutorial on how to use it.
Structure of this tutorial
This tutorial has several parts:
- Philosophy and motivation.
- How to use
universum. - Changes in
Prelude(some gotchas). - Already known things that weren't in
Preludebrought into scope. - New things added.
- Migration guide from
Prelude.
This is neither a tutorial on Haskell nor tutorial on each function contained in Universum. For detailed documentation of every function together with examples and usage, see Haddock documentation.
Why another custom Prelude? ↑
Motivation
At Serokell, we strive to be as productive as possible. That's why we are using Haskell. This choice of language implies that we're restricted to use Prelude: implicit import of basic functions, type classes and data types. Unfortunately, the default Preludeis considered to be not so good due to some historical reasons.
This is why we decided to use a better tool. Luckily, Haskell provides us with the ability to replace default Prelude with an alternative. All we had to do is to implement a new basic set of defaults. There already were plenty of preludes, so we didn't plan to implement everything from scratch. After some long, hot discussions, our team decided to base our custom prelude on protolude.
The next section explains why we've made this choice and what we are willing to do. This tutorial doesn't cover the differences from protolude. Instead, it explains how Universum is different from regular Prelude.
Main goals
While creating and maintaining a custom prelude, we are pursuing the following goals:
- Avoid all partial functions. We like total and exception-free functions. You can still use some unsafe functions from
Universum.Unsafemodule, but they are not exported by default. - Use more efficient string representations.
Stringtype is crushingly inefficient. All our functions either try to be polymorphic over string type or useTextas the default string type. Because the community is evolving slowly, some libraries still useStringtype, soStringtype alias is still reexported. We recommend to avoidStringas much as you can! - Try to not reinvent the wheel. We're not trying to rebuild whole type hierarchy from scratch, as it's done in
classy-prelude. Instead, we reexport common and well-known things frombaseand some other libraries that are used in everyday production programming in Haskell.Note: well, we did end up inventing some new things.
- Export more useful and commonly used functions. Hello, my name is Dmitry. I was coding Haskell for 3 years but still hoogling which module
liftIOcomes from. Things likeliftIO,ReaderTtype,MVar-related functions have unambiguous names, are used in almost every non-trivial project, and it's really tedious to import them manually every time. - Make changes only when there are enough good reasons to make these changes. We have a code modification policy which semi-formally describes pre-conditions for different types of changes.
Unlike protolude, we are:
- Not trying to be as general as possible (thus we don't export much from
GHC.Generics). - Not trying to maintain every version of
ghccompiler (but at least the latest 3). - Trying to make writing production code easier (see enhancements and fixes).
How to use Universum ↑
Okay, enough philosophy. If you want to just start using universum and explore it with the help of compiler, set everything up according to the instructions below.
Disable the built-in prelude at the top of your file:
{-# LANGUAGE NoImplicitPrelude #-}
Or directly in your project .cabal file, if you want to use in every module by default:
default-extensions: NoImplicitPrelude
Then add the following import to your modules:
import Universum
If you're using Emacs and don't want to type import Universum manually every time, you can modify your configs a little bit.
If you want to get familiar with universum internal structure, you can just read top-level documentation for Universum module.
Gotchas ↑
head,tail,last,init,foldl1,minimumand other were-partial functions work withNonEmpty ainstead of[a].- Safe analogue for
head,foldl1,foldr1,minimum,maximumfunctions, for instance:safeHead :: [a] -> Maybe a. undefinedtriggers a compiler warning, which is probably not what you want. Either usethrowIO,Except,errororbug.mapisfmapnow.- Multiple sorting functions are available without imports:
sortBy :: (a -> a -> Ordering) -> [a] -> [a]: sorts list using given custom comparator.sortWith :: Ord b => (a -> b) -> [a] -> [a]: sorts a list based on some property of its elements.sortOn :: Ord b => (a -> b) -> [a] -> [a]: just likesortWith, but more time-efficient if function is calculated slowly (though less space-efficient). So you should writesortOn length(would sort elements by length) butsortWith fst(would sort list of pairs by first element).
- Functions
sumandproductare strict now, which makes them more efficient. - If you try to do something like
putStrLn "hi", you'll get an error message ifOverloadedStringsis enabled – it happens because the compiler doesn't know what type to infer for the string. UseputTextLnin this case. - Since
showdoesn't come fromShowanymore, you can't writeShowinstances easily. See migration guide for details. - You can't call some
Foldablemethods overMaybeand some other types.Foldablegeneralization is useful but potentially error-prone. Instead we created our own fully compatible withFoldableContainertype class but that restricts the usage of functions likelengthoverMaybe,Either,Identityand tuples. We're also using GHC 8 feature of custom compile-time errors to produce more helpful messages. - As a consequence of previous point, some functions like
traverse_,forM_,sequenceA_, etc. are generalized overContainertype classes. errortakesText.- We are exporting a rewrite rule which replaces
toString . toText :: Text -> Textwithid. Note that this changes semantics in some corner cases.
Things that you were already using, but now you don't have to import them explicitly ↑
Commonly used libraries
First of all, we reexport some generally useful modules: Control.Applicative, Data.Traversable, Data.Monoid, Control.DeepSeq, Data.List, and lots of others. Just remove unneeded imports after importing Universum (GHC should tell you which ones).
Then, some commonly used types: Map/HashMap/IntMap, Set/HashSet/IntSet, Seq, Text and ByteString (as well as synonyms LText and LByteString for lazy versions).
liftIO and MonadIO are exported by default. A lot of IO functions are generalized to MonadIO.
deepseq is exported. For instance, if you want to force deep evaluation of some value (in IO), you can write evaluateNF a. WHNF evaluation is possible with evaluateWHNF a.
We also reexport big chunks of these libraries: mtl, stm, microlens, microlens-mtl.
Bifunctor type class with useful instances is exported.
firstandsecondfunctions apply a function to first/second part of a tuple (for tuples).bimaptakes two functions and applies them to first and second parts respectively.
Text
We export Text and LText, and some functions work with Text instead of String – specifically, IO functions (readFile, putStrLn, etc) and show. In fact, show is polymorphic and can produce strict or lazy Text, String, or ByteString. Also, toText/toLText/toString can convert Text|LText|String types to Text/LText/String. If you want to convert to and from ByteString use encodeUtf8/decodeUtf8 functions.
Debugging and undefineds
trace, traceM, traceShow, etc. are available by default. GHC will warn you if you accidentally leave them in code, however (same for undefined).
We also have data Undefined = Undefined (which, too, comes with warnings).
Exceptions
We use safe-exceptions library for exceptions handling. Don't import Control.Exceptions module explicitly. Instead use functionality from safe-exceptions provided by universum or import Control.Exceptions.Safe module.
What's new? ↑
Finally, we can move to part describing the new cool features we bring with universum.
unconssplits a list at the first element.ordNubandsortNubare O(n log n) versions ofnub(which is quadratic) andhashNubandunstableNubare almost O(n) versions ofnub.(&)– reverse application.x & f & ginstead ofg $ f $ xis useful sometimes.whenM,unlessM,ifM,guardMare available and do what you expect them to do (e.g.whenM (doesFileExist "foo")).Very generalized version of
concatMapM, too, is available and does what expected.readMaybeandreadEitherare likereadbut total and give eitherMaybeorEitherwith parse error.when(Just|Nothing|Left|Right|NotEmpty)[M][_]let you conditionally execute something. Before:case mbX of Nothing -> return () Just x -> ... x ...After:
whenJust mbX $ \x -> ... x ...for_for loops. There's alsoforM_butfor_looks a bit nicer.for_ [1..10] $ \i -> do ...andM,allM,anyM,orMare monadic version of corresponding functions frombase.Type operator
$for writing types likeMaybe $ Either String $ Maybe Int.Eachtype family. So this:f :: Each [Show, Read] [a, b] => a -> b -> Stringtranslates into this:
f :: (Show a, Show b, Read a, Read b) => a -> b -> StringWithtype operator. So this:a :: With [Show, Read] a => a -> atranslates into this:
a :: (Show a, Read a) => a -> aVariadic composition operator
(...). So you can write:ghci> (show ... (+)) 1 2 "3" ghci> show ... 5 "5" ghci> (null ... zip5) [1] [2] [3] [] [5] True ghci> let process = map (+3) ... filter ghci> process even [1..5] [5,7]Conversions between
EitherandMaybelikerightToMaybeandmaybeToLeftwith clear semantic.using(Reader|State)[T]functions as aliases forflip run(Reader|State)[T].Onetype class for creating singleton containers. Even monomorhpic ones likeText.evaluateWHNFandevaluateNFfunctions as clearer and lifted aliases forevaluateandevaluate . force.ToPairstype class for data types that can be converted to list of pairs (likeMaporHashMaporIntMap).
Migration guide from Prelude ↑
In order to replace default Prelude with universum you should start with instructions given in how to use universum section.
This section describes what you need to change to make your code compile with universum.
Enable
-XOverloadedStringsand-XTypeFamiliesextension by default for your project.Since
head,tail,minimumand some other functions work forNonEmptyyou should refactor your code in one of the multiple ways described below:- Change
[a]toNonEmpty awhere it makes sense. - Use functions which return
Maybe. They can be implemented usingnonEmptyfunction. Likehead <$> nonEmpty l.head <$> nonEmpty lissafeHead ltailisdrop 1. It's almost never a good idea to usetailfromPrelude.
- Add
import qualified Universum.Unsafe as Unsafeand replace function with qualified usage.
- Change
If you use
fromJustor!!you should use them fromimport qualified Universum.Unsafe as Unsafe.Derive or implement
Containerinstances for your data types which implementFoldableinstances. This can be done in a single line becauseContainertype class automatically derives fromFoldable.Containertype class fromuniversumreplacesFoldableand doesn't have instances forMaybe a,(a, b),Identity aandEither a b. If you usefoldrorforM_or similar for something likeMaybe ayou should replace usages of such function with monomorhpic alternatives:Maybe(?:) :: Maybe a -> a -> afromMaybe :: a -> Maybe a -> amaybeToList :: Maybe a -> [a]maybeToMonoid :: Monoid m => Maybe m -> mmaybe :: b -> (a -> b) -> Maybe a -> bwhenJust :: Applicative f => Maybe a -> (a -> f ()) -> f ()whenJustM :: Monad m => m (Maybe a) -> (a -> m ()) -> m ()
EitherfromLeft :: a -> Either a b -> afromRight :: b -> Either a b -> beither :: (a -> c) -> (b -> c) -> Either a b -> cwhenRight :: Applicative f => Either l r -> (r -> f ()) -> f ()whenRightM :: Monad m => m (Either l r) -> (r -> m ()) -> m ()
If you have types like
foo :: Foldable f => f a -> a -> ayou should chose one of the following:Right: Modify types forContainerlikefoo :: (Container t, Element t ~ a) => t -> a -> a.Left: ImportData.Foldablemodulequalifiedand use everythingFoldable-related qualified.
Forget about
Stringtype.- Replace
putStrandputStrLnwithputTextandputTextLn. - Replace
(++)with(<>)forString-like types. - Try to use
fmtlibrary if you need to construct messages. - Use
toText/toLText/toStringfunctions to convert toText/LazyText/Stringtypes. - Use
encodeUtf8/decodeUtf8to convert to/fromByteString.
- Replace
Run
hlintusing.hlint.yamlfile fromuniversumpackage to cleanup code and imports.Since vanilla
showfrom theShowclass is not available, your customShowinstances will fail to compile. You canimport qualified Text.Showto bring vanillashowto scope with qualified name. It will not conflict withshowfromuniversumand yourShowinstances will compile successfully.