Skip to content

Commit 5b13ff4

Browse files
author
Sasa Bogicevic
committed
Merge branch 'master' into 309-handle-application-exceptions-with-500-errors
2 parents 64686f3 + f536c90 commit 5b13ff4

File tree

89 files changed

+2415
-1506
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

89 files changed

+2415
-1506
lines changed

.travis.yml

+34-27
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,6 @@ git:
1313
branches:
1414
only:
1515
- master
16-
- release-0.12
1716

1817
cache:
1918
directories:
@@ -35,19 +34,19 @@ matrix:
3534
include:
3635
- compiler: "ghc-8.4.3"
3736
# env: TEST=--disable-tests BENCH=--disable-benchmarks
38-
addons: {apt: {packages: [ghc-ppa-tools,ghc-8.4.3], sources: [hvr-ghc]}}
37+
addons: {apt: {packages: [ghc-ppa-tools,cabal-install-2.2,ghc-8.4.3], sources: [hvr-ghc]}}
3938
- compiler: "ghc-8.2.2"
4039
# env: TEST=--disable-tests BENCH=--disable-benchmarks
41-
addons: {apt: {packages: [ghc-ppa-tools,ghc-8.2.2], sources: [hvr-ghc]}}
40+
addons: {apt: {packages: [ghc-ppa-tools,cabal-install-2.2,ghc-8.2.2], sources: [hvr-ghc]}}
4241
- compiler: "ghc-8.0.2"
4342
# env: TEST=--disable-tests BENCH=--disable-benchmarks
44-
addons: {apt: {packages: [ghc-ppa-tools,ghc-8.0.2], sources: [hvr-ghc]}}
43+
addons: {apt: {packages: [ghc-ppa-tools,cabal-install-2.2,ghc-8.0.2], sources: [hvr-ghc]}}
4544
- compiler: "ghc-7.10.3"
4645
# env: TEST=--disable-tests BENCH=--disable-benchmarks
47-
addons: {apt: {packages: [ghc-ppa-tools,ghc-7.10.3], sources: [hvr-ghc]}}
46+
addons: {apt: {packages: [ghc-ppa-tools,cabal-install-2.2,ghc-7.10.3], sources: [hvr-ghc]}}
4847
- compiler: "ghc-7.8.4"
4948
# env: TEST=--disable-tests BENCH=--disable-benchmarks
50-
addons: {apt: {packages: [ghc-ppa-tools,ghc-7.8.4], sources: [hvr-ghc]}}
49+
addons: {apt: {packages: [ghc-ppa-tools,cabal-install-2.2,ghc-7.8.4], sources: [hvr-ghc]}}
5150

5251
before_install:
5352
- HC=${CC}
@@ -59,34 +58,27 @@ before_install:
5958
- HCNUMVER=$(( $(${HC} --numeric-version|sed -E 's/([0-9]+)\.([0-9]+)\.([0-9]+).*/\1 * 10000 + \2 * 100 + \3/') ))
6059
- echo $HCNUMVER
6160

