aboutsummaryrefslogtreecommitdiff
path: root/src/AdListener.hs
blob: 3db4c6a763697f2b5a2021dbd372049fc5bd8a96 (plain)
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
{-# LANGUAGE OverloadedStrings #-}

module AdListener
  ( start
  ) where

import Prelude hiding (error)

import Data.Text (Text)
import qualified Data.Text.IO as T
import qualified Data.Text.Lazy as LT
import Data.Text.Lazy.Builder (toLazyText, fromText)

import Control.Concurrent (threadDelay)

import Fetch (fetchResumes, fetchAds)

import Model.Ad
import Model.URL
import Model.Resume

import qualified View.Plain.Ad as P
import qualified View.Html.Ad as H

import Mail
import Model.Mail (Mail(Mail))

import Conf (Conf)
import qualified Conf

import Time (getCurrentFormattedTime)

start :: Conf -> IO ()
start conf = do
  eitherResumes <- fetchResumes (Conf.url conf)
  case eitherResumes of
    Left error ->
      showErrorAndListenBack conf [] error
    Right resumes -> do
      let newURLs = map url resumes
      T.putStrLn "Listening to new ads…"
      waitListenInterval conf
      listenToNewAdsWithViewedURLs conf newURLs

listenToNewAdsWithViewedURLs :: Conf -> [URL] -> IO ()
listenToNewAdsWithViewedURLs conf viewedURLs = do
  eitherResumes <- fetchResumes (Conf.url conf)
  case eitherResumes of
    Left error ->
      showErrorAndListenBack conf viewedURLs error
    Right resumes ->
      listenToNewAdsWithResumes conf viewedURLs resumes

listenToNewAdsWithResumes :: Conf -> [URL] -> [Resume] -> IO ()
listenToNewAdsWithResumes conf viewedURLs resumes =
  let (newURLs, newResumes) = getNewResumes viewedURLs resumes
  in  do
        eitherNewAds <- fetchAds newResumes
        case eitherNewAds of
          Left error ->
            showErrorAndListenBack conf viewedURLs error
          Right newAds -> do
            time <- getCurrentFormattedTime
            if not (null newAds)
              then
                let message = P.renderConsoleAds conf time newAds
                in  do
                      T.putStrLn message
                      trySendMail conf newAds
              else
                return ()
            waitListenInterval conf
            listenToNewAdsWithViewedURLs conf (viewedURLs ++ newURLs)

trySendMail :: Conf -> [Ad] -> IO ()
trySendMail conf ads =
  let (title, plainBody) = P.renderAds conf ads
      htmlBody = H.renderAds conf ads
      mail = Mail (Conf.mailFrom conf) (Conf.mailTo conf) title (strictToLazy plainBody) (strictToLazy htmlBody)
  in  Mail.send mail >> return ()

strictToLazy :: Text -> LT.Text
strictToLazy = toLazyText . fromText

showErrorAndListenBack :: Conf -> [URL] -> Text -> IO ()
showErrorAndListenBack conf viewedURLs error = do
  T.putStrLn error
  waitListenInterval conf
  listenToNewAdsWithViewedURLs conf viewedURLs

waitListenInterval :: Conf -> IO ()
waitListenInterval = threadDelay . (*) 1000000 . round . Conf.listenInterval