Universum
:milky_way: Prelude written in @Serokell
Install / Use
/learn @serokell/UniversumREADME
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 Prelude
is 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.
Things like
liftIO,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
Related Skills
node-connect
349.7kDiagnose OpenClaw node connection and pairing failures for Android, iOS, and macOS companion apps
frontend-design
109.7kCreate distinctive, production-grade frontend interfaces with high design quality. Use this skill when the user asks to build web components, pages, or applications. Generates creative, polished code that avoids generic AI aesthetics.
openai-whisper-api
349.7kTranscribe audio via OpenAI Audio Transcriptions API (Whisper).
qqbot-media
349.7kQQBot 富媒体收发能力。使用 <qqmedia> 标签,系统根据文件扩展名自动识别类型(图片/语音/视频/文件)。