62-
# Let's download "better" cabal
63-
- "curl -L http://oleg.fi/cabal-grayjay-buildable-fix.xz | xz -d > $HOME/.local/bin/cabal"
64-
- |
65-
if [ "$(cd $HOME/.local/bin && sha256sum cabal)" != "e281e9466b8eef30ac0d1371e8ea83c9d2e856bda4714a728ac474138b09b20f cabal" ]; then
66-
rm -f $HOME/.local/bin/cabal;
67-
sha256sum $HOME/.local/bin/cabal;
68-
false;
69-
else
70-
chmod a+x $HOME/.local/bin/cabal;
71-
fi
72-
7361
install:
7462
- cabal --version
7563
- echo "$(${HC} --version) [$(${HC} --print-project-git-commit-id 2> /dev/null || echo '?')]"
7664
- BENCH=${BENCH---enable-benchmarks}
7765
- TEST=${TEST---enable-tests}
7866
- HADDOCK=${HADDOCK-true}
79-
- INSTALLED=${INSTALLED-true}
67+
- UNCONSTRAINED=${UNCONSTRAINED-true}
68+
- NOINSTALLEDCONSTRAINTS=${NOINSTALLEDCONSTRAINTS-false}
8069
- GHCHEAD=${GHCHEAD-false}
8170
- travis_retry cabal update -v
8271
- "sed -i.bak 's/^jobs:/-- jobs:/' ${HOME}/.cabal/config"
8372
- rm -fv cabal.project cabal.project.local
8473
- "if [ $HCNUMVER -ge 70800 ]; then sed -i.bak 's/-- ghc-options:.*/ghc-options: -j2/' ${HOME}/.cabal/config; fi"
8574
- grep -Ev -- '^\s*--' ${HOME}/.cabal/config | grep -Ev '^\s*$'
86-
- "printf 'packages: \"servant\" \"servant-client\" \"servant-client-core\" \"servant-docs\" \"servant-foreign\" \"servant-server\" \"doc/tutorial\" \"doc/cookbook/basic-auth\" \"doc/cookbook/db-postgres-pool\" \"doc/cookbook/db-sqlite-simple\" \"doc/cookbook/https\" \"doc/cookbook/pagination\" \"doc/cookbook/structuring-apis\" \"doc/cookbook/using-custom-monad\"\\n' > cabal.project"
75+
- "printf 'packages: \"servant\" \"servant-client\" \"servant-client-core\" \"servant-docs\" \"servant-foreign\" \"servant-server\" \"doc/tutorial\" \"doc/cookbook/basic-auth\" \"doc/cookbook/db-postgres-pool\" \"doc/cookbook/db-sqlite-simple\" \"doc/cookbook/file-upload\" \"doc/cookbook/https\" \"doc/cookbook/jwt-and-basic-auth\" \"doc/cookbook/pagination\" \"doc/cookbook/structuring-apis\" \"doc/cookbook/using-custom-monad\"\\n' > cabal.project"
8776
- "echo 'constraints: foundation >=0.0.14,memory <0.14.12 || >0.14.12' >> cabal.project"
88-
- "echo 'allow-newer: servant-auth-server:http-types,servant-auth-server:servant-server, http-media:base' >> cabal.project"
89-
- cat cabal.project
77+
- "echo 'allow-newer: servant-auth-server:http-types,servant-auth-server:servant-server, servant-pagination:servant,servant-pagination:servant-server' >> cabal.project"
78+
- touch cabal.project.local
79+
- "if ! $NOINSTALLEDCONSTRAINTS; then for pkg in $($HCPKG list --simple-output); do echo $pkg | sed 's/^/constraints: /' | sed 's/-[^-]*$/ installed/' >> cabal.project.local; done; fi"
80+
- cat cabal.project || true
81+
- cat cabal.project.local || true
9082
- if [ -f "servant/configure.ac" ]; then
9183
(cd "servant" && autoreconf -i);
9284
fi
@@ -117,9 +109,15 @@ install:
117109
- if [ -f "doc/cookbook/db-sqlite-simple/configure.ac" ]; then
118110
(cd "doc/cookbook/db-sqlite-simple" && autoreconf -i);
119111
fi
112+
- if [ -f "doc/cookbook/file-upload/configure.ac" ]; then
113+
(cd "doc/cookbook/file-upload" && autoreconf -i);
114+
fi
120115
- if [ -f "doc/cookbook/https/configure.ac" ]; then
121116
(cd "doc/cookbook/https" && autoreconf -i);
122117
fi
118+
- if [ -f "doc/cookbook/jwt-and-basic-auth/configure.ac" ]; then
119+
(cd "doc/cookbook/jwt-and-basic-auth" && autoreconf -i);
120+
fi
123121
- if [ -f "doc/cookbook/pagination/configure.ac" ]; then
124122
(cd "doc/cookbook/pagination" && autoreconf -i);
125123
fi
@@ -130,7 +128,7 @@ install:
130128
(cd "doc/cookbook/using-custom-monad" && autoreconf -i);
131129
fi
132130
- rm -f cabal.project.freeze
133-
- rm -rf .ghc.environment.* "servant"/dist "servant-client"/dist "servant-client-core"/dist "servant-docs"/dist "servant-foreign"/dist "servant-server"/dist "doc/tutorial"/dist "doc/cookbook/basic-auth"/dist "doc/cookbook/db-postgres-pool"/dist "doc/cookbook/db-sqlite-simple"/dist "doc/cookbook/https"/dist "doc/cookbook/pagination"/dist "doc/cookbook/structuring-apis"/dist "doc/cookbook/using-custom-monad"/dist
131+
- rm -rf .ghc.environment.* "servant"/dist "servant-client"/dist "servant-client-core"/dist "servant-docs"/dist "servant-foreign"/dist "servant-server"/dist "doc/tutorial"/dist "doc/cookbook/basic-auth"/dist "doc/cookbook/db-postgres-pool"/dist "doc/cookbook/db-sqlite-simple"/dist "doc/cookbook/file-upload"/dist "doc/cookbook/https"/dist "doc/cookbook/jwt-and-basic-auth"/dist "doc/cookbook/pagination"/dist "doc/cookbook/structuring-apis"/dist "doc/cookbook/using-custom-monad"/dist
134132
- DISTDIR=$(mktemp -d /tmp/dist-test.XXXX)
135133

