forked from snoyberg/http-client
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathintegration-test.hs
executable file
·177 lines (160 loc) · 6.44 KB
/
integration-test.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
#!/usr/bin/env stack
{- stack --install-ghc exec
--package typed-process
--package http-conduit
--package hspec
--package safe-exceptions
-- ghc -O2 -threaded -Wall -Werror
-}
{-# LANGUAGE OverloadedStrings #-}
import Control.Exception.Safe
import Control.Monad (forM_, void)
import qualified Data.ByteString as B
import Network.HTTP.Client (HttpExceptionContent (..))
import Network.HTTP.Client.TLS (setGlobalManager, newTlsManager)
import Network.HTTP.Simple
import System.Directory (copyFile, createDirectoryIfMissing)
import System.Environment as E
import System.FilePath (takeDirectory, (</>))
import System.Process.Typed
import Test.Hspec
-- | Main entry point. No arguments: launch the wrapper which will kick
-- off the Docker containers. If given inner, runs the test suite
-- inside the Docker container.
main :: IO ()
main = do
args <- getArgs
case args of
[] -> outer
-- withArgs hides the "inner" argument to avoid confusing hspec
["inner"] -> withArgs [] inner
_ -> error $ "Invalid args: " ++ show args
-- | Run from outside the container
outer :: IO ()
outer = do
-- Kill a previously launched squid and ssh containers. We
-- ignore any errors in case the container isn't running.
let killit = do
let run = void
. runProcess
. setStdin closed
. setStdout closed
. setStderr closed
run $ proc "docker" $ words "kill http-client-squid"
run $ proc "docker" $ words "rm http-client-squid"
run $ proc "docker" $ words "kill http-client-ssh"
run $ proc "docker" $ words "rm http-client-ssh"
-- How to launch the Squid proxy container
squid = proc "docker"
$ words "run -d --name=http-client-squid sameersbn/squid:3.3.8-23"
-- How to launch the SSH container
ssh = proc "docker"
$ words "run -d --name=http-client-ssh http-client-base /usr/sbin/sshd -D"
-- Create the Docker image to run the test suite itself, containing
-- this executable. NOTE: I tried initially just bind-mounting the
-- executable into the running container, but this failed on Gitlab
-- which is using Docker-within-Docker.
let dockerfile = "/tmp/http-client-base/Dockerfile"
dockerfileBS =
"FROM fpco/pid1:16.04\n\
\RUN apt-get update\n\
\RUN apt-get install -y iptables curl ca-certificates netbase openssh-server\n\
\RUN ssh-keygen -t rsa -b 4096 -C test@example.com -f /root/.ssh/id_rsa\n\
\RUN cp /root/.ssh/id_rsa.pub /root/.ssh/authorized_keys\n\
\RUN mkdir -p /var/run/sshd\n\
\COPY test-suite /usr/bin/test-suite\n"
-- Check if the Dockerfile is out of date. We want to avoid updating
-- if we can to not invalidate the Docker cache.
ecurrent <- tryIO $ B.readFile dockerfile
case ecurrent of
Right current | current == dockerfileBS -> return ()
_ -> do
createDirectoryIfMissing True $ takeDirectory dockerfile
B.writeFile dockerfile dockerfileBS
-- Copy over the executable to the temporary directory
exeName <- getExecutablePath
copyFile exeName (takeDirectory dockerfile </> "test-suite")
-- Now build the image
runProcess_ $ proc "docker"
[ "build"
, "--tag"
, "http-client-base"
, takeDirectory dockerfile
]
-- Launch the squid container, killing a preexisting one if it
-- exists, and then kill the container when we're done.
bracket_ (killit >> readProcess_ squid >> readProcess_ ssh) killit $ do
-- Run the test suite itself inside a Docker container. We need
-- --privileged to modify the iptables rules.
runProcess_ $ proc "docker" $ words
"run --rm -t --link http-client-squid:squid --link http-client-ssh:ssh\n\
\--privileged http-client-base /usr/bin/test-suite inner"
-- | Run a set of tests with the given environment variable key/value
-- pair set. Reload the global HTTP manager each time.
describe' :: String -- ^ key
-> String -- ^ value
-> Spec
-> Spec
describe' key val =
describe name . around_ (bracket set unset . const)
where
name = concat [key, "=", val]
set = do
morig <- lookupEnv key
E.setEnv key val
man <- newTlsManager
setGlobalManager man
return morig
unset mval = do
case mval of
Nothing -> unsetEnv key
Just val' -> E.setEnv key val'
setGlobalManager $ error "should not be used"
-- | Base URL for HTTP or HTTPS httpbin site
base :: Bool -> String
base False = "http://httpbin.org/"
base True = "https://httpbin.org/"
-- | Ensure that connections fail.
fails :: Bool -- ^ use HTTPS?
-> Spec
fails https = it ("fails to connect " ++ (if https then "secure" else "insecure")) $
httpLBS (parseRequest_ (base https)) `shouldThrow` (\e ->
case e of
HttpExceptionRequest _ (ConnectionFailure _) -> True
HttpExceptionRequest _ (InternalException _) -> True
_ -> False)
-- | Ensure that connections succeed.
succeeds :: Bool -- ^ use HTTPS?
-> Spec
succeeds https = describe ("succeeds " ++ (if https then "secure" else "insecure")) $ do
describe "basic status code check" $ forM_ [200, 400, 500] $ \code -> it (show code) $ do
res <- httpLBS (parseRequest_ (base https ++ "status/" ++ show code))
getResponseStatusCode res `shouldBe` code
-- | Code to run the test suites themselves inside the Docker container.
inner :: IO ()
inner = do
-- Block all outgoing connections to everything except squid and ssh
runProcess_ $ proc "iptables" $ words "-A OUTPUT -p tcp -d squid -j ACCEPT"
runProcess_ $ proc "iptables" $ words "-A OUTPUT -p tcp -d ssh -j ACCEPT"
runProcess_ $ proc "iptables" $ words "-A OUTPUT -p tcp -d 127.0.0.1 -j ACCEPT"
runProcess_ $ proc "iptables" $ words "-A OUTPUT -p tcp -j REJECT"
-- Establish an SSH connection with a SOCKS proxy
void $ runProcess_ $ proc "ssh" $ words "-D 127.0.0.1:1080 -N ssh -f -o StrictHostKeyChecking=no"
-- Run the test suite
hspec $ do
describe "no proxy" $ do
fails False
fails True
let values =
[ "http://squid:3128"
, "squid:3128"
, "socks5://127.0.0.1:1080"
, "socks5h://127.0.0.1:1080"
]
forM_ values $ \value -> do
describe' "http_proxy" value $ do
succeeds False
fails True
describe' "https_proxy" value $ do
succeeds True
fails False