File: TestSuite.hs

package info (click to toggle)
haskell-openssl-streams 1.2.3.0-3
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 104 kB
  • sloc: haskell: 186; makefile: 4
file content (120 lines) | stat: -rw-r--r-- 4,116 bytes parent folder | download | duplicates (3)
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
{-# LANGUAGE CPP               #-}
{-# LANGUAGE OverloadedStrings #-}
module Main where

------------------------------------------------------------------------------
import           Control.Concurrent
import           Data.ByteString.Char8          ()
import qualified Network.Socket                 as N
import qualified OpenSSL                        as SSL
import qualified OpenSSL.Session                as SSL
import qualified System.IO.Streams              as Streams
import qualified System.IO.Streams.SSL          as SSLStreams
import           System.Timeout                 (timeout)
import           Test.Framework
import           Test.Framework.Providers.HUnit
import           Test.HUnit                     hiding (Test)
------------------------------------------------------------------------------


main :: IO ()
main = N.withSocketsDo
         $ SSL.withOpenSSL
         $ defaultMain [sanityCheck, withConnection]

sClose :: N.Socket -> IO ()
#if MIN_VERSION_network(2,7,0)
sClose = N.close
#else
sClose = N.sClose
#endif

------------------------------------------------------------------------------
sanityCheck :: Test
sanityCheck = testCase "sanitycheck" $ do
    ctx1 <- setupContext
    ctx2 <- setupContext
    x <- timeout (10 * 10^(6::Int)) $ go ctx1 ctx2
    assertEqual "ok" (Just ()) x

  where
    go ctx1 ctx2 = do
        portMVar   <- newEmptyMVar
        resultMVar <- newEmptyMVar
        forkIO $ client ctx1 portMVar resultMVar
        server ctx2 portMVar
        l <- takeMVar resultMVar
        assertEqual "testSocket" l ["ok"]


    client ctx mvar resultMVar = do
        port <- takeMVar mvar
        (is, os, ssl) <- SSLStreams.connect ctx "127.0.0.1" port
        Streams.fromList ["ok"] >>= Streams.connectTo os
        SSL.shutdown ssl SSL.Unidirectional
        Streams.toList is >>= putMVar resultMVar
        maybe (return ()) sClose $ SSL.sslSocket ssl


------------------------------------------------------------------------------
withConnection :: Test
withConnection = testCase "withConnection" $ do
    ctx1 <- setupContext
    ctx2 <- setupContext
    x <- timeout (10 * 10^(6::Int)) $ go ctx1 ctx2
    assertEqual "ok" (Just ()) x

  where
    go ctx1 ctx2 = do
        portMVar   <- newEmptyMVar
        resultMVar <- newEmptyMVar
        forkIO $ client ctx1 portMVar resultMVar
        server ctx2 portMVar
        l <- takeMVar resultMVar
        assertEqual "testSocket" l ["ok"]

    client ctx mvar resultMVar = do
        port <- takeMVar mvar
        SSLStreams.withConnection ctx "127.0.0.1" port $ \is os ssl -> do
            Streams.fromList ["ok"] >>= Streams.connectTo os
            SSL.shutdown ssl SSL.Unidirectional
            Streams.toList is >>= putMVar resultMVar
            maybe (return ()) sClose $ SSL.sslSocket ssl

------------------------------------------------------------------------------
server :: SSL.SSLContext -> MVar N.PortNumber -> IO ()
server ctx mvar = do
    let hint = N.defaultHints {
                    N.addrFamily     = N.AF_INET,
                    N.addrSocketType = N.Stream,
                    N.addrFlags      = [N.AI_NUMERICHOST]
                  }
    xs <- N.getAddrInfo (Just hint) (Just "127.0.0.1") Nothing
    let [addr] = xs
    sock  <- N.socket N.AF_INET N.Stream N.defaultProtocol
    let saddr = N.addrAddress addr
    N.bind sock saddr
    N.listen sock 5
    port  <- N.socketPort sock
    putMVar mvar port
    (csock, _) <- N.accept sock
    ssl <- SSL.connection ctx csock
    SSL.accept ssl
    (is, os) <- SSLStreams.sslToStreams ssl
    Streams.toList is >>= flip Streams.writeList os
    SSL.shutdown ssl SSL.Unidirectional
    maybe (return ()) N.close $ SSL.sslSocket ssl
    N.close sock


setupContext :: IO SSL.SSLContext
setupContext = do
    ctx <- SSL.context
    SSL.contextSetDefaultCiphers ctx
    SSL.contextSetCertificateFile ctx "test/cert.pem"
    SSL.contextSetPrivateKeyFile ctx "test/key.pem"
    SSL.contextSetVerificationMode ctx SSL.VerifyNone

    certOK <- SSL.contextCheckPrivateKey ctx
    assertBool "private key is bad" certOK
    return ctx