136134
# Here starts the actual work to be performed for the package under test;
@@ -148,22 +146,26 @@ script:
148146
- (cd "doc/cookbook/basic-auth" && cabal sdist)
149147
- (cd "doc/cookbook/db-postgres-pool" && cabal sdist)
150148
- (cd "doc/cookbook/db-sqlite-simple" && cabal sdist)
149+
- (cd "doc/cookbook/file-upload" && cabal sdist)
151150
- (cd "doc/cookbook/https" && cabal sdist)
151+
- (cd "doc/cookbook/jwt-and-basic-auth" && cabal sdist)
152152
- (cd "doc/cookbook/pagination" && cabal sdist)
153153
- (cd "doc/cookbook/structuring-apis" && cabal sdist)
154154
- (cd "doc/cookbook/using-custom-monad" && cabal sdist)
155155
- echo -en 'travis_fold:end:sdist\\r'
156156
- echo Unpacking... && echo -en 'travis_fold:start:unpack\\r'
157-
- mv "servant"/dist/servant-*.tar.gz "servant-client"/dist/servant-client-*.tar.gz "servant-client-core"/dist/servant-client-core-*.tar.gz "servant-docs"/dist/servant-docs-*.tar.gz "servant-foreign"/dist/servant-foreign-*.tar.gz "servant-server"/dist/servant-server-*.tar.gz "doc/tutorial"/dist/tutorial-*.tar.gz "doc/cookbook/basic-auth"/dist/cookbook-basic-auth-*.tar.gz "doc/cookbook/db-postgres-pool"/dist/cookbook-db-postgres-pool-*.tar.gz "doc/cookbook/db-sqlite-simple"/dist/cookbook-db-sqlite-simple-*.tar.gz "doc/cookbook/https"/dist/cookbook-https-*.tar.gz "doc/cookbook/pagination"/dist/cookbook-pagination-*.tar.gz "doc/cookbook/structuring-apis"/dist/cookbook-structuring-apis-*.tar.gz "doc/cookbook/using-custom-monad"/dist/cookbook-using-custom-monad-*.tar.gz ${DISTDIR}/
157+
- mv "servant"/dist/servant-*.tar.gz "servant-client"/dist/servant-client-*.tar.gz "servant-client-core"/dist/servant-client-core-*.tar.gz "servant-docs"/dist/servant-docs-*.tar.gz "servant-foreign"/dist/servant-foreign-*.tar.gz "servant-server"/dist/servant-server-*.tar.gz "doc/tutorial"/dist/tutorial-*.tar.gz "doc/cookbook/basic-auth"/dist/cookbook-basic-auth-*.tar.gz "doc/cookbook/db-postgres-pool"/dist/cookbook-db-postgres-pool-*.tar.gz "doc/cookbook/db-sqlite-simple"/dist/cookbook-db-sqlite-simple-*.tar.gz "doc/cookbook/file-upload"/dist/cookbook-file-upload-*.tar.gz "doc/cookbook/https"/dist/cookbook-https-*.tar.gz "doc/cookbook/jwt-and-basic-auth"/dist/cookbook-jwt-and-basic-auth-*.tar.gz "doc/cookbook/pagination"/dist/cookbook-pagination-*.tar.gz "doc/cookbook/structuring-apis"/dist/cookbook-structuring-apis-*.tar.gz "doc/cookbook/using-custom-monad"/dist/cookbook-using-custom-monad-*.tar.gz ${DISTDIR}/
158158
- cd ${DISTDIR} || false
159159
- find . -maxdepth 1 -name '*.tar.gz' -exec tar -xvf '{}' \;
160-
- "printf 'packages: servant-*/*.cabal servant-client-*/*.cabal servant-client-core-*/*.cabal servant-docs-*/*.cabal servant-foreign-*/*.cabal servant-server-*/*.cabal tutorial-*/*.cabal cookbook-basic-auth-*/*.cabal cookbook-db-postgres-pool-*/*.cabal cookbook-db-sqlite-simple-*/*.cabal cookbook-https-*/*.cabal cookbook-pagination-*/*.cabal cookbook-structuring-apis-*/*.cabal cookbook-using-custom-monad-*/*.cabal\\n' > cabal.project"
160+
- "printf 'packages: servant-*/*.cabal servant-client-*/*.cabal servant-client-core-*/*.cabal servant-docs-*/*.cabal servant-foreign-*/*.cabal servant-server-*/*.cabal tutorial-*/*.cabal cookbook-basic-auth-*/*.cabal cookbook-db-postgres-pool-*/*.cabal cookbook-db-sqlite-simple-*/*.cabal cookbook-file-upload-*/*.cabal cookbook-https-*/*.cabal cookbook-jwt-and-basic-auth-*/*.cabal cookbook-pagination-*/*.cabal cookbook-structuring-apis-*/*.cabal cookbook-using-custom-monad-*/*.cabal\\n' > cabal.project"
161161
- "echo 'constraints: foundation >=0.0.14,memory <0.14.12 || >0.14.12' >> cabal.project"
162-
- "echo 'allow-newer: servant-auth-server:http-types,servant-auth-server:servant-server, http-media:base' >> cabal.project"
163-
- cat cabal.project
162+
- "echo 'allow-newer: servant-auth-server:http-types,servant-auth-server:servant-server, servant-pagination:servant,servant-pagination:servant-server' >> cabal.project"
163+
- touch cabal.project.local
164+
- "if ! $NOINSTALLEDCONSTRAINTS; then for pkg in $($HCPKG list --simple-output); do echo $pkg | sed 's/^/constraints: /' | sed 's/-[^-]*$/ installed/' >> cabal.project.local; done; fi"
165+
- cat cabal.project || true
166+
- cat cabal.project.local || true
164167
- echo -en 'travis_fold:end:unpack\\r'
165168

