forked from simonmar/parconc-examples
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathserver2.hs
74 lines (66 loc) · 2.01 KB
/
server2.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
import Network
import Control.Monad
import Control.Concurrent
import System.IO
import Text.Printf
import Control.Exception
import Control.Concurrent.Async
import Control.Concurrent.STM
import ConcurrentUtils (forkFinally)
-- <<main
main = withSocketsDo $ do
sock <- listenOn (PortNumber (fromIntegral port))
printf "Listening on port %d\n" port
factor <- atomically $ newTVar 2 -- <1>
forever $ do
(handle, host, port) <- accept sock
printf "Accepted connection from %s: %s\n" host (show port)
forkFinally (talk handle factor) (\_ -> hClose handle) -- <2>
port :: Int
port = 44444
-- >>
-- <<talk
talk :: Handle -> TVar Integer -> IO ()
talk h factor = do
hSetBuffering h LineBuffering
c <- atomically newTChan -- <1>
race (server h factor c) (receive h c) -- <2>
return ()
-- >>
-- <<receive
receive :: Handle -> TChan String -> IO ()
receive h c = forever $ do
line <- hGetLine h
atomically $ writeTChan c line
-- >>
-- <<server
server :: Handle -> TVar Integer -> TChan String -> IO ()
server h factor c = do
f <- atomically $ readTVar factor -- <1>
hPrintf h "Current factor: %d\n" f -- <2>
loop f -- <3>
where
loop f = do
action <- atomically $ do -- <4>
f' <- readTVar factor -- <5>
if (f /= f') -- <6>
then return (newfactor f') -- <7>
else do
l <- readTChan c -- <8>
return (command f l) -- <9>
action
newfactor f = do -- <10>
hPrintf h "new factor: %d\n" f
loop f
command f s -- <11>
= case s of
"end" ->
hPutStrLn h ("Thank you for using the " ++
"Haskell doubling service.") -- <12>
'*':s -> do
atomically $ writeTVar factor (read s :: Integer) -- <13>
loop f
line -> do
hPutStrLn h (show (f * (read line :: Integer)))
loop f
-- >>