The Hpack package format is based on YAML. Packages
are described in a file named package.yaml
.
Hpack is designed to remove redundancies from your package descriptions.
There is reference documentation below, but introductory documentation is still lacking. For the time being, take a look at the slides from my talk about Hpack at the Singapore Haskell meetup: http://typeful.net/talks/hpack
- Given this package.yaml running
hpack
will generate hpack.cabal - Given this package.yaml running
hpack
will generate getopt-generics.cabal - Given this package.yaml running
hpack
will generate sensei.cabal - Given this package.yaml running
hpack
will generate base-orphans.cabal
One easy way of getting started is to use
hpack-convert to convert an
existing cabal file into a package.yaml
.
Hpack | Cabal | Default | Notes | Example | Since |
---|---|---|---|---|---|
name |
· | ||||
version |
· | 0.0.0 |
|||
synopsis |
· | ||||
description |
· | ||||
category |
· | ||||
stability |
· | ||||
homepage |
· | If github given, <repo>#readme |
|||
bug-reports |
· | If github given, <repo>/issues |
|||
author |
· | May be a list | |||
maintainer |
· | May be a list | |||
copyright |
· | May be a list | |||
license |
· | ||||
license-file |
license-file or license-files |
LICENSE if file exists |
May be a list | ||
tested-with |
· | ||||
build-type |
· | Simple , or Custom if custom-setup section exists |
Must be Simple , Configure , Make , or Custom |
||
cabal-version |
>= 1.10 or >= 1.21 |
>= 1.21 if library component has reexported-modules field |
|||
extra-source-files |
· | Accepts glob patterns | |||
extra-doc-files |
· | Accepts glob patterns | 0.21.2 |
||
data-files |
· | Accepts glob patterns | |||
github |
source-repository head |
Accepts user/repo or user/repo/subdir |
github: foo/bar |
||
git |
source-repository head |
No effect if github given |
git: https://my.repo.com/foo |
||
custom-setup |
· | See Custom setup | |||
flags |
flag <name> |
Map from flag name to flag (see Flags) | |||
library |
· | See Library fields | |||
internal-libraries |
library <name> |
Map from internal library name to a dict of library fields and global top-level fields. | 0.21.0 |
||
executables |
executable <name> |
Map from executable name to executable (see Executable fields) | |||
executable |
executable <package-name> |
Shortcut for executables: { package-name: ... } |
0.18.0 |
||
tests |
test-suite <name> |
Map from test name to test (see Test fields) | |||
benchmarks |
benchmark <name> |
Map from benchmark name to benchmark (see Benchmark fields) |
Hpack | Cabal | Default | Notes | Example |
---|---|---|---|---|
dependencies |
setup-depends |
Implies build-type: Custom |
These fields are merged with all library, executable, test, and benchmark components.
Hpack | Cabal | Default | Notes |
---|---|---|---|
source-dirs |
hs-source-dirs |
||
default-extensions |
· | ||
other-extensions |
· | ||
ghc-options |
· | ||
ghc-prof-options |
· | ||
ghcjs-options |
· | ||
cpp-options |
· | ||
cc-options |
· | ||
c-sources |
· | Accepts glob patterns | |
js-sources |
· | Accepts glob patterns | |
extra-lib-dirs |
· | ||
extra-libraries |
· | ||
include-dirs |
· | ||
install-includes |
· | ||
frameworks |
· | ||
extra-frameworks-dirs |
· | ||
ld-options |
· | ||
buildable |
· | May be overridden by later stanza | |
dependencies |
build-depends |
||
build-tools |
· | ||
when |
Accepts a list of conditionals (see Conditionals) |
Hpack | Cabal | Default | Default inside conditional |
---|---|---|---|
exposed |
· | ||
exposed-modules |
· | All modules in source-dirs less other-modules less any modules mentioned in when |
|
other-modules |
· | All modules in source-dirs less exposed-modules less any modules mentioned in when |
Only if exposed-modules is not specified inside the conditional: All modules in source-dirs of the conditional less any modules mentioned in when of the conditional |
reexported-modules |
· | ||
default-language |
Haskell2010 |
Hpack | Cabal | Default | Notes |
---|---|---|---|
main |
main-is |
||
other-modules |
· | All modules in source-dirs less main less any modules mentioned in when |
|
default-language |
Haskell2010 |
Hpack | Cabal | Default | Notes |
---|---|---|---|
type |
exitcode-stdio-1.0 |
||
main |
main-is |
||
other-modules |
· | All modules in source-dirs less main less any modules mentioned in when |
|
default-language |
Haskell2010 |
Hpack | Cabal | Default | Notes |
---|---|---|---|
type |
exitcode-stdio-1.0 |
||
main |
main-is |
||
other-modules |
· | All modules in source-dirs less main less any modules mentioned in when |
|
default-language |
Haskell2010 |
Hpack | Cabal | Default | Notes |
---|---|---|---|
description |
· | Optional | |
manual |
· | Required (unlike Cabal) | |
default |
· | Required (unlike Cabal) |
Conditionals with no else branch:
- Must have a
condition
field - May have any number of other fields
For example,
when:
- condition: os(darwin)
extra-lib-dirs: lib/darwin
becomes
if os(darwin)
extra-lib-dirs:
lib/darwin
Conditionals with an else branch:
- Must have a
condition
field - Must have a
then
field, itself an object containing any number of other fields - Must have a
else
field, itself an object containing any number of other fields - All other top-level fields are ignored
For example,
when:
- condition: flag(fast)
then:
ghc-options: -O2
else:
ghc-options: -O0
becomes
if flag(fast)
ghc-options: -O2
else
ghc-options: -O0
At place where you can specify a list of files you can also use glob patterns. Glob patters and ordinary file names can be freely mixed, e.g.:
extra-source-files:
- static/*.js
- static/site.css
Glob patterns are expanded according to the following rules:
?
and*
are expanded according to POSIX (they match arbitrary characters, except for directory separators)**
is expanded in azsh
-like fashion (matching across directory separators)?
,*
and**
do not match a.
at the beginning of a file/directory
It is possible to use YAML anchors (&
), aliases
(*
) and merge keys (<<
) to define fields and reference them
later.
executables:
my-exe-1: &my-exe
main: my-exe-1.hs
dependencies: [base, my-lib]
ghc-options: [-threaded]
my-exe-2:
<<: *my-exe
main: my-exe-2.hs
Warnings for unknown fields will not be emitted for top-level fields starting with an underscore, so you can declare global aliases too:
_exe-ghc-options: &exe-ghc-options
- -threaded
- -rtsopts
executables:
my-exe-1:
ghc-options: *exe-ghc-options
It is also possible to use the !include
directive:
# ...
tests:
hlint: !include "../common/hlint.yaml"
hlint.yaml
:
source-dirs: test
main: hlint.hs
dependencies: [base, hlint]
This can also be used to provide entire libraries of snippets:
_common/lib: !include "../common/lib.yaml"
name: example1
version: '0.1.0.0'
synopsis: Example
<<: *legal
<<: *defaults
library:
source-dirs: src
tests:
hlint: *test_hlint
lib.yaml:
- &legal
maintainer: Some One <[email protected]>
copyright: (c) 2017 Some One
license: BSD3
- &defaults
dependencies:
- base
- containers
ghc-options:
- -Wall
- -Werror
- &test_hlint
source-dirs: test
main: hlint.hs
dependencies: [hlint]
To run hpack
automatically on modifications to package.yaml
add the
following to your ~/.vimrc
:
autocmd BufWritePost package.yaml silent !hpack --silent
Stack has built-in support for Hpack.
If you are using Stack you can use package.yaml
instead of a .cabal
file. No additional
steps are required.