166-
167169
- echo Building with tests and benchmarks... && echo -en 'travis_fold:start:build-everything\\r'
168170
# build & run tests, build benchmarks
169171
- cabal new-build -w ${HC} ${TEST} ${BENCH} all
@@ -176,5 +178,10 @@ script:
176178
- if $HADDOCK; then cabal new-haddock -w ${HC} ${TEST} ${BENCH} all; else echo "Skipping haddock generation";fi
177179

178180
- echo -en 'travis_fold:end:haddock\\r'
181+
- echo Building without installed constraints for packages in global-db... && echo -en 'travis_fold:start:build-installed\\r'
182+
# Build without installed constraints for packages in global-db
183+
- if $UNCONSTRAINED; then rm -f cabal.project.local; echo cabal new-build -w ${HC} --disable-tests --disable-benchmarks all; else echo "Not building without installed constraints"; fi
184+
185+
- echo -en 'travis_fold:end:build-installed\\r'
179186
# REGENDATA ["--config=cabal.make-travis-yml","--output=.travis.yml","cabal.project"]
180187
# EOF

cabal.make-travis-yml

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,11 @@
11
folds: all-but-test
2-
branches: master release-0.12
2+
branches: master
33

44
-- We have inplace packages (servant-js) so we skip installing dependencies in a separate step
55
install-dependencies-step: False
66

77
-- this speed-ups the build a little, but we have to check these for release
88
no-tests-no-benchmarks: False
9-
build-with-installed-step: False
109

1110
-- Don't run cabal check, as cookbook examples won't pass it
1211
cabal-check: False

cabal.project

+5-5
Original file line numberDiff line numberDiff line change
@@ -11,11 +11,10 @@ packages: servant/
1111
doc/cookbook/basic-auth
1212
doc/cookbook/db-postgres-pool
1313
doc/cookbook/db-sqlite-simple
14-
-- MkLink changed
15-
-- doc/cookbook/file-upload
14+
doc/cookbook/file-upload
15+
doc/cookbook/generic
1616
doc/cookbook/https
17-
-- servant-auth-* doesn't support GHC-8.4
18-
-- doc/cookbook/jwt-and-basic-auth
17+
doc/cookbook/jwt-and-basic-auth
1918
doc/cookbook/pagination
2019
doc/cookbook/structuring-apis
2120
doc/cookbook/using-custom-monad
@@ -30,4 +29,5 @@ constraints:
3029
memory <0.14.12 || >0.14.12
3130

3231
allow-newer:
33-
http-media:base
32+
servant-pagination:servant,
33+
servant-pagination:servant-server

doc/cookbook/generic/Generic.lhs

+106
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,106 @@
1+
# Using generics
2+
3+
```haskell
4+
{-# LANGUAGE DataKinds #-}
5+
{-# LANGUAGE DeriveGeneric #-}
6+
{-# LANGUAGE RankNTypes #-}
7+
{-# LANGUAGE TypeOperators #-}
8+
module Main (main, api, getLink, routesLinks, cliGet) where
9+
10+
import Control.Exception (throwIO)
11+
import Data.Proxy (Proxy (..))
12+
import Network.Wai.Handler.Warp (run)
13+
import System.Environment (getArgs)
14+
15+
import Servant
16+
import Servant.Client
17+
18+
import Servant.API.Generic
19+
import Servant.Client.Generic
20+
import Servant.Server.Generic
21+
```
22+
23+
The usage is simple, if you only need a collection of routes.
24+
First you define a record with field types prefixed by a parameter `route`:
25+
26+
```haskell
27+
data Routes route = Routes
28+
{ _get :: route :- Capture "id" Int :> Get '[JSON] String
29+
, _put :: route :- ReqBody '[JSON] Int :> Put '[JSON] Bool
30+
}
31+
deriving (Generic)
32+
```
33+
34+
Then we'll use this data type to define API, links, server and client.
35+
36+
## API
37+
38+
You can get a `Proxy` of the API using `genericApi`:
39+
40+
```haskell
41+
api :: Proxy (ToServantApi Routes)
42+
api = genericApi (Proxy :: Proxy Routes)
43+
```
44+
45+
It's recommented to use `genericApi` function, as then you'll get
46+
better error message, for example if you forget to `derive Generic`.
47+
48+
## Links
49+
50+
The clear advantage of record-based generics approach, is that
51+
we can get safe links very conviently. We don't need to define endpoint types,
52+
as field accessors work as proxies:
53+
54+
```haskell
55+
getLink :: Int -> Link
56+
getLink = fieldLink _get
57+
```
58+
59+
We can also get all links at once, as a record:
60+
61+
```haskell
62+
routesLinks :: Routes (AsLink Link)
63+
routesLinks = allFieldLinks
64+
```
65+
66+
## Client
67+
68+
Even more power starts to show when we generate a record of client functions.
69+
Here we use `genericClientHoist` function, which let us simultaneously
70+
hoist the monad, in this case from `ClientM` to `IO`.
71+
72+
```haskell
73+
cliRoutes :: Routes (AsClientT IO)
74+
cliRoutes = genericClientHoist
75+
(\x -> runClientM x env >>= either throwIO return)
76+
where
77+
env = error "undefined environment"
78+
79+
cliGet :: Int -> IO String
80+
cliGet = _get cliRoutes
81+
```
82+
83+
## Server
84+
85+
Finally, probably the most handy usage: we can convert record of handlers into
86+
the server implementation:
87+
88+
```haskell
89+
record :: Routes AsServer
90+
record = Routes
91+
{ _get = return . show
92+
, _put = return . odd
93+
}
94+
95+
app :: Application
96+
app = genericServe record
97+
98+
main :: IO ()
99+
main = do
100+
args <- getArgs
101+
case args of
102+
("run":_) -> do
103+
putStrLn "Starting cookbook-generic at http://localhost:8000"
104+
run 8000 app
105+
_ -> putStrLn "To run, pass 'run' argument: cabal new-run cookbook-generic run"
106+
```

doc/cookbook/generic/generic.cabal

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
name: cookbook-generic
2+
version: 0.1
3+
synopsis: Using custom monad to pass a state between handlers
4+
homepage: http://haskell-servant.readthedocs.org/
5+
license: BSD3
6+
license-file: ../../../servant/LICENSE
7+
author: Servant Contributors
8+
maintainer: [email protected]
9+
build-type: Simple
10+
cabal-version: >=1.10
11+
tested-with: GHC==7.8.4, GHC==7.10.3, GHC==8.0.2, GHC==8.2.2, GHC==8.4.3
12+
13+
executable cookbook-using-custom-monad
14+
main-is: Generic.lhs
15+
build-depends: base == 4.*
16+
, servant
17+
, servant-client
18+
, servant-client-core
19+
, servant-server
20+
, base-compat
21+
, warp >= 3.2
22+
, transformers >= 0.3
23+
default-language: Haskell2010
24+
ghc-options: -Wall -pgmL markdown-unlit
25+
build-tool-depends: markdown-unlit:markdown-unlit >= 0.4

doc/cookbook/index.rst

+1
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@ you name it!
1818
:maxdepth: 1
1919

2020
structuring-apis/StructuringApis.lhs
21+
generic/Generic.lhs
2122
https/Https.lhs
2223
db-sqlite-simple/DBConnection.lhs
2324
db-postgres-pool/PostgresPool.lhs

doc/index.rst

+11-9
Original file line numberDiff line numberDiff line change
@@ -3,22 +3,24 @@ servant – A Type-Level Web DSL
33

44
.. image:: https://raw.githubusercontent.com/haskell-servant/servant/master/servant.png
55

6-
**servant** is a set of packages for declaring web APIs at the type-level and
7-
then using those API specifications to:
6+
**servant** is a set of Haskell libraries for writing *type-safe* web
7+
applications but also *deriving* clients (in Haskell and other languages) or
8+
generating documentation for them, and more.
89

9-
- write servers (this part of **servant** can be considered a web framework),
10-
- obtain client functions (in haskell),
11-
- generate client functions for other programming languages,
12-
- generate documentation for your web applications
13-
- and more...
10+
This is achieved by taking as input a description of the web API
11+
as a Haskell type. Servant is then able to check that your server-side request
12+
handlers indeed implement your web API faithfully, or to automatically derive
13+
Haskell functions that can hit a web application that implements this API,
14+
generate a Swagger description or code for client functions in some other
15+
languages directly.
1416

15-
All in a type-safe manner.
17+
If you would like to learn more, click the tutorial link below.
1618

1719
.. toctree::
1820
:maxdepth: 2
1921

20-
introduction.rst
2122
tutorial/index.rst
2223
cookbook/index.rst
2324
examples.md
2425
links.rst
26+
principles.rst

doc/introduction.rst doc/principles.rst

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
1-
Introduction
2-
------------
1+
Principles
2+
----------
33

44
**servant** has the following guiding principles:
55

doc/tutorial/Javascript.lhs

+1-2
Original file line numberDiff line numberDiff line change
@@ -477,7 +477,7 @@ data AngularOptions = AngularOptions
477477
}
478478
```
479479
480-
# Custom function name builder
480+
## Custom function name builder
481481
482482
Servant comes with three name builders included:
483483
@@ -518,4 +518,3 @@ var get_books = function(q, onSuccess, onError)
518518
}
519519
520520
```
521-

0 commit comments

Comments
 (0)